diff --git a/.bumpversion.cfg b/.bumpversion.cfg index cb713915504..0cfbd91b50b 100644 --- a/.bumpversion.cfg +++ b/.bumpversion.cfg @@ -1,5 +1,5 @@ [bumpversion] -current_version = 1.9.0a1 +current_version = 1.9.0b3 parse = (?P[\d]+) # major version number \.(?P[\d]+) # minor version number \.(?P[\d]+) # patch version number diff --git a/.changes/1.9.0-b1.md b/.changes/1.9.0-b1.md new file mode 100644 index 00000000000..185338db39c --- /dev/null +++ b/.changes/1.9.0-b1.md @@ -0,0 +1,136 @@ +## dbt-core 1.9.0-b1 - October 01, 2024 + +### Breaking Changes + +- Fix changing the current working directory when using dpt deps, clean and init. ([#8997](https://github.com/dbt-labs/dbt-core/issues/8997)) + +### Features + +- serialize inferred primary key ([#9824](https://github.com/dbt-labs/dbt-core/issues/9824)) +- Add unit_test: selection method ([#10053](https://github.com/dbt-labs/dbt-core/issues/10053)) +- Maximally parallelize dbt clone in clone command" ([#7914](https://github.com/dbt-labs/dbt-core/issues/7914)) +- Add --host flag to dbt docs serve, defaulting to '127.0.0.1' ([#10229](https://github.com/dbt-labs/dbt-core/issues/10229)) +- Update data_test to accept arbitrary config options ([#10197](https://github.com/dbt-labs/dbt-core/issues/10197)) +- add pre_model and post_model hook calls to data and unit tests to be able to provide extra config options ([#10198](https://github.com/dbt-labs/dbt-core/issues/10198)) +- add --empty value to jinja context as flags.EMPTY ([#10317](https://github.com/dbt-labs/dbt-core/issues/10317)) +- Warning message for snapshot timestamp data types ([#10234](https://github.com/dbt-labs/dbt-core/issues/10234)) +- Support cumulative_type_params & sub-daily granularities in semantic manifest. ([#10360](https://github.com/dbt-labs/dbt-core/issues/10360)) +- Add time_granularity to metric spec. ([#10376](https://github.com/dbt-labs/dbt-core/issues/10376)) +- Support standard schema/database fields for snapshots ([#10301](https://github.com/dbt-labs/dbt-core/issues/10301)) +- Support ref and source in foreign key constraint expressions, bump dbt-common minimum to 1.6 ([#8062](https://github.com/dbt-labs/dbt-core/issues/8062)) +- Support new semantic layer time spine configs to enable sub-daily granularity. ([#10475](https://github.com/dbt-labs/dbt-core/issues/10475)) +- Include models that depend on changed vars in state:modified, add state:modified.vars selection method ([#4304](https://github.com/dbt-labs/dbt-core/issues/4304)) +- Add support for behavior flags ([#10618](https://github.com/dbt-labs/dbt-core/issues/10618)) +- Enable `--resource-type` and `--exclude-resource-type` CLI flags and environment variables for `dbt test` ([#10656](https://github.com/dbt-labs/dbt-core/issues/10656)) +- Allow configuring snapshot column names ([#10185](https://github.com/dbt-labs/dbt-core/issues/10185)) +- Add custom_granularities to YAML spec for time spines. ([#9265](https://github.com/dbt-labs/dbt-core/issues/9265)) +- Add basic functionality for creating microbatch incremental models ([#9490](https://github.com/dbt-labs/dbt-core/issues/9490), [#10635](https://github.com/dbt-labs/dbt-core/issues/10635), [#10637](https://github.com/dbt-labs/dbt-core/issues/10637), [#10638](https://github.com/dbt-labs/dbt-core/issues/10638), [#10636](https://github.com/dbt-labs/dbt-core/issues/10636), [#10662](https://github.com/dbt-labs/dbt-core/issues/10662), [#10639](https://github.com/dbt-labs/dbt-core/issues/10639)) +- Execute microbatch models in batches ([#10700](https://github.com/dbt-labs/dbt-core/issues/10700)) +- Create 'skip_nodes_if_on_run_start_fails' behavior change flag ([#7387](https://github.com/dbt-labs/dbt-core/issues/7387)) +- Allow snapshots to be defined in YAML. ([#10246](https://github.com/dbt-labs/dbt-core/issues/10246)) +- Write microbatch compiled/run targets to separate files, one per batch ([#10714](https://github.com/dbt-labs/dbt-core/issues/10714)) +- Track incremental_strategy as part of model_run tracking event ([#10761](https://github.com/dbt-labs/dbt-core/issues/10761)) +- Support required 'begin' config for microbatch models ([#10701](https://github.com/dbt-labs/dbt-core/issues/10701)) +- Parse-time validation of microbatch configs: require event_time, batch_size, lookback and validate input event_time ([#10709](https://github.com/dbt-labs/dbt-core/issues/10709)) +- Added the --inline-direct parameter to 'dbt show' ([#10770](https://github.com/dbt-labs/dbt-core/issues/10770)) +- Enable `retry` support for microbatch models ([#10715](https://github.com/dbt-labs/dbt-core/issues/10715), [#10729](https://github.com/dbt-labs/dbt-core/issues/10729)) +- Use unrendered database and schema source properties during state:modified, behind state_modified_compare_more_unrendered_values behavoiur flag ([#9573](https://github.com/dbt-labs/dbt-core/issues/9573)) +- Ensure microbatch models respect `full_refresh` model config ([#10785](https://github.com/dbt-labs/dbt-core/issues/10785)) +- Adds validations for custom_granularities to ensure unique naming. ([#9265](https://github.com/dbt-labs/dbt-core/issues/9265)) + +### Fixes + +- Remove unused check_new method ([#7586](https://github.com/dbt-labs/dbt-core/issues/7586)) +- Test case for `merge_exclude_columns` ([#8267](https://github.com/dbt-labs/dbt-core/issues/8267)) +- Convert "Skipping model due to fail_fast" message to DEBUG level ([#8774](https://github.com/dbt-labs/dbt-core/issues/8774)) +- Restore previous behavior for --favor-state: only favor defer_relation if not selected in current command" ([#10107](https://github.com/dbt-labs/dbt-core/issues/10107)) +- Unit test fixture (csv) returns null for empty value ([#9881](https://github.com/dbt-labs/dbt-core/issues/9881)) +- Fix json format log and --quiet for ls and jinja print by converting print call to fire events ([#8756](https://github.com/dbt-labs/dbt-core/issues/8756)) +- Add resource type to saved_query ([#10168](https://github.com/dbt-labs/dbt-core/issues/10168)) +- Fix: Order-insensitive unit test equality assertion for expected/actual with multiple nulls ([#10167](https://github.com/dbt-labs/dbt-core/issues/10167)) +- Renaming or removing a contracted model should raise a BreakingChange warning/error ([#10116](https://github.com/dbt-labs/dbt-core/issues/10116)) +- prefer disabled project nodes to external node ([#10224](https://github.com/dbt-labs/dbt-core/issues/10224)) +- Fix issues with selectors and inline nodes ([#8943](https://github.com/dbt-labs/dbt-core/issues/8943), [#9269](https://github.com/dbt-labs/dbt-core/issues/9269)) +- Fix snapshot config to work in yaml files ([#4000](https://github.com/dbt-labs/dbt-core/issues/4000)) +- Improve handling of error when loading schema file list ([#10284](https://github.com/dbt-labs/dbt-core/issues/10284)) +- Use model alias for the CTE identifier generated during ephemeral materialization ([#5273](https://github.com/dbt-labs/dbt-core/issues/5273)) +- Implement state:modified for saved queries ([#10294](https://github.com/dbt-labs/dbt-core/issues/10294)) +- Saved Query node fail during skip ([#10029](https://github.com/dbt-labs/dbt-core/issues/10029)) +- DOn't warn on `unit_test` config paths that are properly used ([#10311](https://github.com/dbt-labs/dbt-core/issues/10311)) +- Fix setting `silence` of `warn_error_options` via `dbt_project.yaml` flags ([#10160](https://github.com/dbt-labs/dbt-core/issues/10160)) +- Attempt to provide test fixture tables with all values to set types correctly for comparisong with source tables ([#10365](https://github.com/dbt-labs/dbt-core/issues/10365)) +- Limit data_tests deprecation to root_project ([#9835](https://github.com/dbt-labs/dbt-core/issues/9835)) +- CLI flags should take precedence over env var flags ([#10304](https://github.com/dbt-labs/dbt-core/issues/10304)) +- Fix typing for artifact schemas ([#10442](https://github.com/dbt-labs/dbt-core/issues/10442)) +- Fix over deletion of generated_metrics in partial parsing ([#10450](https://github.com/dbt-labs/dbt-core/issues/10450)) +- Fix error constructing warn_error_options ([#10452](https://github.com/dbt-labs/dbt-core/issues/10452)) +- Do not update varchar column definitions if a contract exists ([#10362](https://github.com/dbt-labs/dbt-core/issues/10362)) +- fix all_constraints access, disabled node parsing of non-uniquely named resources ([#10509](https://github.com/dbt-labs/dbt-core/issues/10509)) +- respect --quiet and --warn-error-options for flag deprecations ([#10105](https://github.com/dbt-labs/dbt-core/issues/10105)) +- Propagate measure label when using create_metrics ([#10536](https://github.com/dbt-labs/dbt-core/issues/10536)) +- Fix state:modified check for exports ([#10138](https://github.com/dbt-labs/dbt-core/issues/10138)) +- Filter out empty nodes after graph selection to support consistent selection of nodes that depend on upstream public models ([#8987](https://github.com/dbt-labs/dbt-core/issues/8987)) +- Late render pre- and post-hooks configs in properties / schema YAML files ([#10603](https://github.com/dbt-labs/dbt-core/issues/10603)) +- Allow the use of env_var function in certain macros in which it was previously unavailable. ([#10609](https://github.com/dbt-labs/dbt-core/issues/10609)) +- Remove deprecation for tests: to data_tests: change ([#10564](https://github.com/dbt-labs/dbt-core/issues/10564)) +- Fix `--resource-type test` for `dbt list` and `dbt build` ([#10730](https://github.com/dbt-labs/dbt-core/issues/10730)) +- Fix unit tests for incremental model with alias ([#10754](https://github.com/dbt-labs/dbt-core/issues/10754)) +- Allow singular tests to be documented in properties.yml ([#9005](https://github.com/dbt-labs/dbt-core/issues/9005)) +- Ignore --empty in unit test ref/source rendering ([#10516](https://github.com/dbt-labs/dbt-core/issues/10516)) +- Ignore rendered jinja in configs for state:modified, behind state_modified_compare_more_unrendered_values behaviour flag ([#9564](https://github.com/dbt-labs/dbt-core/issues/9564)) +- Improve performance of infer primary key ([#10781](https://github.com/dbt-labs/dbt-core/issues/10781)) +- Attempt to skip saved query processing when no semantic manifest changes ([#10563](https://github.com/dbt-labs/dbt-core/issues/10563)) +- Ensure dbt retry of microbatch models doesn't lose prior successful state ([#10800](https://github.com/dbt-labs/dbt-core/issues/10800)) + +### Docs + +- Enable display of unit tests ([dbt-docs/#501](https://github.com/dbt-labs/dbt-docs/issues/501)) +- Unit tests not rendering ([dbt-docs/#506](https://github.com/dbt-labs/dbt-docs/issues/506)) +- Add support for Saved Query node ([dbt-docs/#486](https://github.com/dbt-labs/dbt-docs/issues/486)) +- Fix npm security vulnerabilities as of June 2024 ([dbt-docs/#513](https://github.com/dbt-labs/dbt-docs/issues/513)) + +### Under the Hood + +- Clear error message for Private package in dbt-core ([#10083](https://github.com/dbt-labs/dbt-core/issues/10083)) +- Enable use of context in serialization ([#10093](https://github.com/dbt-labs/dbt-core/issues/10093)) +- Make RSS high water mark measurement more accurate on Linux ([#10177](https://github.com/dbt-labs/dbt-core/issues/10177)) +- Enable record filtering by type. ([#10240](https://github.com/dbt-labs/dbt-core/issues/10240)) +- Remove IntermediateSnapshotNode ([#10326](https://github.com/dbt-labs/dbt-core/issues/10326)) +- Additional logging for skipped ephemeral models ([#10389](https://github.com/dbt-labs/dbt-core/issues/10389)) +- bump black to 24.3.0 ([#10454](https://github.com/dbt-labs/dbt-core/issues/10454)) +- generate protos with protoc version 5.26.1 ([#10457](https://github.com/dbt-labs/dbt-core/issues/10457)) +- Move from minimal-snowplow-tracker fork back to snowplow-tracker ([#8409](https://github.com/dbt-labs/dbt-core/issues/8409)) +- Add group info to RunResultError, RunResultFailure, RunResultWarning log lines ([#](https://github.com/dbt-labs/dbt-core/issues/)) +- Improve speed of tree traversal when finding children, increasing build speed for some selectors ([#10434](https://github.com/dbt-labs/dbt-core/issues/10434)) +- Add test for sources tables with quotes ([#10582](https://github.com/dbt-labs/dbt-core/issues/10582)) +- Additional type hints for `core/dbt/version.py` ([#10612](https://github.com/dbt-labs/dbt-core/issues/10612)) +- Fix typing issues in core/dbt/contracts/sql.py ([#10614](https://github.com/dbt-labs/dbt-core/issues/10614)) +- Fix type errors in `dbt/core/task/clean.py` ([#10616](https://github.com/dbt-labs/dbt-core/issues/10616)) +- Add Snowplow tracking for behavior flag deprecations ([#10552](https://github.com/dbt-labs/dbt-core/issues/10552)) +- Add test utility patch_microbatch_end_time for adapters testing ([#10713](https://github.com/dbt-labs/dbt-core/issues/10713)) +- Replace `TestSelector` with `ResourceTypeSelector` ([#10718](https://github.com/dbt-labs/dbt-core/issues/10718)) +- Standardize returning `ResourceTypeSelector` instances in `dbt list` and `dbt build` ([#10739](https://github.com/dbt-labs/dbt-core/issues/10739)) +- Add group metadata info to LogModelResult and LogTestResult ([#10775](https://github.com/dbt-labs/dbt-core/issues/10775)) + +### Dependencies + +- Remove logbook dependency ([#8027](https://github.com/dbt-labs/dbt-core/issues/8027)) +- Increase supported version range for dbt-semantic-interfaces. Needed to support custom calendar features. ([#9265](https://github.com/dbt-labs/dbt-core/issues/9265)) + +### Security + +- Explicitly bind to localhost in docs serve ([#10209](https://github.com/dbt-labs/dbt-core/issues/10209)) + +### Contributors +- [@McKnight-42](https://github.com/McKnight-42) ([#10197](https://github.com/dbt-labs/dbt-core/issues/10197), [#10198](https://github.com/dbt-labs/dbt-core/issues/10198)) +- [@TowardOliver](https://github.com/TowardOliver) ([#10656](https://github.com/dbt-labs/dbt-core/issues/10656)) +- [@aliceliu](https://github.com/aliceliu) ([#10536](https://github.com/dbt-labs/dbt-core/issues/10536), [#10138](https://github.com/dbt-labs/dbt-core/issues/10138)) +- [@courtneyholcomb](https://github.com/courtneyholcomb) ([#10360](https://github.com/dbt-labs/dbt-core/issues/10360), [#10376](https://github.com/dbt-labs/dbt-core/issues/10376), [#10475](https://github.com/dbt-labs/dbt-core/issues/10475), [#9265](https://github.com/dbt-labs/dbt-core/issues/9265), [#9265](https://github.com/dbt-labs/dbt-core/issues/9265), [#9265](https://github.com/dbt-labs/dbt-core/issues/9265)) +- [@dave-connors-3](https://github.com/dave-connors-3) ([#9824](https://github.com/dbt-labs/dbt-core/issues/9824)) +- [@jeancochrane](https://github.com/jeancochrane) ([#5273](https://github.com/dbt-labs/dbt-core/issues/5273)) +- [@katsugeneration](https://github.com/katsugeneration) ([#10754](https://github.com/dbt-labs/dbt-core/issues/10754)) +- [@kevinneville](https://github.com/kevinneville) ([#7586](https://github.com/dbt-labs/dbt-core/issues/7586)) +- [@nakamichiworks](https://github.com/nakamichiworks) ([#10442](https://github.com/dbt-labs/dbt-core/issues/10442)) +- [@rariyama](https://github.com/rariyama) ([#8997](https://github.com/dbt-labs/dbt-core/issues/8997)) +- [@scottgigante,nevdelap](https://github.com/scottgigante,nevdelap) ([#8774](https://github.com/dbt-labs/dbt-core/issues/8774)) +- [@ttusing](https://github.com/ttusing) ([#10434](https://github.com/dbt-labs/dbt-core/issues/10434)) diff --git a/.changes/1.9.0-b2.md b/.changes/1.9.0-b2.md new file mode 100644 index 00000000000..5df2eac492c --- /dev/null +++ b/.changes/1.9.0-b2.md @@ -0,0 +1 @@ +## dbt-core 1.9.0-b2 - October 07, 2024 diff --git a/.changes/1.9.0-b3.md b/.changes/1.9.0-b3.md new file mode 100644 index 00000000000..b645db330e6 --- /dev/null +++ b/.changes/1.9.0-b3.md @@ -0,0 +1,38 @@ +## dbt-core 1.9.0-b3 - October 30, 2024 + +### Features + +- Add `order_by` and `limit` fields to saved queries. ([#10531](https://github.com/dbt-labs/dbt-core/issues/10531)) +- Enable specification of dbt_valid_to for current records ([#10187](https://github.com/dbt-labs/dbt-core/issues/10187)) +- Enable use of multi-column unique key in snapshots ([#9992](https://github.com/dbt-labs/dbt-core/issues/9992)) +- Ensure `--event-time-start` is before `--event-time-end` ([#10786](https://github.com/dbt-labs/dbt-core/issues/10786)) +- Ensure microbatch models use same `current_time` value ([#10819](https://github.com/dbt-labs/dbt-core/issues/10819)) +- Emit warning when microbatch model has no input with `event_time` config ([#10926](https://github.com/dbt-labs/dbt-core/issues/10926)) + +### Fixes + +- Pass test user config to adapter pre_hook by explicitly adding test builder config to node ([#10484](https://github.com/dbt-labs/dbt-core/issues/10484)) +- Handle edge cases when a specified `--event-time-end` is equivalent to the batch size truncated batch start time ([#10824](https://github.com/dbt-labs/dbt-core/issues/10824)) +- Begin tracking execution time of microbatch model batches ([#10825](https://github.com/dbt-labs/dbt-core/issues/10825)) +- Allow instances of generic data tests to be documented ([#2578](https://github.com/dbt-labs/dbt-core/issues/2578)) +- Fix warnings for models referring to a deprecated model ([#10833](https://github.com/dbt-labs/dbt-core/issues/10833)) +- Change `lookback` default from `0` to `1` to ensure better data completeness ([#10867](https://github.com/dbt-labs/dbt-core/issues/10867)) +- Make `--event-time-start` and `--event-time-end` mutually required ([#10874](https://github.com/dbt-labs/dbt-core/issues/10874)) +- Exclude hook result from results in on-run-end context ([#7387](https://github.com/dbt-labs/dbt-core/issues/7387)) +- Implement partial parsing for all-yaml snapshots ([#10903](https://github.com/dbt-labs/dbt-core/issues/10903)) +- Restore source quoting behaviour when quoting config provided in dbt_project.yml ([#10892](https://github.com/dbt-labs/dbt-core/issues/10892)) +- Fix bug when referencing deprecated models ([#10915](https://github.com/dbt-labs/dbt-core/issues/10915)) +- Fix 'model' jinja context variable type to dict ([#10927](https://github.com/dbt-labs/dbt-core/issues/10927)) +- Take `end_time` for batches to the ceiling to handle edge case where `event_time` column is a date ([#10868](https://github.com/dbt-labs/dbt-core/issues/10868)) + +### Under the Hood + +- Remove support and testing for Python 3.8, which is now EOL. ([#10861](https://github.com/dbt-labs/dbt-core/issues/10861)) + +### Dependencies + +- Bump minimnum allowed dbt-adapters version to 1.8.0 ([#N/A](https://github.com/dbt-labs/dbt-core/issues/N/A)) + +### Contributors +- [@danlsn](https://github.com/danlsn) ([#10915](https://github.com/dbt-labs/dbt-core/issues/10915)) +- [@plypaul](https://github.com/plypaul) ([#10531](https://github.com/dbt-labs/dbt-core/issues/10531)) diff --git a/.changes/1.9.0/Breaking Changes-20231206-192442.yaml b/.changes/1.9.0/Breaking Changes-20231206-192442.yaml new file mode 100644 index 00000000000..c6aad99bce5 --- /dev/null +++ b/.changes/1.9.0/Breaking Changes-20231206-192442.yaml @@ -0,0 +1,6 @@ +kind: Breaking Changes +body: Fix changing the current working directory when using dpt deps, clean and init. +time: 2023-12-06T19:24:42.575372+09:00 +custom: + Author: rariyama + Issue: "8997" diff --git a/.changes/unreleased/Dependencies-20240509-093717.yaml b/.changes/1.9.0/Dependencies-20240509-093717.yaml similarity index 100% rename from .changes/unreleased/Dependencies-20240509-093717.yaml rename to .changes/1.9.0/Dependencies-20240509-093717.yaml diff --git a/.changes/1.9.0/Dependencies-20240820-131909.yaml b/.changes/1.9.0/Dependencies-20240820-131909.yaml new file mode 100644 index 00000000000..c2392d917db --- /dev/null +++ b/.changes/1.9.0/Dependencies-20240820-131909.yaml @@ -0,0 +1,7 @@ +kind: Dependencies +body: Increase supported version range for dbt-semantic-interfaces. Needed to support + custom calendar features. +time: 2024-08-20T13:19:09.015225-07:00 +custom: + Author: courtneyholcomb + Issue: "9265" diff --git a/.changes/1.9.0/Dependencies-20241030-134526.yaml b/.changes/1.9.0/Dependencies-20241030-134526.yaml new file mode 100644 index 00000000000..ed015170f9a --- /dev/null +++ b/.changes/1.9.0/Dependencies-20241030-134526.yaml @@ -0,0 +1,6 @@ +kind: Dependencies +body: Bump minimnum allowed dbt-adapters version to 1.8.0 +time: 2024-10-30T13:45:26.144328-05:00 +custom: + Author: QMalcolm + Issue: N/A diff --git a/.changes/unreleased/Docs-20240311-140344.yaml b/.changes/1.9.0/Docs-20240311-140344.yaml similarity index 100% rename from .changes/unreleased/Docs-20240311-140344.yaml rename to .changes/1.9.0/Docs-20240311-140344.yaml diff --git a/.changes/unreleased/Docs-20240501-021050.yaml b/.changes/1.9.0/Docs-20240501-021050.yaml similarity index 100% rename from .changes/unreleased/Docs-20240501-021050.yaml rename to .changes/1.9.0/Docs-20240501-021050.yaml diff --git a/.changes/unreleased/Docs-20240516-223036.yaml b/.changes/1.9.0/Docs-20240516-223036.yaml similarity index 100% rename from .changes/unreleased/Docs-20240516-223036.yaml rename to .changes/1.9.0/Docs-20240516-223036.yaml diff --git a/.changes/unreleased/Docs-20240613-151048.yaml b/.changes/1.9.0/Docs-20240613-151048.yaml similarity index 100% rename from .changes/unreleased/Docs-20240613-151048.yaml rename to .changes/1.9.0/Docs-20240613-151048.yaml diff --git a/.changes/unreleased/Features-20240506-175642.yaml b/.changes/1.9.0/Features-20240506-175642.yaml similarity index 100% rename from .changes/unreleased/Features-20240506-175642.yaml rename to .changes/1.9.0/Features-20240506-175642.yaml diff --git a/.changes/unreleased/Features-20240507-162717.yaml b/.changes/1.9.0/Features-20240507-162717.yaml similarity index 100% rename from .changes/unreleased/Features-20240507-162717.yaml rename to .changes/1.9.0/Features-20240507-162717.yaml diff --git a/.changes/unreleased/Features-20240522-000309.yaml b/.changes/1.9.0/Features-20240522-000309.yaml similarity index 100% rename from .changes/unreleased/Features-20240522-000309.yaml rename to .changes/1.9.0/Features-20240522-000309.yaml diff --git a/.changes/unreleased/Features-20240527-124405.yaml b/.changes/1.9.0/Features-20240527-124405.yaml similarity index 100% rename from .changes/unreleased/Features-20240527-124405.yaml rename to .changes/1.9.0/Features-20240527-124405.yaml diff --git a/.changes/unreleased/Features-20240531-150816.yaml b/.changes/1.9.0/Features-20240531-150816.yaml similarity index 100% rename from .changes/unreleased/Features-20240531-150816.yaml rename to .changes/1.9.0/Features-20240531-150816.yaml diff --git a/.changes/unreleased/Features-20240606-112334.yaml b/.changes/1.9.0/Features-20240606-112334.yaml similarity index 100% rename from .changes/unreleased/Features-20240606-112334.yaml rename to .changes/1.9.0/Features-20240606-112334.yaml diff --git a/.changes/unreleased/Features-20240617-103948.yaml b/.changes/1.9.0/Features-20240617-103948.yaml similarity index 100% rename from .changes/unreleased/Features-20240617-103948.yaml rename to .changes/1.9.0/Features-20240617-103948.yaml diff --git a/.changes/1.9.0/Features-20240621-141635.yaml b/.changes/1.9.0/Features-20240621-141635.yaml new file mode 100644 index 00000000000..4e839edb32d --- /dev/null +++ b/.changes/1.9.0/Features-20240621-141635.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Warning message for snapshot timestamp data types +time: 2024-06-21T14:16:35.717637-04:00 +custom: + Author: gshank + Issue: "10234" diff --git a/.changes/unreleased/Features-20240625-095107.yaml b/.changes/1.9.0/Features-20240625-095107.yaml similarity index 100% rename from .changes/unreleased/Features-20240625-095107.yaml rename to .changes/1.9.0/Features-20240625-095107.yaml diff --git a/.changes/unreleased/Features-20240627-162953.yaml b/.changes/1.9.0/Features-20240627-162953.yaml similarity index 100% rename from .changes/unreleased/Features-20240627-162953.yaml rename to .changes/1.9.0/Features-20240627-162953.yaml diff --git a/.changes/unreleased/Features-20240712-214546.yaml b/.changes/1.9.0/Features-20240712-214546.yaml similarity index 100% rename from .changes/unreleased/Features-20240712-214546.yaml rename to .changes/1.9.0/Features-20240712-214546.yaml diff --git a/.changes/1.9.0/Features-20240719-161841.yaml b/.changes/1.9.0/Features-20240719-161841.yaml new file mode 100644 index 00000000000..a802faf6888 --- /dev/null +++ b/.changes/1.9.0/Features-20240719-161841.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Support ref and source in foreign key constraint expressions, bump dbt-common minimum to 1.6 +time: 2024-07-19T16:18:41.434278-04:00 +custom: + Author: michelleark + Issue: "8062" diff --git a/.changes/1.9.0/Features-20240722-202238.yaml b/.changes/1.9.0/Features-20240722-202238.yaml new file mode 100644 index 00000000000..5fda200b919 --- /dev/null +++ b/.changes/1.9.0/Features-20240722-202238.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Support new semantic layer time spine configs to enable sub-daily granularity. +time: 2024-07-22T20:22:38.258249-07:00 +custom: + Author: courtneyholcomb + Issue: "10475" diff --git a/.changes/1.9.0/Features-20240806-144859.yaml b/.changes/1.9.0/Features-20240806-144859.yaml new file mode 100644 index 00000000000..69df7fff614 --- /dev/null +++ b/.changes/1.9.0/Features-20240806-144859.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Add `order_by` and `limit` fields to saved queries. +time: 2024-08-06T14:48:59.035914-07:00 +custom: + Author: plypaul + Issue: "10531" diff --git a/.changes/1.9.0/Features-20240829-135320.yaml b/.changes/1.9.0/Features-20240829-135320.yaml new file mode 100644 index 00000000000..c7f5cf9d8b4 --- /dev/null +++ b/.changes/1.9.0/Features-20240829-135320.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Add support for behavior flags +time: 2024-08-29T13:53:20.16122-04:00 +custom: + Author: mikealfare + Issue: "10618" diff --git a/.changes/1.9.0/Features-20240903-132428.yaml b/.changes/1.9.0/Features-20240903-132428.yaml new file mode 100644 index 00000000000..08df6958990 --- /dev/null +++ b/.changes/1.9.0/Features-20240903-132428.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Enable `--resource-type` and `--exclude-resource-type` CLI flags and environment variables for `dbt test` +time: 2024-09-03T13:24:28.592837+01:00 +custom: + Author: TowardOliver dbeatty10 + Issue: "10656" diff --git a/.changes/1.9.0/Features-20240903-154133.yaml b/.changes/1.9.0/Features-20240903-154133.yaml new file mode 100644 index 00000000000..fe45b8d4d10 --- /dev/null +++ b/.changes/1.9.0/Features-20240903-154133.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Allow configuring snapshot column names +time: 2024-09-03T15:41:33.167097-04:00 +custom: + Author: gshank + Issue: "10185" diff --git a/.changes/1.9.0/Features-20240904-182320.yaml b/.changes/1.9.0/Features-20240904-182320.yaml new file mode 100644 index 00000000000..7d216ec749a --- /dev/null +++ b/.changes/1.9.0/Features-20240904-182320.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Add custom_granularities to YAML spec for time spines. +time: 2024-09-04T18:23:20.234952-07:00 +custom: + Author: courtneyholcomb + Issue: "9265" diff --git a/.changes/1.9.0/Features-20240911-121029.yaml b/.changes/1.9.0/Features-20240911-121029.yaml new file mode 100644 index 00000000000..365faf7fadd --- /dev/null +++ b/.changes/1.9.0/Features-20240911-121029.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Add basic functionality for creating microbatch incremental models +time: 2024-09-11T12:10:29.822189-05:00 +custom: + Author: MichelleArk QMalcolm + Issue: 9490 10635 10637 10638 10636 10662 10639 diff --git a/.changes/1.9.0/Features-20240913-232111.yaml b/.changes/1.9.0/Features-20240913-232111.yaml new file mode 100644 index 00000000000..8f0fc74643e --- /dev/null +++ b/.changes/1.9.0/Features-20240913-232111.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Execute microbatch models in batches +time: 2024-09-13T23:21:11.935434-04:00 +custom: + Author: michelleark + Issue: "10700" diff --git a/.changes/1.9.0/Features-20240918-162959.yaml b/.changes/1.9.0/Features-20240918-162959.yaml new file mode 100644 index 00000000000..62c037540fb --- /dev/null +++ b/.changes/1.9.0/Features-20240918-162959.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Create 'skip_nodes_if_on_run_start_fails' behavior change flag +time: 2024-09-18T16:29:59.268422+01:00 +custom: + Author: aranke + Issue: "7387" diff --git a/.changes/1.9.0/Features-20240920-110447.yaml b/.changes/1.9.0/Features-20240920-110447.yaml new file mode 100644 index 00000000000..ed6d9cb09de --- /dev/null +++ b/.changes/1.9.0/Features-20240920-110447.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Allow snapshots to be defined in YAML. +time: 2024-09-20T11:04:47.703117-04:00 +custom: + Author: peterallenwebb + Issue: "10246" diff --git a/.changes/1.9.0/Features-20240920-172419.yaml b/.changes/1.9.0/Features-20240920-172419.yaml new file mode 100644 index 00000000000..1647d48f1da --- /dev/null +++ b/.changes/1.9.0/Features-20240920-172419.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Write microbatch compiled/run targets to separate files, one per batch +time: 2024-09-20T17:24:19.219556+01:00 +custom: + Author: michelleark + Issue: "10714" diff --git a/.changes/1.9.0/Features-20240923-155335.yaml b/.changes/1.9.0/Features-20240923-155335.yaml new file mode 100644 index 00000000000..dc885a2c3b2 --- /dev/null +++ b/.changes/1.9.0/Features-20240923-155335.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Track incremental_strategy as part of model_run tracking event +time: 2024-09-23T15:53:35.898928+01:00 +custom: + Author: michelleark + Issue: "10761" diff --git a/.changes/1.9.0/Features-20240923-155903.yaml b/.changes/1.9.0/Features-20240923-155903.yaml new file mode 100644 index 00000000000..b05e81b94d2 --- /dev/null +++ b/.changes/1.9.0/Features-20240923-155903.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Support required 'begin' config for microbatch models +time: 2024-09-23T15:59:03.924079+01:00 +custom: + Author: michelleark + Issue: "10701" diff --git a/.changes/1.9.0/Features-20240924-152922.yaml b/.changes/1.9.0/Features-20240924-152922.yaml new file mode 100644 index 00000000000..095a82365ce --- /dev/null +++ b/.changes/1.9.0/Features-20240924-152922.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Added the --inline-direct parameter to 'dbt show' +time: 2024-09-24T15:29:22.874496-04:00 +custom: + Author: aranke peterallenwebb + Issue: "10770" diff --git a/.changes/1.9.0/Features-20240924-154639.yaml b/.changes/1.9.0/Features-20240924-154639.yaml new file mode 100644 index 00000000000..41fbdeaaa6f --- /dev/null +++ b/.changes/1.9.0/Features-20240924-154639.yaml @@ -0,0 +1,7 @@ +kind: Features +body: 'Parse-time validation of microbatch configs: require event_time, batch_size, + lookback and validate input event_time' +time: 2024-09-24T15:46:39.83112+01:00 +custom: + Author: michelleark + Issue: "10709" diff --git a/.changes/1.9.0/Features-20240925-120855.yaml b/.changes/1.9.0/Features-20240925-120855.yaml new file mode 100644 index 00000000000..394e5c91659 --- /dev/null +++ b/.changes/1.9.0/Features-20240925-120855.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Enable specification of dbt_valid_to for current records +time: 2024-09-25T12:08:55.926848-04:00 +custom: + Author: gshank + Issue: "10187" diff --git a/.changes/1.9.0/Features-20240925-165002.yaml b/.changes/1.9.0/Features-20240925-165002.yaml new file mode 100644 index 00000000000..7f8f6d7ac4a --- /dev/null +++ b/.changes/1.9.0/Features-20240925-165002.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Enable `retry` support for microbatch models +time: 2024-09-25T16:50:02.105069-05:00 +custom: + Author: QMalcolm MichelleArk + Issue: 10715 10729 diff --git a/.changes/1.9.0/Features-20240926-140206.yaml b/.changes/1.9.0/Features-20240926-140206.yaml new file mode 100644 index 00000000000..bcee58d0348 --- /dev/null +++ b/.changes/1.9.0/Features-20240926-140206.yaml @@ -0,0 +1,7 @@ +kind: Features +body: Use unrendered database and schema source properties during state:modified, + behind state_modified_compare_more_unrendered_values behavoiur flag +time: 2024-09-26T14:02:06.732068+01:00 +custom: + Author: michelleark + Issue: "9573" diff --git a/.changes/1.9.0/Features-20240926-151057.yaml b/.changes/1.9.0/Features-20240926-151057.yaml new file mode 100644 index 00000000000..682bc892bea --- /dev/null +++ b/.changes/1.9.0/Features-20240926-151057.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Adds validations for custom_granularities to ensure unique naming. +time: 2024-09-26T15:10:57.907694-07:00 +custom: + Author: courtneyholcomb + Issue: "9265" diff --git a/.changes/1.9.0/Features-20240926-153210.yaml b/.changes/1.9.0/Features-20240926-153210.yaml new file mode 100644 index 00000000000..8f8919b918f --- /dev/null +++ b/.changes/1.9.0/Features-20240926-153210.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Ensure microbatch models respect `full_refresh` model config +time: 2024-09-26T15:32:10.202789-05:00 +custom: + Author: QMalcolm + Issue: "10785" diff --git a/.changes/1.9.0/Features-20241001-134051.yaml b/.changes/1.9.0/Features-20241001-134051.yaml new file mode 100644 index 00000000000..60ada51ece3 --- /dev/null +++ b/.changes/1.9.0/Features-20241001-134051.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Enable use of multi-column unique key in snapshots +time: 2024-10-01T13:40:51.297529-04:00 +custom: + Author: gshank + Issue: "9992" diff --git a/.changes/1.9.0/Features-20241003-170529.yaml b/.changes/1.9.0/Features-20241003-170529.yaml new file mode 100644 index 00000000000..8b4ed1b8d74 --- /dev/null +++ b/.changes/1.9.0/Features-20241003-170529.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Ensure `--event-time-start` is before `--event-time-end` +time: 2024-10-03T17:05:29.984398-05:00 +custom: + Author: QMalcolm + Issue: "10786" diff --git a/.changes/1.9.0/Features-20241007-115853.yaml b/.changes/1.9.0/Features-20241007-115853.yaml new file mode 100644 index 00000000000..ac2e61c5b59 --- /dev/null +++ b/.changes/1.9.0/Features-20241007-115853.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Ensure microbatch models use same `current_time` value +time: 2024-10-07T11:58:53.460941-05:00 +custom: + Author: QMalcolm + Issue: "10819" diff --git a/.changes/1.9.0/Features-20241028-173419.yaml b/.changes/1.9.0/Features-20241028-173419.yaml new file mode 100644 index 00000000000..eaf233822c7 --- /dev/null +++ b/.changes/1.9.0/Features-20241028-173419.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Emit warning when microbatch model has no input with `event_time` config +time: 2024-10-28T17:34:19.195209-04:00 +custom: + Author: michelleark + Issue: "10926" diff --git a/.changes/unreleased/Fixes-20230601-204157.yaml b/.changes/1.9.0/Fixes-20230601-204157.yaml similarity index 100% rename from .changes/unreleased/Fixes-20230601-204157.yaml rename to .changes/1.9.0/Fixes-20230601-204157.yaml diff --git a/.changes/1.9.0/Fixes-20230801-094626.yaml b/.changes/1.9.0/Fixes-20230801-094626.yaml new file mode 100644 index 00000000000..28664b70acb --- /dev/null +++ b/.changes/1.9.0/Fixes-20230801-094626.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Test case for `merge_exclude_columns` +time: 2023-08-01T09:46:26.829362-06:00 +custom: + Author: dbeatty10 + Issue: "8267" diff --git a/.changes/unreleased/Fixes-20240113-073615.yaml b/.changes/1.9.0/Fixes-20240113-073615.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240113-073615.yaml rename to .changes/1.9.0/Fixes-20240113-073615.yaml diff --git a/.changes/unreleased/Fixes-20240508-151127.yaml b/.changes/1.9.0/Fixes-20240508-151127.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240508-151127.yaml rename to .changes/1.9.0/Fixes-20240508-151127.yaml diff --git a/.changes/unreleased/Fixes-20240509-091411.yaml b/.changes/1.9.0/Fixes-20240509-091411.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240509-091411.yaml rename to .changes/1.9.0/Fixes-20240509-091411.yaml diff --git a/.changes/unreleased/Fixes-20240516-153913.yaml b/.changes/1.9.0/Fixes-20240516-153913.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240516-153913.yaml rename to .changes/1.9.0/Fixes-20240516-153913.yaml diff --git a/.changes/unreleased/Fixes-20240516-223510.yaml b/.changes/1.9.0/Fixes-20240516-223510.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240516-223510.yaml rename to .changes/1.9.0/Fixes-20240516-223510.yaml diff --git a/.changes/unreleased/Fixes-20240522-182855.yaml b/.changes/1.9.0/Fixes-20240522-182855.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240522-182855.yaml rename to .changes/1.9.0/Fixes-20240522-182855.yaml diff --git a/.changes/unreleased/Fixes-20240523-204251.yaml b/.changes/1.9.0/Fixes-20240523-204251.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240523-204251.yaml rename to .changes/1.9.0/Fixes-20240523-204251.yaml diff --git a/.changes/unreleased/Fixes-20240524-131135.yaml b/.changes/1.9.0/Fixes-20240524-131135.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240524-131135.yaml rename to .changes/1.9.0/Fixes-20240524-131135.yaml diff --git a/.changes/unreleased/Fixes-20240605-111652.yaml b/.changes/1.9.0/Fixes-20240605-111652.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240605-111652.yaml rename to .changes/1.9.0/Fixes-20240605-111652.yaml diff --git a/.changes/unreleased/Fixes-20240607-134648.yaml b/.changes/1.9.0/Fixes-20240607-134648.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240607-134648.yaml rename to .changes/1.9.0/Fixes-20240607-134648.yaml diff --git a/.changes/unreleased/Fixes-20240610-132130.yaml b/.changes/1.9.0/Fixes-20240610-132130.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240610-132130.yaml rename to .changes/1.9.0/Fixes-20240610-132130.yaml diff --git a/.changes/1.9.0/Fixes-20240610-200522.yaml b/.changes/1.9.0/Fixes-20240610-200522.yaml new file mode 100644 index 00000000000..456575644ac --- /dev/null +++ b/.changes/1.9.0/Fixes-20240610-200522.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Use model alias for the CTE identifier generated during ephemeral materialization +time: 2024-06-10T20:05:22.510814008Z +custom: + Author: jeancochrane + Issue: "5273" diff --git a/.changes/unreleased/Fixes-20240612-124256.yaml b/.changes/1.9.0/Fixes-20240612-124256.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240612-124256.yaml rename to .changes/1.9.0/Fixes-20240612-124256.yaml diff --git a/.changes/unreleased/Fixes-20240612-152139.yaml b/.changes/1.9.0/Fixes-20240612-152139.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240612-152139.yaml rename to .changes/1.9.0/Fixes-20240612-152139.yaml diff --git a/.changes/unreleased/Fixes-20240613-183117.yaml b/.changes/1.9.0/Fixes-20240613-183117.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240613-183117.yaml rename to .changes/1.9.0/Fixes-20240613-183117.yaml diff --git a/.changes/unreleased/Fixes-20240624-171729.yaml b/.changes/1.9.0/Fixes-20240624-171729.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240624-171729.yaml rename to .changes/1.9.0/Fixes-20240624-171729.yaml diff --git a/.changes/unreleased/Fixes-20240625-171737.yaml b/.changes/1.9.0/Fixes-20240625-171737.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240625-171737.yaml rename to .changes/1.9.0/Fixes-20240625-171737.yaml diff --git a/.changes/unreleased/Fixes-20240627-154448.yaml b/.changes/1.9.0/Fixes-20240627-154448.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240627-154448.yaml rename to .changes/1.9.0/Fixes-20240627-154448.yaml diff --git a/.changes/unreleased/Fixes-20240709-172440.yaml b/.changes/1.9.0/Fixes-20240709-172440.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240709-172440.yaml rename to .changes/1.9.0/Fixes-20240709-172440.yaml diff --git a/.changes/1.9.0/Fixes-20240714-100254.yaml b/.changes/1.9.0/Fixes-20240714-100254.yaml new file mode 100644 index 00000000000..442abc03498 --- /dev/null +++ b/.changes/1.9.0/Fixes-20240714-100254.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Fix typing for artifact schemas +time: 2024-07-14T10:02:54.452099+09:00 +custom: + Author: nakamichiworks + Issue: "10442" diff --git a/.changes/unreleased/Fixes-20240716-133703.yaml b/.changes/1.9.0/Fixes-20240716-133703.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240716-133703.yaml rename to .changes/1.9.0/Fixes-20240716-133703.yaml diff --git a/.changes/unreleased/Fixes-20240716-171427.yaml b/.changes/1.9.0/Fixes-20240716-171427.yaml similarity index 100% rename from .changes/unreleased/Fixes-20240716-171427.yaml rename to .changes/1.9.0/Fixes-20240716-171427.yaml diff --git a/.changes/1.9.0/Fixes-20240728-221421.yaml b/.changes/1.9.0/Fixes-20240728-221421.yaml new file mode 100644 index 00000000000..420414e3f52 --- /dev/null +++ b/.changes/1.9.0/Fixes-20240728-221421.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Do not update varchar column definitions if a contract exists +time: 2024-07-28T22:14:21.67712-04:00 +custom: + Author: gshank + Issue: "10362" diff --git a/.changes/1.9.0/Fixes-20240731-095152.yaml b/.changes/1.9.0/Fixes-20240731-095152.yaml new file mode 100644 index 00000000000..c7899f6c30b --- /dev/null +++ b/.changes/1.9.0/Fixes-20240731-095152.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: fix all_constraints access, disabled node parsing of non-uniquely named resources +time: 2024-07-31T09:51:52.751135-04:00 +custom: + Author: michelleark gshank + Issue: "10509" diff --git a/.changes/1.9.0/Fixes-20240806-172110.yaml b/.changes/1.9.0/Fixes-20240806-172110.yaml new file mode 100644 index 00000000000..716bf861639 --- /dev/null +++ b/.changes/1.9.0/Fixes-20240806-172110.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Propagate measure label when using create_metrics +time: 2024-08-06T17:21:10.265494-07:00 +custom: + Author: aliceliu + Issue: "10536" diff --git a/.changes/1.9.0/Fixes-20240806-194843.yaml b/.changes/1.9.0/Fixes-20240806-194843.yaml new file mode 100644 index 00000000000..7eb5a4bd8d8 --- /dev/null +++ b/.changes/1.9.0/Fixes-20240806-194843.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: respect --quiet and --warn-error-options for flag deprecations +time: 2024-08-06T19:48:43.399453-04:00 +custom: + Author: michelleark + Issue: "10105" diff --git a/.changes/1.9.0/Fixes-20240813-154235.yaml b/.changes/1.9.0/Fixes-20240813-154235.yaml new file mode 100644 index 00000000000..03c3a3c7cac --- /dev/null +++ b/.changes/1.9.0/Fixes-20240813-154235.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Fix state:modified check for exports +time: 2024-08-13T15:42:35.471685-07:00 +custom: + Author: aliceliu + Issue: "10138" diff --git a/.changes/1.9.0/Fixes-20240816-140807.yaml b/.changes/1.9.0/Fixes-20240816-140807.yaml new file mode 100644 index 00000000000..1f6c60da6da --- /dev/null +++ b/.changes/1.9.0/Fixes-20240816-140807.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Filter out empty nodes after graph selection to support consistent selection of nodes that depend on upstream public models +time: 2024-08-16T14:08:07.426235-07:00 +custom: + Author: jtcohen6 + Issue: "8987" diff --git a/.changes/1.9.0/Fixes-20240824-210903.yaml b/.changes/1.9.0/Fixes-20240824-210903.yaml new file mode 100644 index 00000000000..bf0dc9cb022 --- /dev/null +++ b/.changes/1.9.0/Fixes-20240824-210903.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Late render pre- and post-hooks configs in properties / schema YAML files +time: 2024-08-24T21:09:03.252733-06:00 +custom: + Author: dbeatty10 + Issue: "10603" diff --git a/.changes/1.9.0/Fixes-20240829-105701.yaml b/.changes/1.9.0/Fixes-20240829-105701.yaml new file mode 100644 index 00000000000..170f2463fa2 --- /dev/null +++ b/.changes/1.9.0/Fixes-20240829-105701.yaml @@ -0,0 +1,7 @@ +kind: Fixes +body: Allow the use of env_var function in certain macros in which it was previously + unavailable. +time: 2024-08-29T10:57:01.160613-04:00 +custom: + Author: peterallenwebb + Issue: "10609" diff --git a/.changes/1.9.0/Fixes-20240905-180248.yaml b/.changes/1.9.0/Fixes-20240905-180248.yaml new file mode 100644 index 00000000000..3d18c28a4f6 --- /dev/null +++ b/.changes/1.9.0/Fixes-20240905-180248.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: 'Remove deprecation for tests: to data_tests: change' +time: 2024-09-05T18:02:48.086421-04:00 +custom: + Author: gshank + Issue: "10564" diff --git a/.changes/1.9.0/Fixes-20240917-174446.yaml b/.changes/1.9.0/Fixes-20240917-174446.yaml new file mode 100644 index 00000000000..89dd65e6f35 --- /dev/null +++ b/.changes/1.9.0/Fixes-20240917-174446.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Fix `--resource-type test` for `dbt list` and `dbt build` +time: 2024-09-17T17:44:46.121032-06:00 +custom: + Author: dbeatty10 + Issue: "10730" diff --git a/.changes/1.9.0/Fixes-20240922-133527.yaml b/.changes/1.9.0/Fixes-20240922-133527.yaml new file mode 100644 index 00000000000..f31fe8c3365 --- /dev/null +++ b/.changes/1.9.0/Fixes-20240922-133527.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Fix unit tests for incremental model with alias +time: 2024-09-22T13:35:27.991398741Z +custom: + Author: katsugeneration + Issue: "10754" diff --git a/.changes/1.9.0/Fixes-20240923-190758.yaml b/.changes/1.9.0/Fixes-20240923-190758.yaml new file mode 100644 index 00000000000..4d005ec5999 --- /dev/null +++ b/.changes/1.9.0/Fixes-20240923-190758.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Allow singular tests to be documented in properties.yml +time: 2024-09-23T19:07:58.151069+01:00 +custom: + Author: aranke + Issue: "9005" diff --git a/.changes/1.9.0/Fixes-20240923-202024.yaml b/.changes/1.9.0/Fixes-20240923-202024.yaml new file mode 100644 index 00000000000..2170773494c --- /dev/null +++ b/.changes/1.9.0/Fixes-20240923-202024.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Ignore --empty in unit test ref/source rendering +time: 2024-09-23T20:20:24.151285+01:00 +custom: + Author: michelleark + Issue: "10516" diff --git a/.changes/1.9.0/Fixes-20240925-131028.yaml b/.changes/1.9.0/Fixes-20240925-131028.yaml new file mode 100644 index 00000000000..2e36f340f1c --- /dev/null +++ b/.changes/1.9.0/Fixes-20240925-131028.yaml @@ -0,0 +1,7 @@ +kind: Fixes +body: Ignore rendered jinja in configs for state:modified, behind state_modified_compare_more_unrendered_values + behaviour flag +time: 2024-09-25T13:10:28.490042+01:00 +custom: + Author: michelleark + Issue: "9564" diff --git a/.changes/1.9.0/Fixes-20240925-154514.yaml b/.changes/1.9.0/Fixes-20240925-154514.yaml new file mode 100644 index 00000000000..8db05cc6a0e --- /dev/null +++ b/.changes/1.9.0/Fixes-20240925-154514.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Pass test user config to adapter pre_hook by explicitly adding test builder config to node +time: 2024-09-25T15:45:14.459598-07:00 +custom: + Author: colin-rogers-dbt + Issue: "10484" diff --git a/.changes/1.9.0/Fixes-20240925-160543.yaml b/.changes/1.9.0/Fixes-20240925-160543.yaml new file mode 100644 index 00000000000..68af365c820 --- /dev/null +++ b/.changes/1.9.0/Fixes-20240925-160543.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Improve performance of infer primary key +time: 2024-09-25T16:05:43.59536-04:00 +custom: + Author: gshank + Issue: "10781" diff --git a/.changes/1.9.0/Fixes-20240926-101220.yaml b/.changes/1.9.0/Fixes-20240926-101220.yaml new file mode 100644 index 00000000000..677ee458048 --- /dev/null +++ b/.changes/1.9.0/Fixes-20240926-101220.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Attempt to skip saved query processing when no semantic manifest changes +time: 2024-09-26T10:12:20.193453-04:00 +custom: + Author: gshank + Issue: "10563" diff --git a/.changes/1.9.0/Fixes-20240930-153158.yaml b/.changes/1.9.0/Fixes-20240930-153158.yaml new file mode 100644 index 00000000000..33970504367 --- /dev/null +++ b/.changes/1.9.0/Fixes-20240930-153158.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Ensure dbt retry of microbatch models doesn't lose prior successful state +time: 2024-09-30T15:31:58.541656-05:00 +custom: + Author: QMalcolm + Issue: "10800" diff --git a/.changes/1.9.0/Fixes-20241004-133630.yaml b/.changes/1.9.0/Fixes-20241004-133630.yaml new file mode 100644 index 00000000000..7d23bda0c13 --- /dev/null +++ b/.changes/1.9.0/Fixes-20241004-133630.yaml @@ -0,0 +1,7 @@ +kind: Fixes +body: Handle edge cases when a specified `--event-time-end` is equivalent to the batch + size truncated batch start time +time: 2024-10-04T13:36:30.357936-05:00 +custom: + Author: QMalcolm + Issue: "10824" diff --git a/.changes/1.9.0/Fixes-20241004-163908.yaml b/.changes/1.9.0/Fixes-20241004-163908.yaml new file mode 100644 index 00000000000..12aec93d0ed --- /dev/null +++ b/.changes/1.9.0/Fixes-20241004-163908.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Begin tracking execution time of microbatch model batches +time: 2024-10-04T16:39:08.464064-05:00 +custom: + Author: QMalcolm + Issue: "10825" diff --git a/.changes/1.9.0/Fixes-20241014-212135.yaml b/.changes/1.9.0/Fixes-20241014-212135.yaml new file mode 100644 index 00000000000..8cc700095ba --- /dev/null +++ b/.changes/1.9.0/Fixes-20241014-212135.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Allow instances of generic data tests to be documented +time: 2024-10-14T21:21:35.767115+01:00 +custom: + Author: aranke + Issue: "2578" diff --git a/.changes/1.9.0/Fixes-20241015-121825.yaml b/.changes/1.9.0/Fixes-20241015-121825.yaml new file mode 100644 index 00000000000..f5867871868 --- /dev/null +++ b/.changes/1.9.0/Fixes-20241015-121825.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Fix warnings for models referring to a deprecated model +time: 2024-10-15T12:18:25.14525-04:00 +custom: + Author: gshank + Issue: "10833" diff --git a/.changes/1.9.0/Fixes-20241017-134857.yaml b/.changes/1.9.0/Fixes-20241017-134857.yaml new file mode 100644 index 00000000000..407c7bad3e6 --- /dev/null +++ b/.changes/1.9.0/Fixes-20241017-134857.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Change `lookback` default from `0` to `1` to ensure better data completeness +time: 2024-10-17T13:48:57.805205-07:00 +custom: + Author: QMalcolm + Issue: "10867" diff --git a/.changes/1.9.0/Fixes-20241017-145357.yaml b/.changes/1.9.0/Fixes-20241017-145357.yaml new file mode 100644 index 00000000000..8736a1247f7 --- /dev/null +++ b/.changes/1.9.0/Fixes-20241017-145357.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Make `--event-time-start` and `--event-time-end` mutually required +time: 2024-10-17T14:53:57.149238-07:00 +custom: + Author: QMalcolm + Issue: "10874" diff --git a/.changes/1.9.0/Fixes-20241018-135810.yaml b/.changes/1.9.0/Fixes-20241018-135810.yaml new file mode 100644 index 00000000000..c205e15bb09 --- /dev/null +++ b/.changes/1.9.0/Fixes-20241018-135810.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Exclude hook result from results in on-run-end context +time: 2024-10-18T13:58:10.396884-07:00 +custom: + Author: ChenyuLInx + Issue: "7387" diff --git a/.changes/1.9.0/Fixes-20241022-222927.yaml b/.changes/1.9.0/Fixes-20241022-222927.yaml new file mode 100644 index 00000000000..cd294862ba4 --- /dev/null +++ b/.changes/1.9.0/Fixes-20241022-222927.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Implement partial parsing for all-yaml snapshots +time: 2024-10-22T22:29:27.396378-04:00 +custom: + Author: gshank + Issue: "10903" diff --git a/.changes/1.9.0/Fixes-20241023-152054.yaml b/.changes/1.9.0/Fixes-20241023-152054.yaml new file mode 100644 index 00000000000..976f4cf3add --- /dev/null +++ b/.changes/1.9.0/Fixes-20241023-152054.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Restore source quoting behaviour when quoting config provided in dbt_project.yml +time: 2024-10-23T15:20:54.766893-04:00 +custom: + Author: michelleark + Issue: "10892" diff --git a/.changes/1.9.0/Fixes-20241024-104938.yaml b/.changes/1.9.0/Fixes-20241024-104938.yaml new file mode 100644 index 00000000000..f47d9feb98b --- /dev/null +++ b/.changes/1.9.0/Fixes-20241024-104938.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Fix bug when referencing deprecated models +time: 2024-10-24T10:49:38.352328-06:00 +custom: + Author: dbeatty10 danlsn + Issue: "10915" diff --git a/.changes/1.9.0/Fixes-20241028-132751.yaml b/.changes/1.9.0/Fixes-20241028-132751.yaml new file mode 100644 index 00000000000..6e1f79b9b9c --- /dev/null +++ b/.changes/1.9.0/Fixes-20241028-132751.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: 'Fix ''model'' jinja context variable type to dict ' +time: 2024-10-28T13:27:51.604093-04:00 +custom: + Author: michelleark + Issue: "10927" diff --git a/.changes/1.9.0/Fixes-20241029-161615.yaml b/.changes/1.9.0/Fixes-20241029-161615.yaml new file mode 100644 index 00000000000..ba27493d303 --- /dev/null +++ b/.changes/1.9.0/Fixes-20241029-161615.yaml @@ -0,0 +1,7 @@ +kind: Fixes +body: Take `end_time` for batches to the ceiling to handle edge case where `event_time` + column is a date +time: 2024-10-29T16:16:15.714993-05:00 +custom: + Author: QMalcolm MichelleArk + Issue: "10868" diff --git a/.changes/unreleased/Security-20240522-094540.yaml b/.changes/1.9.0/Security-20240522-094540.yaml similarity index 100% rename from .changes/unreleased/Security-20240522-094540.yaml rename to .changes/1.9.0/Security-20240522-094540.yaml diff --git a/.changes/unreleased/Under the Hood-20240502-154430.yaml b/.changes/1.9.0/Under the Hood-20240502-154430.yaml similarity index 100% rename from .changes/unreleased/Under the Hood-20240502-154430.yaml rename to .changes/1.9.0/Under the Hood-20240502-154430.yaml diff --git a/.changes/unreleased/Under the Hood-20240506-145511.yaml b/.changes/1.9.0/Under the Hood-20240506-145511.yaml similarity index 100% rename from .changes/unreleased/Under the Hood-20240506-145511.yaml rename to .changes/1.9.0/Under the Hood-20240506-145511.yaml diff --git a/.changes/unreleased/Under the Hood-20240519-155946.yaml b/.changes/1.9.0/Under the Hood-20240519-155946.yaml similarity index 100% rename from .changes/unreleased/Under the Hood-20240519-155946.yaml rename to .changes/1.9.0/Under the Hood-20240519-155946.yaml diff --git a/.changes/unreleased/Under the Hood-20240529-102814.yaml b/.changes/1.9.0/Under the Hood-20240529-102814.yaml similarity index 100% rename from .changes/unreleased/Under the Hood-20240529-102814.yaml rename to .changes/1.9.0/Under the Hood-20240529-102814.yaml diff --git a/.changes/unreleased/Under the Hood-20240618-140652.yaml b/.changes/1.9.0/Under the Hood-20240618-140652.yaml similarity index 100% rename from .changes/unreleased/Under the Hood-20240618-140652.yaml rename to .changes/1.9.0/Under the Hood-20240618-140652.yaml diff --git a/.changes/unreleased/Under the Hood-20240701-131750.yaml b/.changes/1.9.0/Under the Hood-20240701-131750.yaml similarity index 100% rename from .changes/unreleased/Under the Hood-20240701-131750.yaml rename to .changes/1.9.0/Under the Hood-20240701-131750.yaml diff --git a/.changes/unreleased/Under the Hood-20240716-184859.yaml b/.changes/1.9.0/Under the Hood-20240716-184859.yaml similarity index 100% rename from .changes/unreleased/Under the Hood-20240716-184859.yaml rename to .changes/1.9.0/Under the Hood-20240716-184859.yaml diff --git a/.changes/unreleased/Under the Hood-20240716-205703.yaml b/.changes/1.9.0/Under the Hood-20240716-205703.yaml similarity index 100% rename from .changes/unreleased/Under the Hood-20240716-205703.yaml rename to .changes/1.9.0/Under the Hood-20240716-205703.yaml diff --git a/.changes/1.9.0/Under the Hood-20240806-155406.yaml b/.changes/1.9.0/Under the Hood-20240806-155406.yaml new file mode 100644 index 00000000000..aafcd955ba9 --- /dev/null +++ b/.changes/1.9.0/Under the Hood-20240806-155406.yaml @@ -0,0 +1,6 @@ +kind: Under the Hood +body: Move from minimal-snowplow-tracker fork back to snowplow-tracker +time: 2024-08-06T15:54:06.422444-04:00 +custom: + Author: peterallenwebb + Issue: "8409" diff --git a/.changes/1.9.0/Under the Hood-20240807-155652.yaml b/.changes/1.9.0/Under the Hood-20240807-155652.yaml new file mode 100644 index 00000000000..3bd9bcbe273 --- /dev/null +++ b/.changes/1.9.0/Under the Hood-20240807-155652.yaml @@ -0,0 +1,7 @@ +kind: Under the Hood +body: Add group info to RunResultError, RunResultFailure, RunResultWarning log lines +time: 2024-08-07T15:56:52.171199-05:00 +custom: + Author: aranke + Issue: "" + JiraID: "364" diff --git a/.changes/1.9.0/Under the Hood-20240809-130234.yaml b/.changes/1.9.0/Under the Hood-20240809-130234.yaml new file mode 100644 index 00000000000..964dd2fedf2 --- /dev/null +++ b/.changes/1.9.0/Under the Hood-20240809-130234.yaml @@ -0,0 +1,6 @@ +kind: Under the Hood +body: Improve speed of tree traversal when finding children, increasing build speed for some selectors +time: 2024-08-09T13:02:34.759905-07:00 +custom: + Author: ttusing + Issue: "10434" diff --git a/.changes/1.9.0/Under the Hood-20240821-095516.yaml b/.changes/1.9.0/Under the Hood-20240821-095516.yaml new file mode 100644 index 00000000000..b9335434bf3 --- /dev/null +++ b/.changes/1.9.0/Under the Hood-20240821-095516.yaml @@ -0,0 +1,6 @@ +kind: Under the Hood +body: Add test for sources tables with quotes +time: 2024-08-21T09:55:16.038101-04:00 +custom: + Author: gshank + Issue: "10582" diff --git a/.changes/1.9.0/Under the Hood-20240827-105014.yaml b/.changes/1.9.0/Under the Hood-20240827-105014.yaml new file mode 100644 index 00000000000..318f9ccb1b0 --- /dev/null +++ b/.changes/1.9.0/Under the Hood-20240827-105014.yaml @@ -0,0 +1,6 @@ +kind: Under the Hood +body: Additional type hints for `core/dbt/version.py` +time: 2024-08-27T10:50:14.047859-05:00 +custom: + Author: QMalcolm + Issue: "10612" diff --git a/.changes/1.9.0/Under the Hood-20240827-113123.yaml b/.changes/1.9.0/Under the Hood-20240827-113123.yaml new file mode 100644 index 00000000000..889c07239de --- /dev/null +++ b/.changes/1.9.0/Under the Hood-20240827-113123.yaml @@ -0,0 +1,6 @@ +kind: Under the Hood +body: Fix typing issues in core/dbt/contracts/sql.py +time: 2024-08-27T11:31:23.749912-05:00 +custom: + Author: QMalcolm + Issue: "10614" diff --git a/.changes/1.9.0/Under the Hood-20240827-114810.yaml b/.changes/1.9.0/Under the Hood-20240827-114810.yaml new file mode 100644 index 00000000000..43d17ef4dc5 --- /dev/null +++ b/.changes/1.9.0/Under the Hood-20240827-114810.yaml @@ -0,0 +1,6 @@ +kind: Under the Hood +body: Fix type errors in `dbt/core/task/clean.py` +time: 2024-08-27T11:48:10.438173-05:00 +custom: + Author: QMalcolm + Issue: "10616" diff --git a/.changes/1.9.0/Under the Hood-20240911-162730.yaml b/.changes/1.9.0/Under the Hood-20240911-162730.yaml new file mode 100644 index 00000000000..0d35aeb5262 --- /dev/null +++ b/.changes/1.9.0/Under the Hood-20240911-162730.yaml @@ -0,0 +1,6 @@ +kind: Under the Hood +body: Add Snowplow tracking for behavior flag deprecations +time: 2024-09-11T16:27:30.293832-04:00 +custom: + Author: mikealfare + Issue: "10552" diff --git a/.changes/1.9.0/Under the Hood-20240913-213312.yaml b/.changes/1.9.0/Under the Hood-20240913-213312.yaml new file mode 100644 index 00000000000..495b6f8de53 --- /dev/null +++ b/.changes/1.9.0/Under the Hood-20240913-213312.yaml @@ -0,0 +1,6 @@ +kind: Under the Hood +body: Add test utility patch_microbatch_end_time for adapters testing +time: 2024-09-13T21:33:12.482336-04:00 +custom: + Author: michelleark + Issue: "10713" diff --git a/.changes/1.9.0/Under the Hood-20240916-102201.yaml b/.changes/1.9.0/Under the Hood-20240916-102201.yaml new file mode 100644 index 00000000000..48485d44f48 --- /dev/null +++ b/.changes/1.9.0/Under the Hood-20240916-102201.yaml @@ -0,0 +1,6 @@ +kind: Under the Hood +body: Replace `TestSelector` with `ResourceTypeSelector` +time: 2024-09-16T10:22:01.339462-06:00 +custom: + Author: dbeatty10 + Issue: "10718" diff --git a/.changes/1.9.0/Under the Hood-20240918-170325.yaml b/.changes/1.9.0/Under the Hood-20240918-170325.yaml new file mode 100644 index 00000000000..3f265a36eda --- /dev/null +++ b/.changes/1.9.0/Under the Hood-20240918-170325.yaml @@ -0,0 +1,7 @@ +kind: Under the Hood +body: Standardize returning `ResourceTypeSelector` instances in `dbt list` and `dbt + build` +time: 2024-09-18T17:03:25.639516-06:00 +custom: + Author: dbeatty10 + Issue: "10739" diff --git a/.changes/1.9.0/Under the Hood-20240926-143448.yaml b/.changes/1.9.0/Under the Hood-20240926-143448.yaml new file mode 100644 index 00000000000..95ac0a2ade3 --- /dev/null +++ b/.changes/1.9.0/Under the Hood-20240926-143448.yaml @@ -0,0 +1,6 @@ +kind: Under the Hood +body: Add group metadata info to LogModelResult and LogTestResult +time: 2024-09-26T14:34:48.334703+01:00 +custom: + Author: aranke + Issue: "10775" diff --git a/.changes/1.9.0/Under the Hood-20241016-144056.yaml b/.changes/1.9.0/Under the Hood-20241016-144056.yaml new file mode 100644 index 00000000000..454724607cc --- /dev/null +++ b/.changes/1.9.0/Under the Hood-20241016-144056.yaml @@ -0,0 +1,6 @@ +kind: Under the Hood +body: Remove support and testing for Python 3.8, which is now EOL. +time: 2024-10-16T14:40:56.451972-04:00 +custom: + Author: gshank peterallenwebb + Issue: "10861" diff --git a/.changes/unreleased/Features-20241029-181728.yaml b/.changes/unreleased/Features-20241029-181728.yaml new file mode 100644 index 00000000000..5124bc55dd4 --- /dev/null +++ b/.changes/unreleased/Features-20241029-181728.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Emit debug logging event whenever artifacts are written +time: 2024-10-29T18:17:28.321188-05:00 +custom: + Author: QMalcolm + Issue: 10937 diff --git a/.changes/unreleased/Features-20241031-163149.yaml b/.changes/unreleased/Features-20241031-163149.yaml new file mode 100644 index 00000000000..209f2180daa --- /dev/null +++ b/.changes/unreleased/Features-20241031-163149.yaml @@ -0,0 +1,6 @@ +kind: Features +body: Support --empty for snapshots +time: 2024-10-31T16:31:49.926164-04:00 +custom: + Author: gshank + Issue: "10372" diff --git a/.changes/unreleased/Fixes-20241017-153022.yaml b/.changes/unreleased/Fixes-20241017-153022.yaml new file mode 100644 index 00000000000..905a40db501 --- /dev/null +++ b/.changes/unreleased/Fixes-20241017-153022.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Ensure KeyboardInterrupt/SystemExit halts microbatch model execution +time: 2024-10-17T15:30:22.781854-07:00 +custom: + Author: QMalcolm + Issue: "10862" diff --git a/.changes/unreleased/Fixes-20241029-182034.yaml b/.changes/unreleased/Fixes-20241029-182034.yaml new file mode 100644 index 00000000000..5b5f6f6a07d --- /dev/null +++ b/.changes/unreleased/Fixes-20241029-182034.yaml @@ -0,0 +1,7 @@ +kind: Fixes +body: Handle exceptions in `get_execution_status` more broadly to better ensure `run_results.json` + gets written +time: 2024-10-29T18:20:34.782845-05:00 +custom: + Author: QMalcolm + Issue: "10934" diff --git a/.changes/unreleased/Under the Hood-20241031-093251.yaml b/.changes/unreleased/Under the Hood-20241031-093251.yaml new file mode 100644 index 00000000000..430f10b58ab --- /dev/null +++ b/.changes/unreleased/Under the Hood-20241031-093251.yaml @@ -0,0 +1,6 @@ +kind: Under the Hood +body: Behavior change for mf timespine without yaml configuration +time: 2024-10-31T09:32:51.166594-05:00 +custom: + Author: DevonFulcher + Issue: "10959" diff --git a/.github/actions/setup-postgres-linux/action.yml b/.github/actions/setup-postgres-linux/action.yml index 1c8fc772a8a..30050b1a38d 100644 --- a/.github/actions/setup-postgres-linux/action.yml +++ b/.github/actions/setup-postgres-linux/action.yml @@ -5,6 +5,15 @@ runs: steps: - shell: bash run: | - sudo systemctl start postgresql.service + sudo apt-get --purge remove postgresql postgresql-* + sudo apt update -y + sudo apt install gnupg2 wget vim -y + sudo sh -c 'echo "deb https://apt.postgresql.org/pub/repos/apt $(lsb_release -cs)-pgdg main" > /etc/apt/sources.list.d/pgdg.list' + curl -fsSL https://www.postgresql.org/media/keys/ACCC4CF8.asc|sudo gpg --dearmor -o /etc/apt/trusted.gpg.d/postgresql.gpg + sudo apt update -y + sudo apt install postgresql-16 + sudo apt-get -y install postgresql postgresql-contrib + sudo systemctl start postgresql + sudo systemctl enable postgresql pg_isready sudo -u postgres bash ${{ github.action_path }}/setup_db.sh diff --git a/.github/actions/setup-postgres-macos/action.yml b/.github/actions/setup-postgres-macos/action.yml index af9a9fe1657..3ce1cd7144e 100644 --- a/.github/actions/setup-postgres-macos/action.yml +++ b/.github/actions/setup-postgres-macos/action.yml @@ -5,7 +5,8 @@ runs: steps: - shell: bash run: | - brew services start postgresql + brew install postgresql@16 + brew services start postgresql@16 echo "Check PostgreSQL service is running" i=10 COMMAND='pg_isready' diff --git a/.github/actions/setup-postgres-windows/action.yml b/.github/actions/setup-postgres-windows/action.yml index 419b5e267cb..2b6217d360f 100644 --- a/.github/actions/setup-postgres-windows/action.yml +++ b/.github/actions/setup-postgres-windows/action.yml @@ -5,8 +5,22 @@ runs: steps: - shell: pwsh run: | - $pgService = Get-Service -Name postgresql* + Write-Host -Object "Installing PostgreSQL 16 as windows service..." + $installerArgs = @("--install_runtimes 0", "--superpassword root", "--enable_acledit 1", "--unattendedmodeui none", "--mode unattended") + $filePath = Invoke-DownloadWithRetry -Url "https://get.enterprisedb.com/postgresql/postgresql-16.1-1-windows-x64.exe" -Path "$env:PGROOT/postgresql-16.1-1-windows-x64.exe" + Start-Process -FilePath $filePath -ArgumentList $installerArgs -Wait -PassThru + + Write-Host -Object "Validating PostgreSQL 16 Install..." + Get-Service -Name postgresql* + $pgReady = Start-Process -FilePath "$env:PGBIN\pg_isready" -Wait -PassThru + $exitCode = $pgReady.ExitCode + if ($exitCode -ne 0) { + Write-Host -Object "PostgreSQL is not ready. Exitcode: $exitCode" + exit $exitCode + } + + Write-Host -Object "Starting PostgreSQL 16 Service..." + $pgService = Get-Service -Name postgresql-x64-16 Set-Service -InputObject $pgService -Status running -StartupType automatic - Start-Process -FilePath "$env:PGBIN\pg_isready" -Wait -PassThru $env:Path += ";$env:PGBIN" bash ${{ github.action_path }}/setup_db.sh diff --git a/.github/pull_request_template.md b/.github/pull_request_template.md index 2831dd14422..94c53a02ef1 100644 --- a/.github/pull_request_template.md +++ b/.github/pull_request_template.md @@ -1,7 +1,7 @@ -resolves # +Resolves # /,greedy:!0},prolog:{pattern:/<\?[\s\S]+?\?>/,greedy:!0},doctype:{pattern:/"'[\]]|"[^"]*"|'[^']*')+(?:\[(?:[^<"'\]]|"[^"]*"|'[^']*'|<(?!!--)|)*\]\s*)?>/i,greedy:!0,inside:{"internal-subset":{pattern:/(^[^\[]*\[)[\s\S]+(?=\]>$)/,lookbehind:!0,greedy:!0,inside:null},string:{pattern:/"[^"]*"|'[^']*'/,greedy:!0},punctuation:/^$|[[\]]/,"doctype-tag":/^DOCTYPE/i,name:/[^\s<>'"]+/}},cdata:{pattern://i,greedy:!0},tag:{pattern:/<\/?(?!\d)[^\s>\/=$<%]+(?:\s(?:\s*[^\s>\/=]+(?:\s*=\s*(?:"[^"]*"|'[^']*'|[^\s'">=]+(?=[\s>]))|(?=[\s/>])))+)?\s*\/?>/,greedy:!0,inside:{tag:{pattern:/^<\/?[^\s>\/]+/,inside:{punctuation:/^<\/?/,namespace:/^[^\s>\/:]+:/}},"special-attr":[],"attr-value":{pattern:/=\s*(?:"[^"]*"|'[^']*'|[^\s'">=]+)/,inside:{punctuation:[{pattern:/^=/,alias:"attr-equals"},/"|'/]}},punctuation:/\/?>/,"attr-name":{pattern:/[^\s>\/]+/,inside:{namespace:/^[^\s>\/:]+:/}}}},entity:[{pattern:/&[\da-z]{1,8};/i,alias:"named-entity"},/&#x?[\da-f]{1,8};/i]},n.languages.markup.tag.inside["attr-value"].inside.entity=n.languages.markup.entity,n.languages.markup.doctype.inside["internal-subset"].inside=n.languages.markup,n.hooks.add("wrap",(function(e){"entity"===e.type&&(e.attributes.title=e.content.replace(/&/,"&"))})),Object.defineProperty(n.languages.markup.tag,"addInlined",{value:function(e,t){var r={};r["language-"+t]={pattern:/(^$)/i,lookbehind:!0,inside:n.languages[t]},r.cdata=/^$/i;var i={"included-cdata":{pattern://i,inside:r}};i["language-"+t]={pattern:/[\s\S]+/,inside:n.languages[t]};var o={};o[e]={pattern:RegExp(/(<__[^>]*>)(?:))*\]\]>|(?!)/.source.replace(/__/g,(function(){return e})),"i"),lookbehind:!0,greedy:!0,inside:i},n.languages.insertBefore("markup","cdata",o)}}),Object.defineProperty(n.languages.markup.tag,"addAttribute",{value:function(e,t){n.languages.markup.tag.inside["special-attr"].push({pattern:RegExp(/(^|["'\s])/.source+"(?:"+e+")"+/\s*=\s*(?:"[^"]*"|'[^']*'|[^\s'">=]+(?=[\s>]))/.source,"i"),lookbehind:!0,inside:{"attr-name":/^[^\s=]+/,"attr-value":{pattern:/=[\s\S]+/,inside:{value:{pattern:/(^=\s*(["']|(?!["'])))\S[\s\S]*(?=\2$)/,lookbehind:!0,alias:[t,"language-"+t],inside:n.languages[t]},punctuation:[{pattern:/^=/,alias:"attr-equals"},/"|'/]}}}})}}),n.languages.html=n.languages.markup,n.languages.mathml=n.languages.markup,n.languages.svg=n.languages.markup,n.languages.xml=n.languages.extend("markup",{}),n.languages.ssml=n.languages.xml,n.languages.atom=n.languages.xml,n.languages.rss=n.languages.xml,function(e){var t=/(?:"(?:\\(?:\r\n|[\s\S])|[^"\\\r\n])*"|'(?:\\(?:\r\n|[\s\S])|[^'\\\r\n])*')/;e.languages.css={comment:/\/\*[\s\S]*?\*\//,atrule:{pattern:/@[\w-](?:[^;{\s]|\s+(?![\s{]))*(?:;|(?=\s*\{))/,inside:{rule:/^@[\w-]+/,"selector-function-argument":{pattern:/(\bselector\s*\(\s*(?![\s)]))(?:[^()\s]|\s+(?![\s)])|\((?:[^()]|\([^()]*\))*\))+(?=\s*\))/,lookbehind:!0,alias:"selector"},keyword:{pattern:/(^|[^\w-])(?:and|not|only|or)(?![\w-])/,lookbehind:!0}}},url:{pattern:RegExp("\\burl\\((?:"+t.source+"|"+/(?:[^\\\r\n()"']|\\[\s\S])*/.source+")\\)","i"),greedy:!0,inside:{function:/^url/i,punctuation:/^\(|\)$/,string:{pattern:RegExp("^"+t.source+"$"),alias:"url"}}},selector:{pattern:RegExp("(^|[{}\\s])[^{}\\s](?:[^{};\"'\\s]|\\s+(?![\\s{])|"+t.source+")*(?=\\s*\\{)"),lookbehind:!0},string:{pattern:t,greedy:!0},property:{pattern:/(^|[^-\w\xA0-\uFFFF])(?!\s)[-_a-z\xA0-\uFFFF](?:(?!\s)[-\w\xA0-\uFFFF])*(?=\s*:)/i,lookbehind:!0},important:/!important\b/i,function:{pattern:/(^|[^-a-z0-9])[-a-z0-9]+(?=\()/i,lookbehind:!0},punctuation:/[(){};:,]/},e.languages.css.atrule.inside.rest=e.languages.css;var n=e.languages.markup;n&&(n.tag.addInlined("style","css"),n.tag.addAttribute("style","css"))}(n),n.languages.clike={comment:[{pattern:/(^|[^\\])\/\*[\s\S]*?(?:\*\/|$)/,lookbehind:!0,greedy:!0},{pattern:/(^|[^\\:])\/\/.*/,lookbehind:!0,greedy:!0}],string:{pattern:/(["'])(?:\\(?:\r\n|[\s\S])|(?!\1)[^\\\r\n])*\1/,greedy:!0},"class-name":{pattern:/(\b(?:class|extends|implements|instanceof|interface|new|trait)\s+|\bcatch\s+\()[\w.\\]+/i,lookbehind:!0,inside:{punctuation:/[.\\]/}},keyword:/\b(?:break|catch|continue|do|else|finally|for|function|if|in|instanceof|new|null|return|throw|try|while)\b/,boolean:/\b(?:false|true)\b/,function:/\b\w+(?=\()/,number:/\b0x[\da-f]+\b|(?:\b\d+(?:\.\d*)?|\B\.\d+)(?:e[+-]?\d+)?/i,operator:/[<>]=?|[!=]=?=?|--?|\+\+?|&&?|\|\|?|[?*/~^%]/,punctuation:/[{}[\];(),.:]/},n.languages.javascript=n.languages.extend("clike",{"class-name":[n.languages.clike["class-name"],{pattern:/(^|[^$\w\xA0-\uFFFF])(?!\s)[_$A-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\.(?:constructor|prototype))/,lookbehind:!0}],keyword:[{pattern:/((?:^|\})\s*)catch\b/,lookbehind:!0},{pattern:/(^|[^.]|\.\.\.\s*)\b(?:as|assert(?=\s*\{)|async(?=\s*(?:function\b|\(|[$\w\xA0-\uFFFF]|$))|await|break|case|class|const|continue|debugger|default|delete|do|else|enum|export|extends|finally(?=\s*(?:\{|$))|for|from(?=\s*(?:['"]|$))|function|(?:get|set)(?=\s*(?:[#\[$\w\xA0-\uFFFF]|$))|if|implements|import|in|instanceof|interface|let|new|null|of|package|private|protected|public|return|static|super|switch|this|throw|try|typeof|undefined|var|void|while|with|yield)\b/,lookbehind:!0}],function:/#?(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\s*(?:\.\s*(?:apply|bind|call)\s*)?\()/,number:{pattern:RegExp(/(^|[^\w$])/.source+"(?:"+/NaN|Infinity/.source+"|"+/0[bB][01]+(?:_[01]+)*n?/.source+"|"+/0[oO][0-7]+(?:_[0-7]+)*n?/.source+"|"+/0[xX][\dA-Fa-f]+(?:_[\dA-Fa-f]+)*n?/.source+"|"+/\d+(?:_\d+)*n/.source+"|"+/(?:\d+(?:_\d+)*(?:\.(?:\d+(?:_\d+)*)?)?|\.\d+(?:_\d+)*)(?:[Ee][+-]?\d+(?:_\d+)*)?/.source+")"+/(?![\w$])/.source),lookbehind:!0},operator:/--|\+\+|\*\*=?|=>|&&=?|\|\|=?|[!=]==|<<=?|>>>?=?|[-+*/%&|^!=<>]=?|\.{3}|\?\?=?|\?\.?|[~:]/}),n.languages.javascript["class-name"][0].pattern=/(\b(?:class|extends|implements|instanceof|interface|new)\s+)[\w.\\]+/,n.languages.insertBefore("javascript","keyword",{regex:{pattern:RegExp(/((?:^|[^$\w\xA0-\uFFFF."'\])\s]|\b(?:return|yield))\s*)/.source+/\//.source+"(?:"+/(?:\[(?:[^\]\\\r\n]|\\.)*\]|\\.|[^/\\\[\r\n])+\/[dgimyus]{0,7}/.source+"|"+/(?:\[(?:[^[\]\\\r\n]|\\.|\[(?:[^[\]\\\r\n]|\\.|\[(?:[^[\]\\\r\n]|\\.)*\])*\])*\]|\\.|[^/\\\[\r\n])+\/[dgimyus]{0,7}v[dgimyus]{0,7}/.source+")"+/(?=(?:\s|\/\*(?:[^*]|\*(?!\/))*\*\/)*(?:$|[\r\n,.;:})\]]|\/\/))/.source),lookbehind:!0,greedy:!0,inside:{"regex-source":{pattern:/^(\/)[\s\S]+(?=\/[a-z]*$)/,lookbehind:!0,alias:"language-regex",inside:n.languages.regex},"regex-delimiter":/^\/|\/$/,"regex-flags":/^[a-z]+$/}},"function-variable":{pattern:/#?(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\s*[=:]\s*(?:async\s*)?(?:\bfunction\b|(?:\((?:[^()]|\([^()]*\))*\)|(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*)\s*=>))/,alias:"function"},parameter:[{pattern:/(function(?:\s+(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*)?\s*\(\s*)(?!\s)(?:[^()\s]|\s+(?![\s)])|\([^()]*\))+(?=\s*\))/,lookbehind:!0,inside:n.languages.javascript},{pattern:/(^|[^$\w\xA0-\uFFFF])(?!\s)[_$a-z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\s*=>)/i,lookbehind:!0,inside:n.languages.javascript},{pattern:/(\(\s*)(?!\s)(?:[^()\s]|\s+(?![\s)])|\([^()]*\))+(?=\s*\)\s*=>)/,lookbehind:!0,inside:n.languages.javascript},{pattern:/((?:\b|\s|^)(?!(?:as|async|await|break|case|catch|class|const|continue|debugger|default|delete|do|else|enum|export|extends|finally|for|from|function|get|if|implements|import|in|instanceof|interface|let|new|null|of|package|private|protected|public|return|set|static|super|switch|this|throw|try|typeof|undefined|var|void|while|with|yield)(?![$\w\xA0-\uFFFF]))(?:(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*\s*)\(\s*|\]\s*\(\s*)(?!\s)(?:[^()\s]|\s+(?![\s)])|\([^()]*\))+(?=\s*\)\s*\{)/,lookbehind:!0,inside:n.languages.javascript}],constant:/\b[A-Z](?:[A-Z_]|\dx?)*\b/}),n.languages.insertBefore("javascript","string",{hashbang:{pattern:/^#!.*/,greedy:!0,alias:"comment"},"template-string":{pattern:/`(?:\\[\s\S]|\$\{(?:[^{}]|\{(?:[^{}]|\{[^}]*\})*\})+\}|(?!\$\{)[^\\`])*`/,greedy:!0,inside:{"template-punctuation":{pattern:/^`|`$/,alias:"string"},interpolation:{pattern:/((?:^|[^\\])(?:\\{2})*)\$\{(?:[^{}]|\{(?:[^{}]|\{[^}]*\})*\})+\}/,lookbehind:!0,inside:{"interpolation-punctuation":{pattern:/^\$\{|\}$/,alias:"punctuation"},rest:n.languages.javascript}},string:/[\s\S]+/}},"string-property":{pattern:/((?:^|[,{])[ \t]*)(["'])(?:\\(?:\r\n|[\s\S])|(?!\2)[^\\\r\n])*\2(?=\s*:)/m,lookbehind:!0,greedy:!0,alias:"property"}}),n.languages.insertBefore("javascript","operator",{"literal-property":{pattern:/((?:^|[,{])[ \t]*)(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\s*:)/m,lookbehind:!0,alias:"property"}}),n.languages.markup&&(n.languages.markup.tag.addInlined("script","javascript"),n.languages.markup.tag.addAttribute(/on(?:abort|blur|change|click|composition(?:end|start|update)|dblclick|error|focus(?:in|out)?|key(?:down|up)|load|mouse(?:down|enter|leave|move|out|over|up)|reset|resize|scroll|select|slotchange|submit|unload|wheel)/.source,"javascript")),n.languages.js=n.languages.javascript,function(){if(void 0!==n&&"undefined"!=typeof document){Element.prototype.matches||(Element.prototype.matches=Element.prototype.msMatchesSelector||Element.prototype.webkitMatchesSelector);var e={js:"javascript",py:"python",rb:"ruby",ps1:"powershell",psm1:"powershell",sh:"bash",bat:"batch",h:"c",tex:"latex"},t='pre[data-src]:not([data-src-status="loaded"]):not([data-src-status="loading"])';n.hooks.add("before-highlightall",(function(e){e.selector+=", "+t})),n.hooks.add("before-sanity-check",(function(r){var i=r.element;if(i.matches(t)){r.code="",i.setAttribute("data-src-status","loading");var o=i.appendChild(document.createElement("CODE"));o.textContent="Loading…";var a=i.getAttribute("data-src"),s=r.language;if("none"===s){var l=(/\.(\w+)$/.exec(a)||[,"none"])[1];s=e[l]||l}n.util.setLanguage(o,s),n.util.setLanguage(i,s);var c=n.plugins.autoloader;c&&c.loadLanguages(s),function(e,t,n){var r=new XMLHttpRequest;r.open("GET",e,!0),r.onreadystatechange=function(){var e,i;4==r.readyState&&(r.status<400&&r.responseText?t(r.responseText):r.status>=400?n((e=r.status,i=r.statusText,"✖ Error "+e+" while fetching file: "+i)):n("✖ Error: File does not exist or is empty"))},r.send(null)}(a,(function(e){i.setAttribute("data-src-status","loaded");var t=function(e){var t=/^\s*(\d+)\s*(?:(,)\s*(?:(\d+)\s*)?)?$/.exec(e||"");if(t){var n=Number(t[1]),r=t[2],i=t[3];return r?i?[n,Number(i)]:[n,void 0]:[n,n]}}(i.getAttribute("data-range"));if(t){var r=e.split(/\r\n?|\n/g),a=t[0],s=null==t[1]?r.length:t[1];a<0&&(a+=r.length),a=Math.max(0,Math.min(a-1,r.length)),s<0&&(s+=r.length),s=Math.max(0,Math.min(s,r.length)),e=r.slice(a,s).join("\n"),i.hasAttribute("data-start")||i.setAttribute("data-start",String(a+1))}o.textContent=e,n.highlightElement(o)}),(function(e){i.setAttribute("data-src-status","failed"),o.textContent=e}))}})),n.plugins.fileHighlight={highlight:function(e){for(var r,i=(e||document).querySelectorAll(t),o=0;r=i[o++];)n.highlightElement(r)}};var r=!1;n.fileHighlight=function(){r||(console.warn("Prism.fileHighlight is deprecated. Use `Prism.plugins.fileHighlight.highlight` instead."),r=!0),n.plugins.fileHighlight.highlight.apply(this,arguments)}}}()}).call(this,n(35))},function(e,t,n){var r=n(47),i=n(23);e.exports=function(e){if(!i(e))return!1;var t=r(e);return"[object Function]"==t||"[object GeneratorFunction]"==t||"[object AsyncFunction]"==t||"[object Proxy]"==t}},function(e,t,n){var r=n(78),i=n(79);e.exports=function(e,t,n,o){var a=!n;n||(n={});for(var s=-1,l=t.length;++s-1&&e%1==0&&e=0||(s.push(i[c]),a[i[c]]=this[i[c]]);return Object(r.r)({},a,e)},e}()},function(e,t,n){"use strict";n.d(t,"b",(function(){return m})),n.d(t,"a",(function(){return v}));var r=n(0),i=n(1),o=n(7),a=n(3),s=n(15),l=n(6);function c(e){return e.name}function u(e){return e.self.$$state=function(){return e},e.self}function d(e){return e.parent&&e.parent.data&&(e.data=e.self.data=Object(r.A)(e.parent.data,e.data)),e.data}var p=function(e,t){return function(n){var o=n.self;if(o&&o.url&&o.name&&o.name.match(/\.\*\*$/)){var a={};Object(r.m)(o,a),a.url+="{remainder:any}",o=a}var s=n.parent,l=function(e){if(!Object(i.l)(e))return!1;var t="^"===e.charAt(0);return{val:t?e.substring(1):e,root:t}}(o.url),c=l?e.compile(l.val,{state:o}):o.url;if(!c)return null;if(!e.isMatcher(c))throw new Error("Invalid url '"+c+"' in state '"+n+"'");return l&&l.root?c:(s&&s.navigable||t()).url.append(c)}},f=function(e){return function(t){return!e(t)&&t.url?t:t.parent?t.parent.navigable:null}};function h(e){return e.parent?e.parent.path.concat(e):[e]}function g(e){var t=e.parent?Object(r.r)({},e.parent.includes):{};return t[e.name]=!0,t}function m(e){var t,n,c=function(e){return e.provide||e.token},u=Object(a.l)([[Object(a.n)("resolveFn"),function(e){return new s.a(c(e),e.resolveFn,e.deps,e.policy)}],[Object(a.n)("useFactory"),function(e){return new s.a(c(e),e.useFactory,e.deps||e.dependencies,e.policy)}],[Object(a.n)("useClass"),function(e){return new s.a(c(e),(function(){return new e.useClass}),[],e.policy)}],[Object(a.n)("useValue"),function(e){return new s.a(c(e),(function(){return e.useValue}),[],e.policy,e.useValue)}],[Object(a.n)("useExisting"),function(e){return new s.a(c(e),r.y,[e.useExisting],e.policy)}]]),d=Object(a.l)([[Object(a.m)(Object(a.n)("val"),i.l),function(e){return new s.a(e.token,r.y,[e.val],e.policy)}],[Object(a.m)(Object(a.n)("val"),i.a),function(e){return new s.a(e.token,Object(r.P)(e.val),e.val.slice(0,-1),e.policy)}],[Object(a.m)(Object(a.n)("val"),i.d),function(e){return new s.a(e.token,e.val,(t=e.val,n=l.b.$injector,t.$inject||n&&n.annotate(t,n.strictDi)||"deferred"),e.policy);var t,n}]]),p=Object(a.l)([[Object(a.h)(s.a),function(e){return e}],[function(e){return!(!e.token||!e.resolveFn)},u],[function(e){return!(!e.provide&&!e.token||!(e.useValue||e.useFactory||e.useExisting||e.useClass))},u],[function(e){return!!(e&&e.val&&(Object(i.l)(e.val)||Object(i.a)(e.val)||Object(i.d)(e.val)))},d],[Object(a.p)(!0),function(e){throw new Error("Invalid resolve value: "+Object(o.m)(e))}]]),f=e.resolve;return(Object(i.a)(f)?f:(t=f,n=e.resolvePolicy||{},Object.keys(t||{}).map((function(e){return{token:e,val:t[e],deps:void 0,policy:n[e]}})))).map(p)}var v=function(){function e(e,t){this.matcher=e;var n,i=this,o=function(){return e.find("")},s=function(e){return""===e.name};this.builders={name:[c],self:[u],parent:[function(t){return s(t)?null:e.find(i.parentName(t))||o()}],data:[d],url:[p(t,o)],navigable:[f(s)],params:[(n=t.paramFactory,function(e){var t=e.url&&e.url.parameters({inherit:!1})||[],i=Object(r.U)(Object(r.C)(Object(r.F)(e.params||{},t.map(Object(a.n)("id"))),(function(t,r){return n.fromConfig(r,null,e.self)})));return t.concat(i).map((function(e){return[e.id,e]})).reduce(r.h,{})})],views:[],path:[h],includes:[g],resolvables:[m]}}return e.prototype.builder=function(e,t){var n=this.builders,r=n[e]||[];return Object(i.l)(e)&&!Object(i.c)(t)?r.length>1?r:r[0]:Object(i.l)(e)&&Object(i.d)(t)?(n[e]=r,n[e].push(t),function(){return n[e].splice(n[e].indexOf(t,1))&&null}):void 0},e.prototype.build=function(e){var t=this.matcher,n=this.builders,i=this.parentName(e);if(i&&!t.find(i,void 0,!1))return null;for(var o in n)if(n.hasOwnProperty(o)){var a=n[o].reduce((function(e,t){return function(n){return t(n,e)}}),r.E);e[o]=a(e)}return e},e.prototype.parentName=function(e){var t=e.name||"",n=t.split(".");if("**"===n.pop()&&n.pop(),n.length){if(e.parent)throw new Error("States that specify the 'parent:' property should not have a '.' in their name ("+t+")");return n.join(".")}return e.parent?Object(i.l)(e.parent)?e.parent:e.parent.name:""},e.prototype.name=function(e){var t=e.name;if(-1!==t.indexOf(".")||!e.parent)return t;var n=Object(i.l)(e.parent)?e.parent:e.parent.name;return n?n+"."+t:t},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return a}));var r=n(1),i=n(0),o=n(19),a=function(){function e(e){this._states=e}return e.prototype.isRelative=function(e){return 0===(e=e||"").indexOf(".")||0===e.indexOf("^")},e.prototype.find=function(e,t,n){if(void 0===n&&(n=!0),e||""===e){var a=Object(r.l)(e),s=a?e:e.name;this.isRelative(s)&&(s=this.resolvePath(s,t));var l=this._states[s];if(l&&(a||!(a||l!==e&&l.self!==e)))return l;if(a&&n){var c=Object(i.U)(this._states).filter((function(e){return e.__stateObjectCache.nameGlob&&e.__stateObjectCache.nameGlob.matches(s)}));return c.length>1&&o.a.error("stateMatcher.find: Found multiple matches for "+s+" using glob: ",c.map((function(e){return e.name}))),c[0]}}},e.prototype.resolvePath=function(e,t){if(!t)throw new Error("No reference point given for path '"+e+"'");for(var n=this.find(t),r=e.split("."),i=r.length,o=0,a=n;o0;){var c=t.shift(),u=c.name,d=r.build(c),p=o.indexOf(c);if(d){var f=s(u);if(f&&f.name===u)throw new Error("State '"+u+"' is already defined");var h=s(u+".**");h&&this.router.stateRegistry.deregister(h),n[u]=c,this.attachRoute(c),p>=0&&o.splice(p,1),i.push(c)}else{var g=a[u];if(a[u]=t.length,p>=0&&g===t.length)return t.push(c),l(),n;p<0&&o.push(c),t.push(c)}}return l(),n},e.prototype.attachRoute=function(e){if(!e.abstract&&e.url){var t=this.router.urlService.rules;t.rule(t.urlRuleFactory.create(e))}},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return l}));var r=n(94),i=n(93),o=n(95),a=n(0),s=n(3),l=function(){function e(e){this.router=e,this.states={},this.listeners=[],this.matcher=new r.a(this.states),this.builder=new i.a(this.matcher,e.urlMatcherFactory),this.stateQueue=new o.a(e,this.states,this.builder,this.listeners),this._registerRoot()}return e.prototype._registerRoot=function(){(this._root=this.stateQueue.register({name:"",url:"^",views:null,params:{"#":{value:null,type:"hash",dynamic:!0}},abstract:!0})).navigable=null},e.prototype.dispose=function(){var e=this;this.stateQueue.dispose(),this.listeners=[],this.get().forEach((function(t){return e.get(t)&&e.deregister(t)}))},e.prototype.onStatesChanged=function(e){return this.listeners.push(e),function(){Object(a.L)(this.listeners)(e)}.bind(this)},e.prototype.root=function(){return this._root},e.prototype.register=function(e){return this.stateQueue.register(e)},e.prototype._deregisterTree=function(e){var t=this,n=this.get().map((function(e){return e.$$state()})),r=function(e){var t=n.filter((function(t){return-1!==e.indexOf(t.parent)}));return 0===t.length?t:t.concat(r(t))},i=r([e]),o=[e].concat(i).reverse();return o.forEach((function(e){var n=t.router.urlService.rules;n.rules().filter(Object(s.o)("state",e)).forEach((function(e){return n.removeRule(e)})),delete t.states[e.name]})),o},e.prototype.deregister=function(e){var t=this.get(e);if(!t)throw new Error("Can't deregister state; not found: "+e);var n=this._deregisterTree(t.$$state());return this.listeners.forEach((function(e){return e("deregistered",n.map((function(e){return e.self})))})),n},e.prototype.get=function(e,t){var n=this;if(0===arguments.length)return Object.keys(this.states).map((function(e){return n.states[e].self}));var r=this.matcher.find(e,t);return r&&r.self||null},e.prototype.decorator=function(e,t){return this.builder.builder(e,t)},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return v}));var r=n(0),i=n(1),o=n(50),a=n(6),s=n(18),l=n(52),c=n(56),u=n(9),d=n(26),p=n(16),f=n(62),h=n(27),g=n(115),m=n(3),v=function(){function e(t){this.router=t,this.invalidCallbacks=[],this._defaultErrorHandler=function(e){e instanceof Error&&e.stack?(console.error(e),console.error(e.stack)):e instanceof u.b?(console.error(e.toString()),e.detail&&e.detail.stack&&console.error(e.detail.stack)):console.error(e)};var n=Object.keys(e.prototype).filter(Object(m.i)(Object(r.z)(["current","$current","params","transition"])));Object(r.n)(Object(m.p)(e.prototype),this,Object(m.p)(this),n)}return Object.defineProperty(e.prototype,"transition",{get:function(){return this.router.globals.transition},enumerable:!1,configurable:!0}),Object.defineProperty(e.prototype,"params",{get:function(){return this.router.globals.params},enumerable:!1,configurable:!0}),Object.defineProperty(e.prototype,"current",{get:function(){return this.router.globals.current},enumerable:!1,configurable:!0}),Object.defineProperty(e.prototype,"$current",{get:function(){return this.router.globals.$current},enumerable:!1,configurable:!0}),e.prototype.dispose=function(){this.defaultErrorHandler(r.E),this.invalidCallbacks=[]},e.prototype._handleInvalidTargetState=function(e,t){var n=this,r=s.a.makeTargetState(this.router.stateRegistry,e),i=this.router.globals,l=function(){return i.transitionHistory.peekTail()},c=l(),p=new o.a(this.invalidCallbacks.slice()),f=new h.b(e).injector(),g=function(e){if(e instanceof d.a){var t=e;return(t=n.target(t.identifier(),t.params(),t.options())).valid()?l()!==c?u.b.superseded().toPromise():n.transitionTo(t.identifier(),t.params(),t.options()):u.b.invalid(t.error()).toPromise()}};return function e(){var n=p.dequeue();return void 0===n?u.b.invalid(t.error()).toPromise():a.b.$q.when(n(t,r,f)).then(g).then((function(t){return t||e()}))}()},e.prototype.onInvalid=function(e){return this.invalidCallbacks.push(e),function(){Object(r.L)(this.invalidCallbacks)(e)}.bind(this)},e.prototype.reload=function(e){return this.transitionTo(this.current,this.params,{reload:!Object(i.c)(e)||e,inherit:!1,notify:!1})},e.prototype.go=function(e,t,n){var i={relative:this.$current,inherit:!0},o=Object(r.o)(n,i,c.b);return this.transitionTo(e,t,o)},e.prototype.target=function(e,t,n){if(void 0===n&&(n={}),Object(i.i)(n.reload)&&!n.reload.name)throw new Error("Invalid reload state object");var r=this.router.stateRegistry;if(n.reloadState=!0===n.reload?r.root():r.matcher.find(n.reload,n.relative),n.reload&&!n.reloadState)throw new Error("No such reload state '"+(Object(i.l)(n.reload)?n.reload:n.reload.name)+"'");return new d.a(this.router.stateRegistry,e,t,n)},e.prototype.getCurrentPath=function(){var e=this,t=this.router.globals.successfulTransitions.peekTail();return t?t.treeChanges().to:[new l.a(e.router.stateRegistry.root())]},e.prototype.transitionTo=function(e,t,n){var i=this;void 0===t&&(t={}),void 0===n&&(n={});var o=this.router,s=o.globals;n=Object(r.o)(n,c.b);var l=function(){return s.transition};n=Object(r.r)(n,{current:l});var p=this.target(e,t,n),f=this.getCurrentPath();if(!p.exists())return this._handleInvalidTargetState(f,p);if(!p.valid())return Object(r.O)(p.error());if(!1===n.supercede&&l())return u.b.ignored("Another transition is in progress and supercede has been set to false in TransitionOptions for the transition. So the transition was ignored in favour of the existing one in progress.").toPromise();var h=function(e){return function(t){if(t instanceof u.b){var n=o.globals.lastStartedTransitionId<=e.$id;if(t.type===u.a.IGNORED)return n&&o.urlRouter.update(),a.b.$q.when(s.current);var r=t.detail;if(t.type===u.a.SUPERSEDED&&t.redirected&&r instanceof d.a){var l=e.redirect(r);return l.run().catch(h(l))}if(t.type===u.a.ABORTED)return n&&o.urlRouter.update(),a.b.$q.reject(t)}return i.defaultErrorHandler()(t),a.b.$q.reject(t)}},g=this.router.transitionService.create(f,p),m=g.run().catch(h(g));return Object(r.N)(m),Object(r.r)(m,{transition:g})},e.prototype.is=function(e,t,n){n=Object(r.o)(n,{relative:this.$current});var o=this.router.stateRegistry.matcher.find(e,n.relative);if(Object(i.c)(o)){if(this.$current!==o)return!1;if(!t)return!0;var a=o.parameters({inherit:!0,matchingKeys:t});return p.b.equals(a,p.b.values(a,t),this.params)}},e.prototype.includes=function(e,t,n){n=Object(r.o)(n,{relative:this.$current});var o=Object(i.l)(e)&&f.a.fromString(e);if(o){if(!o.matches(this.$current.name))return!1;e=this.$current.name}var a=this.router.stateRegistry.matcher.find(e,n.relative),s=this.$current.includes;if(Object(i.c)(a)){if(!Object(i.c)(s[a.name]))return!1;if(!t)return!0;var l=a.parameters({inherit:!0,matchingKeys:t});return p.b.equals(l,p.b.values(l,t),this.params)}},e.prototype.href=function(e,t,n){var o={lossy:!0,inherit:!0,absolute:!1,relative:this.$current};n=Object(r.o)(n,o),t=t||{};var a=this.router.stateRegistry.matcher.find(e,n.relative);if(!Object(i.c)(a))return null;n.inherit&&(t=this.params.$inherit(t,this.$current,a));var s=a&&n.lossy?a.navigable:a;return s&&void 0!==s.url&&null!==s.url?this.router.urlRouter.href(s.url,t,{absolute:n.absolute}):null},e.prototype.defaultErrorHandler=function(e){return this._defaultErrorHandler=e||this._defaultErrorHandler},e.prototype.get=function(e,t){var n=this.router.stateRegistry;return 0===arguments.length?n.get():n.get(e,t||this.$current)},e.prototype.lazyLoad=function(e,t){var n=this.get(e);if(!n||!n.lazyLoad)throw new Error("Can not lazy load "+e);var r=this.getCurrentPath(),i=s.a.makeTargetState(this.router.stateRegistry,r);return t=t||this.router.transitionService.create(r,i),Object(g.a)(t,n)},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return s}));var r=n(0),i=n(1),o=n(14),a=n(17),s=function(){function e(e){this.transition=e}return e.prototype.buildHooksForPhase=function(e){var t=this;return this.transition.router.transitionService._pluginapi._getEvents(e).map((function(e){return t.buildHooks(e)})).reduce(r.T,[]).filter(r.y)},e.prototype.buildHooks=function(e){var t=this.transition,n=t.treeChanges(),i=this.getMatchingHooks(e,n,t);if(!i)return[];var s={transition:t,current:t.options().current};return i.map((function(i){return i.matches(n,t)[e.criteriaMatchPath.name].map((function(n){var l=Object(r.r)({bind:i.bind,traceData:{hookType:e.name,context:n}},s),c=e.criteriaMatchPath.scope===o.b.STATE?n.state.self:null,u=new a.a(t,c,i,l);return{hook:i,node:n,transitionHook:u}}))})).reduce(r.T,[]).sort(function(e){void 0===e&&(e=!1);return function(t,n){var r=e?-1:1,i=(t.node.state.path.length-n.node.state.path.length)*r;return 0!==i?i:n.hook.priority-t.hook.priority}}(e.reverseSort)).map((function(e){return e.transitionHook}))},e.prototype.getMatchingHooks=function(e,t,n){var a=e.hookPhase===o.a.CREATE,s=this.transition.router.transitionService;return(a?[s]:[this.transition,s]).map((function(t){return t.getHooks(e.name)})).filter(Object(r.l)(i.a,"broken event named: "+e.name)).reduce(r.T,[]).filter((function(e){return e.matches(t,n)}))},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return b}));var r=n(100),i=n(101),o=n(56),a=n(102),s=n(96),l=n(97),c=n(103),u=n(0),d=n(1),p=n(104),f=n(10),h=n(4),g=0,m=Object(h.gb)("LocationServices",["url","path","search","hash","onChange"]),v=Object(h.gb)("LocationConfig",["port","protocol","host","baseHref","html5Mode","hashPrefix"]),b=function(){function e(e,t){void 0===e&&(e=m),void 0===t&&(t=v),this.locationService=e,this.locationConfig=t,this.$id=g++,this._disposed=!1,this._disposables=[],this.trace=f.c,this.viewService=new a.a(this),this.globals=new c.a,this.transitionService=new o.a(this),this.urlMatcherFactory=new r.b(this),this.urlRouter=new i.a(this),this.urlService=new p.a(this),this.stateRegistry=new s.a(this),this.stateService=new l.a(this),this._plugins={},this.viewService._pluginapi._rootViewContext(this.stateRegistry.root()),this.globals.$current=this.stateRegistry.root(),this.globals.current=this.globals.$current.self,this.disposable(this.globals),this.disposable(this.stateService),this.disposable(this.stateRegistry),this.disposable(this.transitionService),this.disposable(this.urlService),this.disposable(e),this.disposable(t)}return e.prototype.disposable=function(e){this._disposables.push(e)},e.prototype.dispose=function(e){var t=this;e&&Object(d.d)(e.dispose)?e.dispose(this):(this._disposed=!0,this._disposables.slice().forEach((function(e){try{"function"==typeof e.dispose&&e.dispose(t),Object(u.L)(t._disposables,e)}catch(e){}})))},e.prototype.plugin=function(e,t){void 0===t&&(t={});var n=new e(this,t);if(!n.name)throw new Error("Required property `name` missing on plugin: "+n);return this._disposables.push(n),this._plugins[n.name]=n},e.prototype.getPlugin=function(e){return e?this._plugins[e]:Object(u.U)(this._plugins)},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return s})),n.d(t,"b",(function(){return l}));var r=n(4),i=n(32),o=n(28),a=function(){return(a=Object.assign||function(e){for(var t,n=1,r=arguments.length;n Registering",e),this._viewConfigs.push(e)},e.prototype.sync=function(){var t=this,n=this._uiViews.map((function(e){return[e.fqn,e]})).reduce(r.h,{});function o(e){for(var t=e.viewDecl.$context,n=0;++n&&t.parent;)t=t.parent;return n}var s=Object(i.e)((function(e,t,n,r){return t*(e(n)-e(r))})),l=this._uiViews.sort(s((function(e){var t=function(e){return e&&e.parent?t(e.parent)+1:1};return 1e4*e.fqn.split(".").length+t(e.creationContext)}),1)).map((function(r){var i=t._viewConfigs.filter(e.matches(n,r));return i.length>1&&i.sort(s(o,-1)),{uiView:r,viewConfig:i[0]}})),c=l.map((function(e){return e.viewConfig})),u=this._viewConfigs.filter((function(e){return!Object(r.z)(c,e)})).map((function(e){return{uiView:void 0,viewConfig:e}}));l.forEach((function(e){-1!==t._uiViews.indexOf(e.uiView)&&e.uiView.configUpdated(e.viewConfig)}));var d=l.concat(u);this._listeners.forEach((function(e){return e(d)})),a.c.traceViewSync(d)},e.prototype.registerUIView=function(e){a.c.traceViewServiceUIViewEvent("-> Registering",e);var t=this._uiViews;return t.filter((function(t){return t.fqn===e.fqn&&t.$type===e.$type})).length&&a.c.traceViewServiceUIViewEvent("!!!! duplicate uiView named:",e),t.push(e),this.sync(),function(){-1!==t.indexOf(e)?(a.c.traceViewServiceUIViewEvent("<- Deregistering",e),Object(r.L)(t)(e)):a.c.traceViewServiceUIViewEvent("Tried removing non-registered uiView",e)}},e.prototype.available=function(){return this._uiViews.map(Object(i.n)("fqn"))},e.prototype.active=function(){return this._uiViews.filter(Object(i.n)("$config")).map(Object(i.n)("name"))},e.matches=function(e,t){return function(n){if(t.$type!==n.viewDecl.$type)return!1;var i=n.viewDecl,o=i.$uiViewName.split("."),a=t.fqn.split(".");if(!Object(r.q)(o,a.slice(0-o.length)))return!1;var s=1-o.length||void 0,l=a.slice(0,s).join("."),c=e[l].creationContext;return i.$uiViewContextAnchor===(c&&c.name)}},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return o}));var r=n(92),i=n(50),o=function(){function e(){this.params=new r.a,this.lastStartedTransitionId=-1,this.transitionHistory=new i.a([],1),this.successfulTransitions=new i.a([],1)}return e.prototype.dispose=function(){this.transitionHistory.clear(),this.successfulTransitions.clear(),this.transition=null},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return s}));var r=n(4),i=n(105),o=n(106),a=n(43),s=function(){function e(e){var t=this;this.router=e,this.interceptDeferred=!1,this.rules=new i.a(this.router),this.config=new o.a(this.router),this.url=function(e,n,r){return t.router.locationService.url(e,n,r)},this.path=function(){return t.router.locationService.path()},this.search=function(){return t.router.locationService.search()},this.hash=function(){return t.router.locationService.hash()},this.onChange=function(e){return t.router.locationService.onChange(e)}}return e.prototype.dispose=function(){this.listen(!1),this.rules.dispose()},e.prototype.parts=function(){return{path:this.path(),search:this.search(),hash:this.hash()}},e.prototype.sync=function(e){if(!e||!e.defaultPrevented){var t=this.router,n=t.urlService,i=t.stateService,o={path:n.path(),search:n.search(),hash:n.hash()},s=this.match(o);Object(r.sb)([[r.cb,function(e){return n.url(e,!0)}],[a.TargetState.isDef,function(e){return i.go(e.state,e.params,e.options)}],[Object(r.Q)(a.TargetState),function(e){return i.go(e.state(),e.params(),e.options())}]])(s&&s.rule.handler(s.match,o,this.router))}},e.prototype.listen=function(e){var t=this;if(!1!==e)return this._stopListeningFn=this._stopListeningFn||this.router.urlService.onChange((function(e){return t.sync(e)}));this._stopListeningFn&&this._stopListeningFn(),delete this._stopListeningFn},e.prototype.deferIntercept=function(e){void 0===e&&(e=!0),this.interceptDeferred=e},e.prototype.match=function(e){var t=this;e=Object(r.C)({path:"",search:{},hash:""},e);for(var n,i,o,a=this.rules.rules(),s=0;sn.weight?l:n}return n},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return c}));var r,i=n(43),o=n(32),a=n(4),s=n(54);function l(e){if(!(Object(a.U)(e)||Object(a.cb)(e)||Object(a.Q)(i.TargetState)(e)||i.TargetState.isDef(e)))throw new Error("'handler' must be a string, function, TargetState, or have a state: 'newtarget' property");return Object(a.U)(e)?e:Object(a.Sb)(e)}r=function(e,t){var n=function(e,t){return(t.priority||0)-(e.priority||0)}(e,t);return 0!==n||0!==(n=function(e,t){var n={STATE:4,URLMATCHER:4,REGEXP:3,RAW:2,OTHER:1};return(n[e.type]||0)-(n[t.type]||0)}(e,t))||0!==(n=function(e,t){return e.urlMatcher&&t.urlMatcher?o.a.compare(e.urlMatcher,t.urlMatcher):0}(e,t))?n:function(e,t){var n={STATE:!0,URLMATCHER:!0};return n[e.type]&&n[t.type]?0:(e.$id||0)-(t.$id||0)}(e,t)};var c=function(){function e(e){this.router=e,this._sortFn=r,this._rules=[],this._id=0,this.urlRuleFactory=new s.b(e)}return e.prototype.dispose=function(e){this._rules=[],delete this._otherwiseFn},e.prototype.initial=function(e){var t=l(e);this.rule(this.urlRuleFactory.create((function(e,t){return 0===t.globals.transitionHistory.size()&&!!/^\/?$/.exec(e.path)}),t))},e.prototype.otherwise=function(e){var t=l(e);this._otherwiseFn=this.urlRuleFactory.create(Object(a.Sb)(!0),t),this._sorted=!1},e.prototype.removeRule=function(e){Object(a.Ab)(this._rules,e)},e.prototype.rule=function(e){var t=this;if(!s.b.isUrlRule(e))throw new Error("invalid rule");return e.$id=this._id++,e.priority=e.priority||0,this._rules.push(e),this._sorted=!1,function(){return t.removeRule(e)}},e.prototype.rules=function(){return this.ensureSorted(),this._rules.concat(this._otherwiseFn?[this._otherwiseFn]:[])},e.prototype.sort=function(e){for(var t=this.stableSort(this._rules,this._sortFn=e||this._sortFn),n=0,r=0;r")},e.prototype.fromString=function(e,t){return Object(i.isFunction)(e)?e(t):e},e.prototype.fromUrl=function(e,t){return Object(i.isFunction)(e)&&(e=e(t)),null==e?null:this._useHttp?this.$http.get(e,{cache:this.$templateCache,headers:{Accept:"text/html"}}).then((function(e){return e.data})):this.$templateRequest(e)},e.prototype.fromProvider=function(e,t,n){var r=i.services.$injector.annotate(e),o=Object(i.isArray)(e)?Object(i.tail)(e):e;return new i.Resolvable("",o,r).get(n)},e.prototype.fromComponentProvider=function(e,t,n){var r=i.services.$injector.annotate(e),o=Object(i.isArray)(e)?Object(i.tail)(e):e;return new i.Resolvable("",o,r).get(n)},e.prototype.makeComponentTemplate=function(e,t,n,o){o=o||{};var a=r.a.version.minor>=3?"::":"",l=function(e){var t=Object(i.kebobString)(e);return/^(x|data)-/.exec(t)?"x-"+t:t},c=function(e){var t=i.services.$injector.get(e+"Directive");if(!t||!t.length)throw new Error("Unable to find component named '"+e+"'");return t.map(s).reduce(i.unnestR,[])}(n).map((function(n){var r=n.name,s=n.type,c=l(r);if(e.attr(c)&&!o[r])return c+"='"+e.attr(c)+"'";var u=o[r]||r;if("@"===s)return c+"='{{"+a+"$resolve."+u+"}}'";if("&"===s){var d=t.getResolvable(u),p=d&&d.data,f=p&&i.services.$injector.annotate(p)||[];return c+"='$resolve."+u+(Object(i.isArray)(p)?"["+(p.length-1)+"]":"")+"("+f.join(",")+")'"}return c+"='"+a+"$resolve."+u+"'"})).join(" "),u=l(n);return"<"+u+" "+c+">"},e}();var s=function(e){return Object(i.isObject)(e.bindToController)?l(e.bindToController):l(e.scope)},l=function(e){return Object.keys(e||{}).map((function(t){return[t,/^([=<@&])[?]?(.*)/.exec(e[t])]})).filter((function(e){return Object(i.isDefined)(e)&&Object(i.isArray)(e[1])})).map((function(e){return{name:e[1][2]||e[0],type:e[1][1]}}))},c=n(112),u=function(e){return function(t){var n=t[e],r="onExit"===e?"from":"to";return n?function(e,t){var o=new i.ResolveContext(e.treeChanges(r)).subContext(t.$$state()),a=Object(i.extend)(A(o),{$state$:t,$transition$:e});return i.services.$injector.invoke(n,this,a)}:void 0}},d=function(){function e(e){this._urlListeners=[],this.$locationProvider=e;var t=Object(i.val)(e);Object(i.createProxyFunctions)(t,this,t,["hashPrefix"])}return e.monkeyPatchPathParameterType=function(e){var t=e.urlMatcherFactory.type("path");t.encode=function(e){return null!=e?e.toString().replace(/(~|\/)/g,(function(e){return{"~":"~~","/":"~2F"}[e]})):e},t.decode=function(e){return null!=e?e.toString().replace(/(~~|~2F)/g,(function(e){return{"~~":"~","~2F":"/"}[e]})):e}},e.prototype.dispose=function(){},e.prototype.onChange=function(e){var t=this;return this._urlListeners.push(e),function(){return Object(i.removeFrom)(t._urlListeners)(e)}},e.prototype.html5Mode=function(){var e=this.$locationProvider.html5Mode();return(e=Object(i.isObject)(e)?e.enabled:e)&&this.$sniffer.history},e.prototype.baseHref=function(){return this._baseHref||(this._baseHref=this.$browser.baseHref()||this.$window.location.pathname)},e.prototype.url=function(e,t,n){return void 0===t&&(t=!1),Object(i.isDefined)(e)&&this.$location.url(e),t&&this.$location.replace(),n&&this.$location.state(n),this.$location.url()},e.prototype._runtimeServices=function(e,t,n,r,o){var a=this;this.$location=t,this.$sniffer=n,this.$browser=r,this.$window=o,e.$on("$locationChangeSuccess",(function(e){return a._urlListeners.forEach((function(t){return t(e)}))}));var s=Object(i.val)(t);Object(i.createProxyFunctions)(s,this,s,["replace","path","search","hash"]),Object(i.createProxyFunctions)(s,this,s,["port","protocol","host"])},e}(),p=n(113);r.a.module("ui.router.angular1",[]);var f=r.a.module("ui.router.init",["ng"]),h=r.a.module("ui.router.util",["ui.router.init"]),g=r.a.module("ui.router.router",["ui.router.util"]),m=r.a.module("ui.router.state",["ui.router.router","ui.router.util","ui.router.angular1"]),v=r.a.module("ui.router",["ui.router.init","ui.router.state","ui.router.angular1"]),b=(r.a.module("ui.router.compat",["ui.router"]),null);function y(e){(b=this.router=new i.UIRouter).stateProvider=new c.a(b.stateRegistry,b.stateService),b.stateRegistry.decorator("views",o.c),b.stateRegistry.decorator("onExit",u("onExit")),b.stateRegistry.decorator("onRetain",u("onRetain")),b.stateRegistry.decorator("onEnter",u("onEnter")),b.viewService._pluginapi._viewConfigFactory("ng1",Object(o.b)()),b.urlService.config._decodeParams=!1;var t=b.locationService=b.locationConfig=new d(e);function n(e,n,r,i,o,a,s){return t._runtimeServices(o,e,i,n,r),delete b.router,delete b.$get,b}return d.monkeyPatchPathParameterType(b),b.router=b,b.$get=n,n.$inject=["$location","$browser","$window","$sniffer","$rootScope","$http","$templateCache"],b}y.$inject=["$locationProvider"];var x=function(e){return["$uiRouterProvider",function(t){var n=t.router[e];return n.$get=function(){return n},n}]};function w(e,t,n){if(i.services.$injector=e,i.services.$q=t,!Object.prototype.hasOwnProperty.call(e,"strictDi"))try{e.invoke((function(e){}))}catch(t){e.strictDi=!!/strict mode/.exec(t&&t.toString())}n.stateRegistry.get().map((function(e){return e.$$state().resolvables})).reduce(i.unnestR,[]).filter((function(e){return"deferred"===e.deps})).forEach((function(t){return t.deps=e.annotate(t.resolveFn,e.strictDi)}))}w.$inject=["$injector","$q","$uiRouter"];function k(e){e.$watch((function(){i.trace.approximateDigests++}))}k.$inject=["$rootScope"],f.provider("$uiRouter",y),g.provider("$urlRouter",["$uiRouterProvider",function(e){return e.urlRouterProvider=new p.a(e)}]),h.provider("$urlService",x("urlService")),h.provider("$urlMatcherFactory",["$uiRouterProvider",function(){return b.urlMatcherFactory}]),h.provider("$templateFactory",(function(){return new a})),m.provider("$stateRegistry",x("stateRegistry")),m.provider("$uiRouterGlobals",x("globals")),m.provider("$transitions",x("transitionService")),m.provider("$state",["$uiRouterProvider",function(){return Object(i.extend)(b.stateProvider,{$get:function(){return b.stateService}})}]),m.factory("$stateParams",["$uiRouter",function(e){return e.globals.params}]),v.factory("$view",(function(){return b.viewService})),v.service("$trace",(function(){return i.trace})),v.run(k),h.run(["$urlMatcherFactory",function(e){}]),m.run(["$state",function(e){}]),g.run(["$urlRouter",function(e){}]),f.run(w);var A=function(e){return e.getTokens().filter(i.isString).map((function(t){var n=e.getResolvable(t);return[t,"NOWAIT"===e.getPolicy(n).async?n.promise:n.data]})).reduce(i.applyPairs,{})}},function(e,t,n){"use strict";n.d(t,"b",(function(){return o})),n.d(t,"a",(function(){return a}));var r=n(6),i=function(e){var t=e.router;var n=e.entering().filter((function(e){return!!e.$$state().lazyLoad})).map((function(t){return a(e,t)}));return r.b.$q.all(n).then((function(){if("url"!==e.originalTransition().options().source){var n=e.targetState();return t.stateService.target(n.identifier(),n.params(),n.options())}var r=t.urlService,i=r.match(r.parts()),o=i&&i.rule;if(o&&"STATE"===o.type){var a=o.state,s=i.match;return t.stateService.target(a,s,e.options())}t.urlService.sync()}))},o=function(e){return e.onBefore({entering:function(e){return!!e.lazyLoad}},i)};function a(e,t){var n=t.$$state().lazyLoad,i=n._promise;if(!i){i=n._promise=r.b.$q.when(n(e,t)).then((function(t){t&&Array.isArray(t.states)&&t.states.forEach((function(t){return e.router.stateRegistry.register(t)}));return t})).then((function(e){return delete t.lazyLoad,delete t.$$state().lazyLoad,delete n._promise,e}),(function(e){return delete n._promise,r.b.$q.reject(e)}))}return i}},function(e,t,n){"use strict";var r=n(22);e.exports=i;function i(e){this._isDirected=!r.has(e,"directed")||e.directed,this._isMultigraph=!!r.has(e,"multigraph")&&e.multigraph,this._isCompound=!!r.has(e,"compound")&&e.compound,this._label=void 0,this._defaultNodeLabelFn=r.constant(void 0),this._defaultEdgeLabelFn=r.constant(void 0),this._nodes={},this._isCompound&&(this._parent={},this._children={},this._children["\0"]={}),this._in={},this._preds={},this._out={},this._sucs={},this._edgeObjs={},this._edgeLabels={}}function o(e,t){e[t]?e[t]++:e[t]=1}function a(e,t){--e[t]||delete e[t]}function s(e,t,n,i){var o=""+t,a=""+n;if(!e&&o>a){var s=o;o=a,a=s}return o+""+a+""+(r.isUndefined(i)?"\0":i)}function l(e,t,n,r){var i=""+t,o=""+n;if(!e&&i>o){var a=i;i=o,o=a}var s={v:i,w:o};return r&&(s.name=r),s}function c(e,t){return s(e,t.v,t.w,t.name)}i.prototype._nodeCount=0,i.prototype._edgeCount=0,i.prototype.isDirected=function(){return this._isDirected},i.prototype.isMultigraph=function(){return this._isMultigraph},i.prototype.isCompound=function(){return this._isCompound},i.prototype.setGraph=function(e){return this._label=e,this},i.prototype.graph=function(){return this._label},i.prototype.setDefaultNodeLabel=function(e){return r.isFunction(e)||(e=r.constant(e)),this._defaultNodeLabelFn=e,this},i.prototype.nodeCount=function(){return this._nodeCount},i.prototype.nodes=function(){return r.keys(this._nodes)},i.prototype.sources=function(){var e=this;return r.filter(this.nodes(),(function(t){return r.isEmpty(e._in[t])}))},i.prototype.sinks=function(){var e=this;return r.filter(this.nodes(),(function(t){return r.isEmpty(e._out[t])}))},i.prototype.setNodes=function(e,t){var n=arguments,i=this;return r.each(e,(function(e){n.length>1?i.setNode(e,t):i.setNode(e)})),this},i.prototype.setNode=function(e,t){return r.has(this._nodes,e)?(arguments.length>1&&(this._nodes[e]=t),this):(this._nodes[e]=arguments.length>1?t:this._defaultNodeLabelFn(e),this._isCompound&&(this._parent[e]="\0",this._children[e]={},this._children["\0"][e]=!0),this._in[e]={},this._preds[e]={},this._out[e]={},this._sucs[e]={},++this._nodeCount,this)},i.prototype.node=function(e){return this._nodes[e]},i.prototype.hasNode=function(e){return r.has(this._nodes,e)},i.prototype.removeNode=function(e){var t=this;if(r.has(this._nodes,e)){var n=function(e){t.removeEdge(t._edgeObjs[e])};delete this._nodes[e],this._isCompound&&(this._removeFromParentsChildList(e),delete this._parent[e],r.each(this.children(e),(function(e){t.setParent(e)})),delete this._children[e]),r.each(r.keys(this._in[e]),n),delete this._in[e],delete this._preds[e],r.each(r.keys(this._out[e]),n),delete this._out[e],delete this._sucs[e],--this._nodeCount}return this},i.prototype.setParent=function(e,t){if(!this._isCompound)throw new Error("Cannot set parent in a non-compound graph");if(r.isUndefined(t))t="\0";else{for(var n=t+="";!r.isUndefined(n);n=this.parent(n))if(n===e)throw new Error("Setting "+t+" as parent of "+e+" would create a cycle");this.setNode(t)}return this.setNode(e),this._removeFromParentsChildList(e),this._parent[e]=t,this._children[t][e]=!0,this},i.prototype._removeFromParentsChildList=function(e){delete this._children[this._parent[e]][e]},i.prototype.parent=function(e){if(this._isCompound){var t=this._parent[e];if("\0"!==t)return t}},i.prototype.children=function(e){if(r.isUndefined(e)&&(e="\0"),this._isCompound){var t=this._children[e];if(t)return r.keys(t)}else{if("\0"===e)return this.nodes();if(this.hasNode(e))return[]}},i.prototype.predecessors=function(e){var t=this._preds[e];if(t)return r.keys(t)},i.prototype.successors=function(e){var t=this._sucs[e];if(t)return r.keys(t)},i.prototype.neighbors=function(e){var t=this.predecessors(e);if(t)return r.union(t,this.successors(e))},i.prototype.isLeaf=function(e){return 0===(this.isDirected()?this.successors(e):this.neighbors(e)).length},i.prototype.filterNodes=function(e){var t=new this.constructor({directed:this._isDirected,multigraph:this._isMultigraph,compound:this._isCompound});t.setGraph(this.graph());var n=this;r.each(this._nodes,(function(n,r){e(r)&&t.setNode(r,n)})),r.each(this._edgeObjs,(function(e){t.hasNode(e.v)&&t.hasNode(e.w)&&t.setEdge(e,n.edge(e))}));var i={};return this._isCompound&&r.each(t.nodes(),(function(e){t.setParent(e,function e(r){var o=n.parent(r);return void 0===o||t.hasNode(o)?(i[r]=o,o):o in i?i[o]:e(o)}(e))})),t},i.prototype.setDefaultEdgeLabel=function(e){return r.isFunction(e)||(e=r.constant(e)),this._defaultEdgeLabelFn=e,this},i.prototype.edgeCount=function(){return this._edgeCount},i.prototype.edges=function(){return r.values(this._edgeObjs)},i.prototype.setPath=function(e,t){var n=this,i=arguments;return r.reduce(e,(function(e,r){return i.length>1?n.setEdge(e,r,t):n.setEdge(e,r),r})),this},i.prototype.setEdge=function(){var e,t,n,i,a=!1,c=arguments[0];"object"==typeof c&&null!==c&&"v"in c?(e=c.v,t=c.w,n=c.name,2===arguments.length&&(i=arguments[1],a=!0)):(e=c,t=arguments[1],n=arguments[3],arguments.length>2&&(i=arguments[2],a=!0)),e=""+e,t=""+t,r.isUndefined(n)||(n=""+n);var u=s(this._isDirected,e,t,n);if(r.has(this._edgeLabels,u))return a&&(this._edgeLabels[u]=i),this;if(!r.isUndefined(n)&&!this._isMultigraph)throw new Error("Cannot set a named edge when isMultigraph = false");this.setNode(e),this.setNode(t),this._edgeLabels[u]=a?i:this._defaultEdgeLabelFn(e,t,n);var d=l(this._isDirected,e,t,n);return e=d.v,t=d.w,Object.freeze(d),this._edgeObjs[u]=d,o(this._preds[t],e),o(this._sucs[e],t),this._in[t][u]=d,this._out[e][u]=d,this._edgeCount++,this},i.prototype.edge=function(e,t,n){var r=1===arguments.length?c(this._isDirected,arguments[0]):s(this._isDirected,e,t,n);return this._edgeLabels[r]},i.prototype.hasEdge=function(e,t,n){var i=1===arguments.length?c(this._isDirected,arguments[0]):s(this._isDirected,e,t,n);return r.has(this._edgeLabels,i)},i.prototype.removeEdge=function(e,t,n){var r=1===arguments.length?c(this._isDirected,arguments[0]):s(this._isDirected,e,t,n),i=this._edgeObjs[r];return i&&(e=i.v,t=i.w,delete this._edgeLabels[r],delete this._edgeObjs[r],a(this._preds[t],e),a(this._sucs[e],t),delete this._in[t][r],delete this._out[e][r],this._edgeCount--),this},i.prototype.inEdges=function(e,t){var n=this._in[e];if(n){var i=r.values(n);return t?r.filter(i,(function(e){return e.v===t})):i}},i.prototype.outEdges=function(e,t){var n=this._out[e];if(n){var i=r.values(n);return t?r.filter(i,(function(e){return e.w===t})):i}},i.prototype.nodeEdges=function(e,t){var n=this.inEdges(e,t);if(n)return n.concat(this.outEdges(e,t))}},function(e,t,n){var r=n(46)(n(31),"Map");e.exports=r},function(e,t,n){var r=n(265),i=n(272),o=n(274),a=n(275),s=n(276);function l(e){var t=-1,n=null==e?0:e.length;for(this.clear();++t-1&&e%1==0&&e<=9007199254740991}},function(e,t,n){(function(e){var r=n(150),i=t&&!t.nodeType&&t,o=i&&"object"==typeof e&&e&&!e.nodeType&&e,a=o&&o.exports===i&&r.process,s=function(){try{var e=o&&o.require&&o.require("util").types;return e||a&&a.binding&&a.binding("util")}catch(e){}}();e.exports=s}).call(this,n(80)(e))},function(e,t,n){var r=n(83),i=n(282),o=Object.prototype.hasOwnProperty;e.exports=function(e){if(!r(e))return i(e);var t=[];for(var n in Object(e))o.call(e,n)&&"constructor"!=n&&t.push(n);return t}},function(e,t,n){var r=n(157),i=n(158),o=Object.prototype.propertyIsEnumerable,a=Object.getOwnPropertySymbols,s=a?function(e){return null==e?[]:(e=Object(e),r(a(e),(function(t){return o.call(e,t)})))}:i;e.exports=s},function(e,t){e.exports=function(e,t){for(var n=-1,r=t.length,i=e.length;++n0&&o(u)?n>1?e(u,n-1,o,a,s):r(s,u):a||(s[s.length]=u)}return s}},function(e,t,n){var r=n(61);e.exports=function(e,t,n){for(var i=-1,o=e.length;++iu))return!1;var p=l.get(e),f=l.get(t);if(p&&f)return p==t&&f==e;var h=-1,g=!0,m=2&n?new r:void 0;for(l.set(e,t),l.set(t,e);++h0&&(o=l.removeMin(),(a=s[o]).distance!==Number.POSITIVE_INFINITY);)r(o).forEach(c);return s}(e,String(t),n||o,r||function(t){return e.outEdges(t)})};var o=r.constant(1)},function(e,t,n){var r=n(22);function i(){this._arr=[],this._keyIndices={}}e.exports=i,i.prototype.size=function(){return this._arr.length},i.prototype.keys=function(){return this._arr.map((function(e){return e.key}))},i.prototype.has=function(e){return r.has(this._keyIndices,e)},i.prototype.priority=function(e){var t=this._keyIndices[e];if(void 0!==t)return this._arr[t].priority},i.prototype.min=function(){if(0===this.size())throw new Error("Queue underflow");return this._arr[0].key},i.prototype.add=function(e,t){var n=this._keyIndices;if(e=String(e),!r.has(n,e)){var i=this._arr,o=i.length;return n[e]=o,i.push({key:e,priority:t}),this._decrease(o),!0}return!1},i.prototype.removeMin=function(){this._swap(0,this._arr.length-1);var e=this._arr.pop();return delete this._keyIndices[e.key],this._heapify(0),e.key},i.prototype.decrease=function(e,t){var n=this._keyIndices[e];if(t>this._arr[n].priority)throw new Error("New priority is greater than current priority. Key: "+e+" Old: "+this._arr[n].priority+" New: "+t);this._arr[n].priority=t,this._decrease(n)},i.prototype._heapify=function(e){var t=this._arr,n=2*e,r=n+1,i=e;n>1].priority/,greedy:!0},prolog:{pattern:/<\?[\s\S]+?\?>/,greedy:!0},doctype:{pattern:/"'[\]]|"[^"]*"|'[^']*')+(?:\[(?:[^<"'\]]|"[^"]*"|'[^']*'|<(?!!--)|)*\]\s*)?>/i,greedy:!0,inside:{"internal-subset":{pattern:/(^[^\[]*\[)[\s\S]+(?=\]>$)/,lookbehind:!0,greedy:!0,inside:null},string:{pattern:/"[^"]*"|'[^']*'/,greedy:!0},punctuation:/^$|[[\]]/,"doctype-tag":/^DOCTYPE/i,name:/[^\s<>'"]+/}},cdata:{pattern://i,greedy:!0},tag:{pattern:/<\/?(?!\d)[^\s>\/=$<%]+(?:\s(?:\s*[^\s>\/=]+(?:\s*=\s*(?:"[^"]*"|'[^']*'|[^\s'">=]+(?=[\s>]))|(?=[\s/>])))+)?\s*\/?>/,greedy:!0,inside:{tag:{pattern:/^<\/?[^\s>\/]+/,inside:{punctuation:/^<\/?/,namespace:/^[^\s>\/:]+:/}},"special-attr":[],"attr-value":{pattern:/=\s*(?:"[^"]*"|'[^']*'|[^\s'">=]+)/,inside:{punctuation:[{pattern:/^=/,alias:"attr-equals"},/"|'/]}},punctuation:/\/?>/,"attr-name":{pattern:/[^\s>\/]+/,inside:{namespace:/^[^\s>\/:]+:/}}}},entity:[{pattern:/&[\da-z]{1,8};/i,alias:"named-entity"},/&#x?[\da-f]{1,8};/i]},n.languages.markup.tag.inside["attr-value"].inside.entity=n.languages.markup.entity,n.languages.markup.doctype.inside["internal-subset"].inside=n.languages.markup,n.hooks.add("wrap",(function(e){"entity"===e.type&&(e.attributes.title=e.content.replace(/&/,"&"))})),Object.defineProperty(n.languages.markup.tag,"addInlined",{value:function(e,t){var r={};r["language-"+t]={pattern:/(^$)/i,lookbehind:!0,inside:n.languages[t]},r.cdata=/^$/i;var i={"included-cdata":{pattern://i,inside:r}};i["language-"+t]={pattern:/[\s\S]+/,inside:n.languages[t]};var o={};o[e]={pattern:RegExp(/(<__[^>]*>)(?:))*\]\]>|(?!)/.source.replace(/__/g,(function(){return e})),"i"),lookbehind:!0,greedy:!0,inside:i},n.languages.insertBefore("markup","cdata",o)}}),Object.defineProperty(n.languages.markup.tag,"addAttribute",{value:function(e,t){n.languages.markup.tag.inside["special-attr"].push({pattern:RegExp(/(^|["'\s])/.source+"(?:"+e+")"+/\s*=\s*(?:"[^"]*"|'[^']*'|[^\s'">=]+(?=[\s>]))/.source,"i"),lookbehind:!0,inside:{"attr-name":/^[^\s=]+/,"attr-value":{pattern:/=[\s\S]+/,inside:{value:{pattern:/(^=\s*(["']|(?!["'])))\S[\s\S]*(?=\2$)/,lookbehind:!0,alias:[t,"language-"+t],inside:n.languages[t]},punctuation:[{pattern:/^=/,alias:"attr-equals"},/"|'/]}}}})}}),n.languages.html=n.languages.markup,n.languages.mathml=n.languages.markup,n.languages.svg=n.languages.markup,n.languages.xml=n.languages.extend("markup",{}),n.languages.ssml=n.languages.xml,n.languages.atom=n.languages.xml,n.languages.rss=n.languages.xml,function(e){var t=/(?:"(?:\\(?:\r\n|[\s\S])|[^"\\\r\n])*"|'(?:\\(?:\r\n|[\s\S])|[^'\\\r\n])*')/;e.languages.css={comment:/\/\*[\s\S]*?\*\//,atrule:{pattern:/@[\w-](?:[^;{\s]|\s+(?![\s{]))*(?:;|(?=\s*\{))/,inside:{rule:/^@[\w-]+/,"selector-function-argument":{pattern:/(\bselector\s*\(\s*(?![\s)]))(?:[^()\s]|\s+(?![\s)])|\((?:[^()]|\([^()]*\))*\))+(?=\s*\))/,lookbehind:!0,alias:"selector"},keyword:{pattern:/(^|[^\w-])(?:and|not|only|or)(?![\w-])/,lookbehind:!0}}},url:{pattern:RegExp("\\burl\\((?:"+t.source+"|"+/(?:[^\\\r\n()"']|\\[\s\S])*/.source+")\\)","i"),greedy:!0,inside:{function:/^url/i,punctuation:/^\(|\)$/,string:{pattern:RegExp("^"+t.source+"$"),alias:"url"}}},selector:{pattern:RegExp("(^|[{}\\s])[^{}\\s](?:[^{};\"'\\s]|\\s+(?![\\s{])|"+t.source+")*(?=\\s*\\{)"),lookbehind:!0},string:{pattern:t,greedy:!0},property:{pattern:/(^|[^-\w\xA0-\uFFFF])(?!\s)[-_a-z\xA0-\uFFFF](?:(?!\s)[-\w\xA0-\uFFFF])*(?=\s*:)/i,lookbehind:!0},important:/!important\b/i,function:{pattern:/(^|[^-a-z0-9])[-a-z0-9]+(?=\()/i,lookbehind:!0},punctuation:/[(){};:,]/},e.languages.css.atrule.inside.rest=e.languages.css;var n=e.languages.markup;n&&(n.tag.addInlined("style","css"),n.tag.addAttribute("style","css"))}(n),n.languages.clike={comment:[{pattern:/(^|[^\\])\/\*[\s\S]*?(?:\*\/|$)/,lookbehind:!0,greedy:!0},{pattern:/(^|[^\\:])\/\/.*/,lookbehind:!0,greedy:!0}],string:{pattern:/(["'])(?:\\(?:\r\n|[\s\S])|(?!\1)[^\\\r\n])*\1/,greedy:!0},"class-name":{pattern:/(\b(?:class|extends|implements|instanceof|interface|new|trait)\s+|\bcatch\s+\()[\w.\\]+/i,lookbehind:!0,inside:{punctuation:/[.\\]/}},keyword:/\b(?:break|catch|continue|do|else|finally|for|function|if|in|instanceof|new|null|return|throw|try|while)\b/,boolean:/\b(?:false|true)\b/,function:/\b\w+(?=\()/,number:/\b0x[\da-f]+\b|(?:\b\d+(?:\.\d*)?|\B\.\d+)(?:e[+-]?\d+)?/i,operator:/[<>]=?|[!=]=?=?|--?|\+\+?|&&?|\|\|?|[?*/~^%]/,punctuation:/[{}[\];(),.:]/},n.languages.javascript=n.languages.extend("clike",{"class-name":[n.languages.clike["class-name"],{pattern:/(^|[^$\w\xA0-\uFFFF])(?!\s)[_$A-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\.(?:constructor|prototype))/,lookbehind:!0}],keyword:[{pattern:/((?:^|\})\s*)catch\b/,lookbehind:!0},{pattern:/(^|[^.]|\.\.\.\s*)\b(?:as|assert(?=\s*\{)|async(?=\s*(?:function\b|\(|[$\w\xA0-\uFFFF]|$))|await|break|case|class|const|continue|debugger|default|delete|do|else|enum|export|extends|finally(?=\s*(?:\{|$))|for|from(?=\s*(?:['"]|$))|function|(?:get|set)(?=\s*(?:[#\[$\w\xA0-\uFFFF]|$))|if|implements|import|in|instanceof|interface|let|new|null|of|package|private|protected|public|return|static|super|switch|this|throw|try|typeof|undefined|var|void|while|with|yield)\b/,lookbehind:!0}],function:/#?(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\s*(?:\.\s*(?:apply|bind|call)\s*)?\()/,number:{pattern:RegExp(/(^|[^\w$])/.source+"(?:"+/NaN|Infinity/.source+"|"+/0[bB][01]+(?:_[01]+)*n?/.source+"|"+/0[oO][0-7]+(?:_[0-7]+)*n?/.source+"|"+/0[xX][\dA-Fa-f]+(?:_[\dA-Fa-f]+)*n?/.source+"|"+/\d+(?:_\d+)*n/.source+"|"+/(?:\d+(?:_\d+)*(?:\.(?:\d+(?:_\d+)*)?)?|\.\d+(?:_\d+)*)(?:[Ee][+-]?\d+(?:_\d+)*)?/.source+")"+/(?![\w$])/.source),lookbehind:!0},operator:/--|\+\+|\*\*=?|=>|&&=?|\|\|=?|[!=]==|<<=?|>>>?=?|[-+*/%&|^!=<>]=?|\.{3}|\?\?=?|\?\.?|[~:]/}),n.languages.javascript["class-name"][0].pattern=/(\b(?:class|extends|implements|instanceof|interface|new)\s+)[\w.\\]+/,n.languages.insertBefore("javascript","keyword",{regex:{pattern:RegExp(/((?:^|[^$\w\xA0-\uFFFF."'\])\s]|\b(?:return|yield))\s*)/.source+/\//.source+"(?:"+/(?:\[(?:[^\]\\\r\n]|\\.)*\]|\\.|[^/\\\[\r\n])+\/[dgimyus]{0,7}/.source+"|"+/(?:\[(?:[^[\]\\\r\n]|\\.|\[(?:[^[\]\\\r\n]|\\.|\[(?:[^[\]\\\r\n]|\\.)*\])*\])*\]|\\.|[^/\\\[\r\n])+\/[dgimyus]{0,7}v[dgimyus]{0,7}/.source+")"+/(?=(?:\s|\/\*(?:[^*]|\*(?!\/))*\*\/)*(?:$|[\r\n,.;:})\]]|\/\/))/.source),lookbehind:!0,greedy:!0,inside:{"regex-source":{pattern:/^(\/)[\s\S]+(?=\/[a-z]*$)/,lookbehind:!0,alias:"language-regex",inside:n.languages.regex},"regex-delimiter":/^\/|\/$/,"regex-flags":/^[a-z]+$/}},"function-variable":{pattern:/#?(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\s*[=:]\s*(?:async\s*)?(?:\bfunction\b|(?:\((?:[^()]|\([^()]*\))*\)|(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*)\s*=>))/,alias:"function"},parameter:[{pattern:/(function(?:\s+(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*)?\s*\(\s*)(?!\s)(?:[^()\s]|\s+(?![\s)])|\([^()]*\))+(?=\s*\))/,lookbehind:!0,inside:n.languages.javascript},{pattern:/(^|[^$\w\xA0-\uFFFF])(?!\s)[_$a-z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\s*=>)/i,lookbehind:!0,inside:n.languages.javascript},{pattern:/(\(\s*)(?!\s)(?:[^()\s]|\s+(?![\s)])|\([^()]*\))+(?=\s*\)\s*=>)/,lookbehind:!0,inside:n.languages.javascript},{pattern:/((?:\b|\s|^)(?!(?:as|async|await|break|case|catch|class|const|continue|debugger|default|delete|do|else|enum|export|extends|finally|for|from|function|get|if|implements|import|in|instanceof|interface|let|new|null|of|package|private|protected|public|return|set|static|super|switch|this|throw|try|typeof|undefined|var|void|while|with|yield)(?![$\w\xA0-\uFFFF]))(?:(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*\s*)\(\s*|\]\s*\(\s*)(?!\s)(?:[^()\s]|\s+(?![\s)])|\([^()]*\))+(?=\s*\)\s*\{)/,lookbehind:!0,inside:n.languages.javascript}],constant:/\b[A-Z](?:[A-Z_]|\dx?)*\b/}),n.languages.insertBefore("javascript","string",{hashbang:{pattern:/^#!.*/,greedy:!0,alias:"comment"},"template-string":{pattern:/`(?:\\[\s\S]|\$\{(?:[^{}]|\{(?:[^{}]|\{[^}]*\})*\})+\}|(?!\$\{)[^\\`])*`/,greedy:!0,inside:{"template-punctuation":{pattern:/^`|`$/,alias:"string"},interpolation:{pattern:/((?:^|[^\\])(?:\\{2})*)\$\{(?:[^{}]|\{(?:[^{}]|\{[^}]*\})*\})+\}/,lookbehind:!0,inside:{"interpolation-punctuation":{pattern:/^\$\{|\}$/,alias:"punctuation"},rest:n.languages.javascript}},string:/[\s\S]+/}},"string-property":{pattern:/((?:^|[,{])[ \t]*)(["'])(?:\\(?:\r\n|[\s\S])|(?!\2)[^\\\r\n])*\2(?=\s*:)/m,lookbehind:!0,greedy:!0,alias:"property"}}),n.languages.insertBefore("javascript","operator",{"literal-property":{pattern:/((?:^|[,{])[ \t]*)(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\s*:)/m,lookbehind:!0,alias:"property"}}),n.languages.markup&&(n.languages.markup.tag.addInlined("script","javascript"),n.languages.markup.tag.addAttribute(/on(?:abort|blur|change|click|composition(?:end|start|update)|dblclick|error|focus(?:in|out)?|key(?:down|up)|load|mouse(?:down|enter|leave|move|out|over|up)|reset|resize|scroll|select|slotchange|submit|unload|wheel)/.source,"javascript")),n.languages.js=n.languages.javascript,function(){if(void 0!==n&&"undefined"!=typeof document){Element.prototype.matches||(Element.prototype.matches=Element.prototype.msMatchesSelector||Element.prototype.webkitMatchesSelector);var e={js:"javascript",py:"python",rb:"ruby",ps1:"powershell",psm1:"powershell",sh:"bash",bat:"batch",h:"c",tex:"latex"},t='pre[data-src]:not([data-src-status="loaded"]):not([data-src-status="loading"])';n.hooks.add("before-highlightall",(function(e){e.selector+=", "+t})),n.hooks.add("before-sanity-check",(function(r){var i=r.element;if(i.matches(t)){r.code="",i.setAttribute("data-src-status","loading");var o=i.appendChild(document.createElement("CODE"));o.textContent="Loading…";var a=i.getAttribute("data-src"),s=r.language;if("none"===s){var l=(/\.(\w+)$/.exec(a)||[,"none"])[1];s=e[l]||l}n.util.setLanguage(o,s),n.util.setLanguage(i,s);var c=n.plugins.autoloader;c&&c.loadLanguages(s),function(e,t,n){var r=new XMLHttpRequest;r.open("GET",e,!0),r.onreadystatechange=function(){var e,i;4==r.readyState&&(r.status<400&&r.responseText?t(r.responseText):r.status>=400?n((e=r.status,i=r.statusText,"✖ Error "+e+" while fetching file: "+i)):n("✖ Error: File does not exist or is empty"))},r.send(null)}(a,(function(e){i.setAttribute("data-src-status","loaded");var t=function(e){var t=/^\s*(\d+)\s*(?:(,)\s*(?:(\d+)\s*)?)?$/.exec(e||"");if(t){var n=Number(t[1]),r=t[2],i=t[3];return r?i?[n,Number(i)]:[n,void 0]:[n,n]}}(i.getAttribute("data-range"));if(t){var r=e.split(/\r\n?|\n/g),a=t[0],s=null==t[1]?r.length:t[1];a<0&&(a+=r.length),a=Math.max(0,Math.min(a-1,r.length)),s<0&&(s+=r.length),s=Math.max(0,Math.min(s,r.length)),e=r.slice(a,s).join("\n"),i.hasAttribute("data-start")||i.setAttribute("data-start",String(a+1))}o.textContent=e,n.highlightElement(o)}),(function(e){i.setAttribute("data-src-status","failed"),o.textContent=e}))}})),n.plugins.fileHighlight={highlight:function(e){for(var r,i=(e||document).querySelectorAll(t),o=0;r=i[o++];)n.highlightElement(r)}};var r=!1;n.fileHighlight=function(){r||(console.warn("Prism.fileHighlight is deprecated. Use `Prism.plugins.fileHighlight.highlight` instead."),r=!0),n.plugins.fileHighlight.highlight.apply(this,arguments)}}}()}).call(this,n(35))},function(e,t,n){var r=n(47),i=n(23);e.exports=function(e){if(!i(e))return!1;var t=r(e);return"[object Function]"==t||"[object GeneratorFunction]"==t||"[object AsyncFunction]"==t||"[object Proxy]"==t}},function(e,t,n){var r=n(78),i=n(79);e.exports=function(e,t,n,o){var a=!n;n||(n={});for(var s=-1,l=t.length;++s-1&&e%1==0&&e=0||(s.push(i[c]),a[i[c]]=this[i[c]]);return Object(r.r)({},a,e)},e}()},function(e,t,n){"use strict";n.d(t,"b",(function(){return m})),n.d(t,"a",(function(){return v}));var r=n(0),i=n(1),o=n(7),a=n(3),s=n(15),l=n(6);function c(e){return e.name}function u(e){return e.self.$$state=function(){return e},e.self}function d(e){return e.parent&&e.parent.data&&(e.data=e.self.data=Object(r.A)(e.parent.data,e.data)),e.data}var p=function(e,t){return function(n){var o=n.self;if(o&&o.url&&o.name&&o.name.match(/\.\*\*$/)){var a={};Object(r.m)(o,a),a.url+="{remainder:any}",o=a}var s=n.parent,l=function(e){if(!Object(i.l)(e))return!1;var t="^"===e.charAt(0);return{val:t?e.substring(1):e,root:t}}(o.url),c=l?e.compile(l.val,{state:o}):o.url;if(!c)return null;if(!e.isMatcher(c))throw new Error("Invalid url '"+c+"' in state '"+n+"'");return l&&l.root?c:(s&&s.navigable||t()).url.append(c)}},f=function(e){return function(t){return!e(t)&&t.url?t:t.parent?t.parent.navigable:null}};function h(e){return e.parent?e.parent.path.concat(e):[e]}function g(e){var t=e.parent?Object(r.r)({},e.parent.includes):{};return t[e.name]=!0,t}function m(e){var t,n,c=function(e){return e.provide||e.token},u=Object(a.l)([[Object(a.n)("resolveFn"),function(e){return new s.a(c(e),e.resolveFn,e.deps,e.policy)}],[Object(a.n)("useFactory"),function(e){return new s.a(c(e),e.useFactory,e.deps||e.dependencies,e.policy)}],[Object(a.n)("useClass"),function(e){return new s.a(c(e),(function(){return new e.useClass}),[],e.policy)}],[Object(a.n)("useValue"),function(e){return new s.a(c(e),(function(){return e.useValue}),[],e.policy,e.useValue)}],[Object(a.n)("useExisting"),function(e){return new s.a(c(e),r.y,[e.useExisting],e.policy)}]]),d=Object(a.l)([[Object(a.m)(Object(a.n)("val"),i.l),function(e){return new s.a(e.token,r.y,[e.val],e.policy)}],[Object(a.m)(Object(a.n)("val"),i.a),function(e){return new s.a(e.token,Object(r.P)(e.val),e.val.slice(0,-1),e.policy)}],[Object(a.m)(Object(a.n)("val"),i.d),function(e){return new s.a(e.token,e.val,(t=e.val,n=l.b.$injector,t.$inject||n&&n.annotate(t,n.strictDi)||"deferred"),e.policy);var t,n}]]),p=Object(a.l)([[Object(a.h)(s.a),function(e){return e}],[function(e){return!(!e.token||!e.resolveFn)},u],[function(e){return!(!e.provide&&!e.token||!(e.useValue||e.useFactory||e.useExisting||e.useClass))},u],[function(e){return!!(e&&e.val&&(Object(i.l)(e.val)||Object(i.a)(e.val)||Object(i.d)(e.val)))},d],[Object(a.p)(!0),function(e){throw new Error("Invalid resolve value: "+Object(o.m)(e))}]]),f=e.resolve;return(Object(i.a)(f)?f:(t=f,n=e.resolvePolicy||{},Object.keys(t||{}).map((function(e){return{token:e,val:t[e],deps:void 0,policy:n[e]}})))).map(p)}var v=function(){function e(e,t){this.matcher=e;var n,i=this,o=function(){return e.find("")},s=function(e){return""===e.name};this.builders={name:[c],self:[u],parent:[function(t){return s(t)?null:e.find(i.parentName(t))||o()}],data:[d],url:[p(t,o)],navigable:[f(s)],params:[(n=t.paramFactory,function(e){var t=e.url&&e.url.parameters({inherit:!1})||[],i=Object(r.U)(Object(r.C)(Object(r.F)(e.params||{},t.map(Object(a.n)("id"))),(function(t,r){return n.fromConfig(r,null,e.self)})));return t.concat(i).map((function(e){return[e.id,e]})).reduce(r.h,{})})],views:[],path:[h],includes:[g],resolvables:[m]}}return e.prototype.builder=function(e,t){var n=this.builders,r=n[e]||[];return Object(i.l)(e)&&!Object(i.c)(t)?r.length>1?r:r[0]:Object(i.l)(e)&&Object(i.d)(t)?(n[e]=r,n[e].push(t),function(){return n[e].splice(n[e].indexOf(t,1))&&null}):void 0},e.prototype.build=function(e){var t=this.matcher,n=this.builders,i=this.parentName(e);if(i&&!t.find(i,void 0,!1))return null;for(var o in n)if(n.hasOwnProperty(o)){var a=n[o].reduce((function(e,t){return function(n){return t(n,e)}}),r.E);e[o]=a(e)}return e},e.prototype.parentName=function(e){var t=e.name||"",n=t.split(".");if("**"===n.pop()&&n.pop(),n.length){if(e.parent)throw new Error("States that specify the 'parent:' property should not have a '.' in their name ("+t+")");return n.join(".")}return e.parent?Object(i.l)(e.parent)?e.parent:e.parent.name:""},e.prototype.name=function(e){var t=e.name;if(-1!==t.indexOf(".")||!e.parent)return t;var n=Object(i.l)(e.parent)?e.parent:e.parent.name;return n?n+"."+t:t},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return a}));var r=n(1),i=n(0),o=n(19),a=function(){function e(e){this._states=e}return e.prototype.isRelative=function(e){return 0===(e=e||"").indexOf(".")||0===e.indexOf("^")},e.prototype.find=function(e,t,n){if(void 0===n&&(n=!0),e||""===e){var a=Object(r.l)(e),s=a?e:e.name;this.isRelative(s)&&(s=this.resolvePath(s,t));var l=this._states[s];if(l&&(a||!(a||l!==e&&l.self!==e)))return l;if(a&&n){var c=Object(i.U)(this._states).filter((function(e){return e.__stateObjectCache.nameGlob&&e.__stateObjectCache.nameGlob.matches(s)}));return c.length>1&&o.a.error("stateMatcher.find: Found multiple matches for "+s+" using glob: ",c.map((function(e){return e.name}))),c[0]}}},e.prototype.resolvePath=function(e,t){if(!t)throw new Error("No reference point given for path '"+e+"'");for(var n=this.find(t),r=e.split("."),i=r.length,o=0,a=n;o0;){var c=t.shift(),u=c.name,d=r.build(c),p=o.indexOf(c);if(d){var f=s(u);if(f&&f.name===u)throw new Error("State '"+u+"' is already defined");var h=s(u+".**");h&&this.router.stateRegistry.deregister(h),n[u]=c,this.attachRoute(c),p>=0&&o.splice(p,1),i.push(c)}else{var g=a[u];if(a[u]=t.length,p>=0&&g===t.length)return t.push(c),l(),n;p<0&&o.push(c),t.push(c)}}return l(),n},e.prototype.attachRoute=function(e){if(!e.abstract&&e.url){var t=this.router.urlService.rules;t.rule(t.urlRuleFactory.create(e))}},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return l}));var r=n(94),i=n(93),o=n(95),a=n(0),s=n(3),l=function(){function e(e){this.router=e,this.states={},this.listeners=[],this.matcher=new r.a(this.states),this.builder=new i.a(this.matcher,e.urlMatcherFactory),this.stateQueue=new o.a(e,this.states,this.builder,this.listeners),this._registerRoot()}return e.prototype._registerRoot=function(){(this._root=this.stateQueue.register({name:"",url:"^",views:null,params:{"#":{value:null,type:"hash",dynamic:!0}},abstract:!0})).navigable=null},e.prototype.dispose=function(){var e=this;this.stateQueue.dispose(),this.listeners=[],this.get().forEach((function(t){return e.get(t)&&e.deregister(t)}))},e.prototype.onStatesChanged=function(e){return this.listeners.push(e),function(){Object(a.L)(this.listeners)(e)}.bind(this)},e.prototype.root=function(){return this._root},e.prototype.register=function(e){return this.stateQueue.register(e)},e.prototype._deregisterTree=function(e){var t=this,n=this.get().map((function(e){return e.$$state()})),r=function(e){var t=n.filter((function(t){return-1!==e.indexOf(t.parent)}));return 0===t.length?t:t.concat(r(t))},i=r([e]),o=[e].concat(i).reverse();return o.forEach((function(e){var n=t.router.urlService.rules;n.rules().filter(Object(s.o)("state",e)).forEach((function(e){return n.removeRule(e)})),delete t.states[e.name]})),o},e.prototype.deregister=function(e){var t=this.get(e);if(!t)throw new Error("Can't deregister state; not found: "+e);var n=this._deregisterTree(t.$$state());return this.listeners.forEach((function(e){return e("deregistered",n.map((function(e){return e.self})))})),n},e.prototype.get=function(e,t){var n=this;if(0===arguments.length)return Object.keys(this.states).map((function(e){return n.states[e].self}));var r=this.matcher.find(e,t);return r&&r.self||null},e.prototype.decorator=function(e,t){return this.builder.builder(e,t)},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return v}));var r=n(0),i=n(1),o=n(50),a=n(6),s=n(18),l=n(52),c=n(56),u=n(9),d=n(26),p=n(16),f=n(62),h=n(27),g=n(115),m=n(3),v=function(){function e(t){this.router=t,this.invalidCallbacks=[],this._defaultErrorHandler=function(e){e instanceof Error&&e.stack?(console.error(e),console.error(e.stack)):e instanceof u.b?(console.error(e.toString()),e.detail&&e.detail.stack&&console.error(e.detail.stack)):console.error(e)};var n=Object.keys(e.prototype).filter(Object(m.i)(Object(r.z)(["current","$current","params","transition"])));Object(r.n)(Object(m.p)(e.prototype),this,Object(m.p)(this),n)}return Object.defineProperty(e.prototype,"transition",{get:function(){return this.router.globals.transition},enumerable:!1,configurable:!0}),Object.defineProperty(e.prototype,"params",{get:function(){return this.router.globals.params},enumerable:!1,configurable:!0}),Object.defineProperty(e.prototype,"current",{get:function(){return this.router.globals.current},enumerable:!1,configurable:!0}),Object.defineProperty(e.prototype,"$current",{get:function(){return this.router.globals.$current},enumerable:!1,configurable:!0}),e.prototype.dispose=function(){this.defaultErrorHandler(r.E),this.invalidCallbacks=[]},e.prototype._handleInvalidTargetState=function(e,t){var n=this,r=s.a.makeTargetState(this.router.stateRegistry,e),i=this.router.globals,l=function(){return i.transitionHistory.peekTail()},c=l(),p=new o.a(this.invalidCallbacks.slice()),f=new h.b(e).injector(),g=function(e){if(e instanceof d.a){var t=e;return(t=n.target(t.identifier(),t.params(),t.options())).valid()?l()!==c?u.b.superseded().toPromise():n.transitionTo(t.identifier(),t.params(),t.options()):u.b.invalid(t.error()).toPromise()}};return function e(){var n=p.dequeue();return void 0===n?u.b.invalid(t.error()).toPromise():a.b.$q.when(n(t,r,f)).then(g).then((function(t){return t||e()}))}()},e.prototype.onInvalid=function(e){return this.invalidCallbacks.push(e),function(){Object(r.L)(this.invalidCallbacks)(e)}.bind(this)},e.prototype.reload=function(e){return this.transitionTo(this.current,this.params,{reload:!Object(i.c)(e)||e,inherit:!1,notify:!1})},e.prototype.go=function(e,t,n){var i={relative:this.$current,inherit:!0},o=Object(r.o)(n,i,c.b);return this.transitionTo(e,t,o)},e.prototype.target=function(e,t,n){if(void 0===n&&(n={}),Object(i.i)(n.reload)&&!n.reload.name)throw new Error("Invalid reload state object");var r=this.router.stateRegistry;if(n.reloadState=!0===n.reload?r.root():r.matcher.find(n.reload,n.relative),n.reload&&!n.reloadState)throw new Error("No such reload state '"+(Object(i.l)(n.reload)?n.reload:n.reload.name)+"'");return new d.a(this.router.stateRegistry,e,t,n)},e.prototype.getCurrentPath=function(){var e=this,t=this.router.globals.successfulTransitions.peekTail();return t?t.treeChanges().to:[new l.a(e.router.stateRegistry.root())]},e.prototype.transitionTo=function(e,t,n){var i=this;void 0===t&&(t={}),void 0===n&&(n={});var o=this.router,s=o.globals;n=Object(r.o)(n,c.b);var l=function(){return s.transition};n=Object(r.r)(n,{current:l});var p=this.target(e,t,n),f=this.getCurrentPath();if(!p.exists())return this._handleInvalidTargetState(f,p);if(!p.valid())return Object(r.O)(p.error());if(!1===n.supercede&&l())return u.b.ignored("Another transition is in progress and supercede has been set to false in TransitionOptions for the transition. So the transition was ignored in favour of the existing one in progress.").toPromise();var h=function(e){return function(t){if(t instanceof u.b){var n=o.globals.lastStartedTransitionId<=e.$id;if(t.type===u.a.IGNORED)return n&&o.urlRouter.update(),a.b.$q.when(s.current);var r=t.detail;if(t.type===u.a.SUPERSEDED&&t.redirected&&r instanceof d.a){var l=e.redirect(r);return l.run().catch(h(l))}if(t.type===u.a.ABORTED)return n&&o.urlRouter.update(),a.b.$q.reject(t)}return i.defaultErrorHandler()(t),a.b.$q.reject(t)}},g=this.router.transitionService.create(f,p),m=g.run().catch(h(g));return Object(r.N)(m),Object(r.r)(m,{transition:g})},e.prototype.is=function(e,t,n){n=Object(r.o)(n,{relative:this.$current});var o=this.router.stateRegistry.matcher.find(e,n.relative);if(Object(i.c)(o)){if(this.$current!==o)return!1;if(!t)return!0;var a=o.parameters({inherit:!0,matchingKeys:t});return p.b.equals(a,p.b.values(a,t),this.params)}},e.prototype.includes=function(e,t,n){n=Object(r.o)(n,{relative:this.$current});var o=Object(i.l)(e)&&f.a.fromString(e);if(o){if(!o.matches(this.$current.name))return!1;e=this.$current.name}var a=this.router.stateRegistry.matcher.find(e,n.relative),s=this.$current.includes;if(Object(i.c)(a)){if(!Object(i.c)(s[a.name]))return!1;if(!t)return!0;var l=a.parameters({inherit:!0,matchingKeys:t});return p.b.equals(l,p.b.values(l,t),this.params)}},e.prototype.href=function(e,t,n){var o={lossy:!0,inherit:!0,absolute:!1,relative:this.$current};n=Object(r.o)(n,o),t=t||{};var a=this.router.stateRegistry.matcher.find(e,n.relative);if(!Object(i.c)(a))return null;n.inherit&&(t=this.params.$inherit(t,this.$current,a));var s=a&&n.lossy?a.navigable:a;return s&&void 0!==s.url&&null!==s.url?this.router.urlRouter.href(s.url,t,{absolute:n.absolute}):null},e.prototype.defaultErrorHandler=function(e){return this._defaultErrorHandler=e||this._defaultErrorHandler},e.prototype.get=function(e,t){var n=this.router.stateRegistry;return 0===arguments.length?n.get():n.get(e,t||this.$current)},e.prototype.lazyLoad=function(e,t){var n=this.get(e);if(!n||!n.lazyLoad)throw new Error("Can not lazy load "+e);var r=this.getCurrentPath(),i=s.a.makeTargetState(this.router.stateRegistry,r);return t=t||this.router.transitionService.create(r,i),Object(g.a)(t,n)},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return s}));var r=n(0),i=n(1),o=n(14),a=n(17),s=function(){function e(e){this.transition=e}return e.prototype.buildHooksForPhase=function(e){var t=this;return this.transition.router.transitionService._pluginapi._getEvents(e).map((function(e){return t.buildHooks(e)})).reduce(r.T,[]).filter(r.y)},e.prototype.buildHooks=function(e){var t=this.transition,n=t.treeChanges(),i=this.getMatchingHooks(e,n,t);if(!i)return[];var s={transition:t,current:t.options().current};return i.map((function(i){return i.matches(n,t)[e.criteriaMatchPath.name].map((function(n){var l=Object(r.r)({bind:i.bind,traceData:{hookType:e.name,context:n}},s),c=e.criteriaMatchPath.scope===o.b.STATE?n.state.self:null,u=new a.a(t,c,i,l);return{hook:i,node:n,transitionHook:u}}))})).reduce(r.T,[]).sort(function(e){void 0===e&&(e=!1);return function(t,n){var r=e?-1:1,i=(t.node.state.path.length-n.node.state.path.length)*r;return 0!==i?i:n.hook.priority-t.hook.priority}}(e.reverseSort)).map((function(e){return e.transitionHook}))},e.prototype.getMatchingHooks=function(e,t,n){var a=e.hookPhase===o.a.CREATE,s=this.transition.router.transitionService;return(a?[s]:[this.transition,s]).map((function(t){return t.getHooks(e.name)})).filter(Object(r.l)(i.a,"broken event named: "+e.name)).reduce(r.T,[]).filter((function(e){return e.matches(t,n)}))},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return b}));var r=n(100),i=n(101),o=n(56),a=n(102),s=n(96),l=n(97),c=n(103),u=n(0),d=n(1),p=n(104),f=n(10),h=n(4),g=0,m=Object(h.gb)("LocationServices",["url","path","search","hash","onChange"]),v=Object(h.gb)("LocationConfig",["port","protocol","host","baseHref","html5Mode","hashPrefix"]),b=function(){function e(e,t){void 0===e&&(e=m),void 0===t&&(t=v),this.locationService=e,this.locationConfig=t,this.$id=g++,this._disposed=!1,this._disposables=[],this.trace=f.c,this.viewService=new a.a(this),this.globals=new c.a,this.transitionService=new o.a(this),this.urlMatcherFactory=new r.b(this),this.urlRouter=new i.a(this),this.urlService=new p.a(this),this.stateRegistry=new s.a(this),this.stateService=new l.a(this),this._plugins={},this.viewService._pluginapi._rootViewContext(this.stateRegistry.root()),this.globals.$current=this.stateRegistry.root(),this.globals.current=this.globals.$current.self,this.disposable(this.globals),this.disposable(this.stateService),this.disposable(this.stateRegistry),this.disposable(this.transitionService),this.disposable(this.urlService),this.disposable(e),this.disposable(t)}return e.prototype.disposable=function(e){this._disposables.push(e)},e.prototype.dispose=function(e){var t=this;e&&Object(d.d)(e.dispose)?e.dispose(this):(this._disposed=!0,this._disposables.slice().forEach((function(e){try{"function"==typeof e.dispose&&e.dispose(t),Object(u.L)(t._disposables,e)}catch(e){}})))},e.prototype.plugin=function(e,t){void 0===t&&(t={});var n=new e(this,t);if(!n.name)throw new Error("Required property `name` missing on plugin: "+n);return this._disposables.push(n),this._plugins[n.name]=n},e.prototype.getPlugin=function(e){return e?this._plugins[e]:Object(u.U)(this._plugins)},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return s})),n.d(t,"b",(function(){return l}));var r=n(4),i=n(32),o=n(28),a=function(){return(a=Object.assign||function(e){for(var t,n=1,r=arguments.length;n Registering",e),this._viewConfigs.push(e)},e.prototype.sync=function(){var t=this,n=this._uiViews.map((function(e){return[e.fqn,e]})).reduce(r.h,{});function o(e){for(var t=e.viewDecl.$context,n=0;++n&&t.parent;)t=t.parent;return n}var s=Object(i.e)((function(e,t,n,r){return t*(e(n)-e(r))})),l=this._uiViews.sort(s((function(e){var t=function(e){return e&&e.parent?t(e.parent)+1:1};return 1e4*e.fqn.split(".").length+t(e.creationContext)}),1)).map((function(r){var i=t._viewConfigs.filter(e.matches(n,r));return i.length>1&&i.sort(s(o,-1)),{uiView:r,viewConfig:i[0]}})),c=l.map((function(e){return e.viewConfig})),u=this._viewConfigs.filter((function(e){return!Object(r.z)(c,e)})).map((function(e){return{uiView:void 0,viewConfig:e}}));l.forEach((function(e){-1!==t._uiViews.indexOf(e.uiView)&&e.uiView.configUpdated(e.viewConfig)}));var d=l.concat(u);this._listeners.forEach((function(e){return e(d)})),a.c.traceViewSync(d)},e.prototype.registerUIView=function(e){a.c.traceViewServiceUIViewEvent("-> Registering",e);var t=this._uiViews;return t.filter((function(t){return t.fqn===e.fqn&&t.$type===e.$type})).length&&a.c.traceViewServiceUIViewEvent("!!!! duplicate uiView named:",e),t.push(e),this.sync(),function(){-1!==t.indexOf(e)?(a.c.traceViewServiceUIViewEvent("<- Deregistering",e),Object(r.L)(t)(e)):a.c.traceViewServiceUIViewEvent("Tried removing non-registered uiView",e)}},e.prototype.available=function(){return this._uiViews.map(Object(i.n)("fqn"))},e.prototype.active=function(){return this._uiViews.filter(Object(i.n)("$config")).map(Object(i.n)("name"))},e.matches=function(e,t){return function(n){if(t.$type!==n.viewDecl.$type)return!1;var i=n.viewDecl,o=i.$uiViewName.split("."),a=t.fqn.split(".");if(!Object(r.q)(o,a.slice(0-o.length)))return!1;var s=1-o.length||void 0,l=a.slice(0,s).join("."),c=e[l].creationContext;return i.$uiViewContextAnchor===(c&&c.name)}},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return o}));var r=n(92),i=n(50),o=function(){function e(){this.params=new r.a,this.lastStartedTransitionId=-1,this.transitionHistory=new i.a([],1),this.successfulTransitions=new i.a([],1)}return e.prototype.dispose=function(){this.transitionHistory.clear(),this.successfulTransitions.clear(),this.transition=null},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return s}));var r=n(4),i=n(105),o=n(106),a=n(43),s=function(){function e(e){var t=this;this.router=e,this.interceptDeferred=!1,this.rules=new i.a(this.router),this.config=new o.a(this.router),this.url=function(e,n,r){return t.router.locationService.url(e,n,r)},this.path=function(){return t.router.locationService.path()},this.search=function(){return t.router.locationService.search()},this.hash=function(){return t.router.locationService.hash()},this.onChange=function(e){return t.router.locationService.onChange(e)}}return e.prototype.dispose=function(){this.listen(!1),this.rules.dispose()},e.prototype.parts=function(){return{path:this.path(),search:this.search(),hash:this.hash()}},e.prototype.sync=function(e){if(!e||!e.defaultPrevented){var t=this.router,n=t.urlService,i=t.stateService,o={path:n.path(),search:n.search(),hash:n.hash()},s=this.match(o);Object(r.sb)([[r.cb,function(e){return n.url(e,!0)}],[a.TargetState.isDef,function(e){return i.go(e.state,e.params,e.options)}],[Object(r.Q)(a.TargetState),function(e){return i.go(e.state(),e.params(),e.options())}]])(s&&s.rule.handler(s.match,o,this.router))}},e.prototype.listen=function(e){var t=this;if(!1!==e)return this._stopListeningFn=this._stopListeningFn||this.router.urlService.onChange((function(e){return t.sync(e)}));this._stopListeningFn&&this._stopListeningFn(),delete this._stopListeningFn},e.prototype.deferIntercept=function(e){void 0===e&&(e=!0),this.interceptDeferred=e},e.prototype.match=function(e){var t=this;e=Object(r.C)({path:"",search:{},hash:""},e);for(var n,i,o,a=this.rules.rules(),s=0;sn.weight?l:n}return n},e}()},function(e,t,n){"use strict";n.d(t,"a",(function(){return c}));var r,i=n(43),o=n(32),a=n(4),s=n(54);function l(e){if(!(Object(a.U)(e)||Object(a.cb)(e)||Object(a.Q)(i.TargetState)(e)||i.TargetState.isDef(e)))throw new Error("'handler' must be a string, function, TargetState, or have a state: 'newtarget' property");return Object(a.U)(e)?e:Object(a.Sb)(e)}r=function(e,t){var n=function(e,t){return(t.priority||0)-(e.priority||0)}(e,t);return 0!==n||0!==(n=function(e,t){var n={STATE:4,URLMATCHER:4,REGEXP:3,RAW:2,OTHER:1};return(n[e.type]||0)-(n[t.type]||0)}(e,t))||0!==(n=function(e,t){return e.urlMatcher&&t.urlMatcher?o.a.compare(e.urlMatcher,t.urlMatcher):0}(e,t))?n:function(e,t){var n={STATE:!0,URLMATCHER:!0};return n[e.type]&&n[t.type]?0:(e.$id||0)-(t.$id||0)}(e,t)};var c=function(){function e(e){this.router=e,this._sortFn=r,this._rules=[],this._id=0,this.urlRuleFactory=new s.b(e)}return e.prototype.dispose=function(e){this._rules=[],delete this._otherwiseFn},e.prototype.initial=function(e){var t=l(e);this.rule(this.urlRuleFactory.create((function(e,t){return 0===t.globals.transitionHistory.size()&&!!/^\/?$/.exec(e.path)}),t))},e.prototype.otherwise=function(e){var t=l(e);this._otherwiseFn=this.urlRuleFactory.create(Object(a.Sb)(!0),t),this._sorted=!1},e.prototype.removeRule=function(e){Object(a.Ab)(this._rules,e)},e.prototype.rule=function(e){var t=this;if(!s.b.isUrlRule(e))throw new Error("invalid rule");return e.$id=this._id++,e.priority=e.priority||0,this._rules.push(e),this._sorted=!1,function(){return t.removeRule(e)}},e.prototype.rules=function(){return this.ensureSorted(),this._rules.concat(this._otherwiseFn?[this._otherwiseFn]:[])},e.prototype.sort=function(e){for(var t=this.stableSort(this._rules,this._sortFn=e||this._sortFn),n=0,r=0;r")},e.prototype.fromString=function(e,t){return Object(i.isFunction)(e)?e(t):e},e.prototype.fromUrl=function(e,t){return Object(i.isFunction)(e)&&(e=e(t)),null==e?null:this._useHttp?this.$http.get(e,{cache:this.$templateCache,headers:{Accept:"text/html"}}).then((function(e){return e.data})):this.$templateRequest(e)},e.prototype.fromProvider=function(e,t,n){var r=i.services.$injector.annotate(e),o=Object(i.isArray)(e)?Object(i.tail)(e):e;return new i.Resolvable("",o,r).get(n)},e.prototype.fromComponentProvider=function(e,t,n){var r=i.services.$injector.annotate(e),o=Object(i.isArray)(e)?Object(i.tail)(e):e;return new i.Resolvable("",o,r).get(n)},e.prototype.makeComponentTemplate=function(e,t,n,o){o=o||{};var a=r.a.version.minor>=3?"::":"",l=function(e){var t=Object(i.kebobString)(e);return/^(x|data)-/.exec(t)?"x-"+t:t},c=function(e){var t=i.services.$injector.get(e+"Directive");if(!t||!t.length)throw new Error("Unable to find component named '"+e+"'");return t.map(s).reduce(i.unnestR,[])}(n).map((function(n){var r=n.name,s=n.type,c=l(r);if(e.attr(c)&&!o[r])return c+"='"+e.attr(c)+"'";var u=o[r]||r;if("@"===s)return c+"='{{"+a+"$resolve."+u+"}}'";if("&"===s){var d=t.getResolvable(u),p=d&&d.data,f=p&&i.services.$injector.annotate(p)||[];return c+"='$resolve."+u+(Object(i.isArray)(p)?"["+(p.length-1)+"]":"")+"("+f.join(",")+")'"}return c+"='"+a+"$resolve."+u+"'"})).join(" "),u=l(n);return"<"+u+" "+c+">"},e}();var s=function(e){return Object(i.isObject)(e.bindToController)?l(e.bindToController):l(e.scope)},l=function(e){return Object.keys(e||{}).map((function(t){return[t,/^([=<@&])[?]?(.*)/.exec(e[t])]})).filter((function(e){return Object(i.isDefined)(e)&&Object(i.isArray)(e[1])})).map((function(e){return{name:e[1][2]||e[0],type:e[1][1]}}))},c=n(112),u=function(e){return function(t){var n=t[e],r="onExit"===e?"from":"to";return n?function(e,t){var o=new i.ResolveContext(e.treeChanges(r)).subContext(t.$$state()),a=Object(i.extend)(A(o),{$state$:t,$transition$:e});return i.services.$injector.invoke(n,this,a)}:void 0}},d=function(){function e(e){this._urlListeners=[],this.$locationProvider=e;var t=Object(i.val)(e);Object(i.createProxyFunctions)(t,this,t,["hashPrefix"])}return e.monkeyPatchPathParameterType=function(e){var t=e.urlMatcherFactory.type("path");t.encode=function(e){return null!=e?e.toString().replace(/(~|\/)/g,(function(e){return{"~":"~~","/":"~2F"}[e]})):e},t.decode=function(e){return null!=e?e.toString().replace(/(~~|~2F)/g,(function(e){return{"~~":"~","~2F":"/"}[e]})):e}},e.prototype.dispose=function(){},e.prototype.onChange=function(e){var t=this;return this._urlListeners.push(e),function(){return Object(i.removeFrom)(t._urlListeners)(e)}},e.prototype.html5Mode=function(){var e=this.$locationProvider.html5Mode();return(e=Object(i.isObject)(e)?e.enabled:e)&&this.$sniffer.history},e.prototype.baseHref=function(){return this._baseHref||(this._baseHref=this.$browser.baseHref()||this.$window.location.pathname)},e.prototype.url=function(e,t,n){return void 0===t&&(t=!1),Object(i.isDefined)(e)&&this.$location.url(e),t&&this.$location.replace(),n&&this.$location.state(n),this.$location.url()},e.prototype._runtimeServices=function(e,t,n,r,o){var a=this;this.$location=t,this.$sniffer=n,this.$browser=r,this.$window=o,e.$on("$locationChangeSuccess",(function(e){return a._urlListeners.forEach((function(t){return t(e)}))}));var s=Object(i.val)(t);Object(i.createProxyFunctions)(s,this,s,["replace","path","search","hash"]),Object(i.createProxyFunctions)(s,this,s,["port","protocol","host"])},e}(),p=n(113);r.a.module("ui.router.angular1",[]);var f=r.a.module("ui.router.init",["ng"]),h=r.a.module("ui.router.util",["ui.router.init"]),g=r.a.module("ui.router.router",["ui.router.util"]),m=r.a.module("ui.router.state",["ui.router.router","ui.router.util","ui.router.angular1"]),v=r.a.module("ui.router",["ui.router.init","ui.router.state","ui.router.angular1"]),b=(r.a.module("ui.router.compat",["ui.router"]),null);function y(e){(b=this.router=new i.UIRouter).stateProvider=new c.a(b.stateRegistry,b.stateService),b.stateRegistry.decorator("views",o.c),b.stateRegistry.decorator("onExit",u("onExit")),b.stateRegistry.decorator("onRetain",u("onRetain")),b.stateRegistry.decorator("onEnter",u("onEnter")),b.viewService._pluginapi._viewConfigFactory("ng1",Object(o.b)()),b.urlService.config._decodeParams=!1;var t=b.locationService=b.locationConfig=new d(e);function n(e,n,r,i,o,a,s){return t._runtimeServices(o,e,i,n,r),delete b.router,delete b.$get,b}return d.monkeyPatchPathParameterType(b),b.router=b,b.$get=n,n.$inject=["$location","$browser","$window","$sniffer","$rootScope","$http","$templateCache"],b}y.$inject=["$locationProvider"];var x=function(e){return["$uiRouterProvider",function(t){var n=t.router[e];return n.$get=function(){return n},n}]};function w(e,t,n){if(i.services.$injector=e,i.services.$q=t,!Object.prototype.hasOwnProperty.call(e,"strictDi"))try{e.invoke((function(e){}))}catch(t){e.strictDi=!!/strict mode/.exec(t&&t.toString())}n.stateRegistry.get().map((function(e){return e.$$state().resolvables})).reduce(i.unnestR,[]).filter((function(e){return"deferred"===e.deps})).forEach((function(t){return t.deps=e.annotate(t.resolveFn,e.strictDi)}))}w.$inject=["$injector","$q","$uiRouter"];function k(e){e.$watch((function(){i.trace.approximateDigests++}))}k.$inject=["$rootScope"],f.provider("$uiRouter",y),g.provider("$urlRouter",["$uiRouterProvider",function(e){return e.urlRouterProvider=new p.a(e)}]),h.provider("$urlService",x("urlService")),h.provider("$urlMatcherFactory",["$uiRouterProvider",function(){return b.urlMatcherFactory}]),h.provider("$templateFactory",(function(){return new a})),m.provider("$stateRegistry",x("stateRegistry")),m.provider("$uiRouterGlobals",x("globals")),m.provider("$transitions",x("transitionService")),m.provider("$state",["$uiRouterProvider",function(){return Object(i.extend)(b.stateProvider,{$get:function(){return b.stateService}})}]),m.factory("$stateParams",["$uiRouter",function(e){return e.globals.params}]),v.factory("$view",(function(){return b.viewService})),v.service("$trace",(function(){return i.trace})),v.run(k),h.run(["$urlMatcherFactory",function(e){}]),m.run(["$state",function(e){}]),g.run(["$urlRouter",function(e){}]),f.run(w);var A=function(e){return e.getTokens().filter(i.isString).map((function(t){var n=e.getResolvable(t);return[t,"NOWAIT"===e.getPolicy(n).async?n.promise:n.data]})).reduce(i.applyPairs,{})}},function(e,t,n){"use strict";n.d(t,"b",(function(){return o})),n.d(t,"a",(function(){return a}));var r=n(6),i=function(e){var t=e.router;var n=e.entering().filter((function(e){return!!e.$$state().lazyLoad})).map((function(t){return a(e,t)}));return r.b.$q.all(n).then((function(){if("url"!==e.originalTransition().options().source){var n=e.targetState();return t.stateService.target(n.identifier(),n.params(),n.options())}var r=t.urlService,i=r.match(r.parts()),o=i&&i.rule;if(o&&"STATE"===o.type){var a=o.state,s=i.match;return t.stateService.target(a,s,e.options())}t.urlService.sync()}))},o=function(e){return e.onBefore({entering:function(e){return!!e.lazyLoad}},i)};function a(e,t){var n=t.$$state().lazyLoad,i=n._promise;if(!i){i=n._promise=r.b.$q.when(n(e,t)).then((function(t){t&&Array.isArray(t.states)&&t.states.forEach((function(t){return e.router.stateRegistry.register(t)}));return t})).then((function(e){return delete t.lazyLoad,delete t.$$state().lazyLoad,delete n._promise,e}),(function(e){return delete n._promise,r.b.$q.reject(e)}))}return i}},function(e,t,n){"use strict";var r=n(22);e.exports=i;function i(e){this._isDirected=!r.has(e,"directed")||e.directed,this._isMultigraph=!!r.has(e,"multigraph")&&e.multigraph,this._isCompound=!!r.has(e,"compound")&&e.compound,this._label=void 0,this._defaultNodeLabelFn=r.constant(void 0),this._defaultEdgeLabelFn=r.constant(void 0),this._nodes={},this._isCompound&&(this._parent={},this._children={},this._children["\0"]={}),this._in={},this._preds={},this._out={},this._sucs={},this._edgeObjs={},this._edgeLabels={}}function o(e,t){e[t]?e[t]++:e[t]=1}function a(e,t){--e[t]||delete e[t]}function s(e,t,n,i){var o=""+t,a=""+n;if(!e&&o>a){var s=o;o=a,a=s}return o+""+a+""+(r.isUndefined(i)?"\0":i)}function l(e,t,n,r){var i=""+t,o=""+n;if(!e&&i>o){var a=i;i=o,o=a}var s={v:i,w:o};return r&&(s.name=r),s}function c(e,t){return s(e,t.v,t.w,t.name)}i.prototype._nodeCount=0,i.prototype._edgeCount=0,i.prototype.isDirected=function(){return this._isDirected},i.prototype.isMultigraph=function(){return this._isMultigraph},i.prototype.isCompound=function(){return this._isCompound},i.prototype.setGraph=function(e){return this._label=e,this},i.prototype.graph=function(){return this._label},i.prototype.setDefaultNodeLabel=function(e){return r.isFunction(e)||(e=r.constant(e)),this._defaultNodeLabelFn=e,this},i.prototype.nodeCount=function(){return this._nodeCount},i.prototype.nodes=function(){return r.keys(this._nodes)},i.prototype.sources=function(){var e=this;return r.filter(this.nodes(),(function(t){return r.isEmpty(e._in[t])}))},i.prototype.sinks=function(){var e=this;return r.filter(this.nodes(),(function(t){return r.isEmpty(e._out[t])}))},i.prototype.setNodes=function(e,t){var n=arguments,i=this;return r.each(e,(function(e){n.length>1?i.setNode(e,t):i.setNode(e)})),this},i.prototype.setNode=function(e,t){return r.has(this._nodes,e)?(arguments.length>1&&(this._nodes[e]=t),this):(this._nodes[e]=arguments.length>1?t:this._defaultNodeLabelFn(e),this._isCompound&&(this._parent[e]="\0",this._children[e]={},this._children["\0"][e]=!0),this._in[e]={},this._preds[e]={},this._out[e]={},this._sucs[e]={},++this._nodeCount,this)},i.prototype.node=function(e){return this._nodes[e]},i.prototype.hasNode=function(e){return r.has(this._nodes,e)},i.prototype.removeNode=function(e){var t=this;if(r.has(this._nodes,e)){var n=function(e){t.removeEdge(t._edgeObjs[e])};delete this._nodes[e],this._isCompound&&(this._removeFromParentsChildList(e),delete this._parent[e],r.each(this.children(e),(function(e){t.setParent(e)})),delete this._children[e]),r.each(r.keys(this._in[e]),n),delete this._in[e],delete this._preds[e],r.each(r.keys(this._out[e]),n),delete this._out[e],delete this._sucs[e],--this._nodeCount}return this},i.prototype.setParent=function(e,t){if(!this._isCompound)throw new Error("Cannot set parent in a non-compound graph");if(r.isUndefined(t))t="\0";else{for(var n=t+="";!r.isUndefined(n);n=this.parent(n))if(n===e)throw new Error("Setting "+t+" as parent of "+e+" would create a cycle");this.setNode(t)}return this.setNode(e),this._removeFromParentsChildList(e),this._parent[e]=t,this._children[t][e]=!0,this},i.prototype._removeFromParentsChildList=function(e){delete this._children[this._parent[e]][e]},i.prototype.parent=function(e){if(this._isCompound){var t=this._parent[e];if("\0"!==t)return t}},i.prototype.children=function(e){if(r.isUndefined(e)&&(e="\0"),this._isCompound){var t=this._children[e];if(t)return r.keys(t)}else{if("\0"===e)return this.nodes();if(this.hasNode(e))return[]}},i.prototype.predecessors=function(e){var t=this._preds[e];if(t)return r.keys(t)},i.prototype.successors=function(e){var t=this._sucs[e];if(t)return r.keys(t)},i.prototype.neighbors=function(e){var t=this.predecessors(e);if(t)return r.union(t,this.successors(e))},i.prototype.isLeaf=function(e){return 0===(this.isDirected()?this.successors(e):this.neighbors(e)).length},i.prototype.filterNodes=function(e){var t=new this.constructor({directed:this._isDirected,multigraph:this._isMultigraph,compound:this._isCompound});t.setGraph(this.graph());var n=this;r.each(this._nodes,(function(n,r){e(r)&&t.setNode(r,n)})),r.each(this._edgeObjs,(function(e){t.hasNode(e.v)&&t.hasNode(e.w)&&t.setEdge(e,n.edge(e))}));var i={};return this._isCompound&&r.each(t.nodes(),(function(e){t.setParent(e,function e(r){var o=n.parent(r);return void 0===o||t.hasNode(o)?(i[r]=o,o):o in i?i[o]:e(o)}(e))})),t},i.prototype.setDefaultEdgeLabel=function(e){return r.isFunction(e)||(e=r.constant(e)),this._defaultEdgeLabelFn=e,this},i.prototype.edgeCount=function(){return this._edgeCount},i.prototype.edges=function(){return r.values(this._edgeObjs)},i.prototype.setPath=function(e,t){var n=this,i=arguments;return r.reduce(e,(function(e,r){return i.length>1?n.setEdge(e,r,t):n.setEdge(e,r),r})),this},i.prototype.setEdge=function(){var e,t,n,i,a=!1,c=arguments[0];"object"==typeof c&&null!==c&&"v"in c?(e=c.v,t=c.w,n=c.name,2===arguments.length&&(i=arguments[1],a=!0)):(e=c,t=arguments[1],n=arguments[3],arguments.length>2&&(i=arguments[2],a=!0)),e=""+e,t=""+t,r.isUndefined(n)||(n=""+n);var u=s(this._isDirected,e,t,n);if(r.has(this._edgeLabels,u))return a&&(this._edgeLabels[u]=i),this;if(!r.isUndefined(n)&&!this._isMultigraph)throw new Error("Cannot set a named edge when isMultigraph = false");this.setNode(e),this.setNode(t),this._edgeLabels[u]=a?i:this._defaultEdgeLabelFn(e,t,n);var d=l(this._isDirected,e,t,n);return e=d.v,t=d.w,Object.freeze(d),this._edgeObjs[u]=d,o(this._preds[t],e),o(this._sucs[e],t),this._in[t][u]=d,this._out[e][u]=d,this._edgeCount++,this},i.prototype.edge=function(e,t,n){var r=1===arguments.length?c(this._isDirected,arguments[0]):s(this._isDirected,e,t,n);return this._edgeLabels[r]},i.prototype.hasEdge=function(e,t,n){var i=1===arguments.length?c(this._isDirected,arguments[0]):s(this._isDirected,e,t,n);return r.has(this._edgeLabels,i)},i.prototype.removeEdge=function(e,t,n){var r=1===arguments.length?c(this._isDirected,arguments[0]):s(this._isDirected,e,t,n),i=this._edgeObjs[r];return i&&(e=i.v,t=i.w,delete this._edgeLabels[r],delete this._edgeObjs[r],a(this._preds[t],e),a(this._sucs[e],t),delete this._in[t][r],delete this._out[e][r],this._edgeCount--),this},i.prototype.inEdges=function(e,t){var n=this._in[e];if(n){var i=r.values(n);return t?r.filter(i,(function(e){return e.v===t})):i}},i.prototype.outEdges=function(e,t){var n=this._out[e];if(n){var i=r.values(n);return t?r.filter(i,(function(e){return e.w===t})):i}},i.prototype.nodeEdges=function(e,t){var n=this.inEdges(e,t);if(n)return n.concat(this.outEdges(e,t))}},function(e,t,n){var r=n(46)(n(31),"Map");e.exports=r},function(e,t,n){var r=n(265),i=n(272),o=n(274),a=n(275),s=n(276);function l(e){var t=-1,n=null==e?0:e.length;for(this.clear();++t-1&&e%1==0&&e<=9007199254740991}},function(e,t,n){(function(e){var r=n(150),i=t&&!t.nodeType&&t,o=i&&"object"==typeof e&&e&&!e.nodeType&&e,a=o&&o.exports===i&&r.process,s=function(){try{var e=o&&o.require&&o.require("util").types;return e||a&&a.binding&&a.binding("util")}catch(e){}}();e.exports=s}).call(this,n(80)(e))},function(e,t,n){var r=n(83),i=n(282),o=Object.prototype.hasOwnProperty;e.exports=function(e){if(!r(e))return i(e);var t=[];for(var n in Object(e))o.call(e,n)&&"constructor"!=n&&t.push(n);return t}},function(e,t,n){var r=n(157),i=n(158),o=Object.prototype.propertyIsEnumerable,a=Object.getOwnPropertySymbols,s=a?function(e){return null==e?[]:(e=Object(e),r(a(e),(function(t){return o.call(e,t)})))}:i;e.exports=s},function(e,t){e.exports=function(e,t){for(var n=-1,r=t.length,i=e.length;++n0&&o(u)?n>1?e(u,n-1,o,a,s):r(s,u):a||(s[s.length]=u)}return s}},function(e,t,n){var r=n(61);e.exports=function(e,t,n){for(var i=-1,o=e.length;++iu))return!1;var p=l.get(e),f=l.get(t);if(p&&f)return p==t&&f==e;var h=-1,g=!0,m=2&n?new r:void 0;for(l.set(e,t),l.set(t,e);++h0&&(o=l.removeMin(),(a=s[o]).distance!==Number.POSITIVE_INFINITY);)r(o).forEach(c);return s}(e,String(t),n||o,r||function(t){return e.outEdges(t)})};var o=r.constant(1)},function(e,t,n){var r=n(22);function i(){this._arr=[],this._keyIndices={}}e.exports=i,i.prototype.size=function(){return this._arr.length},i.prototype.keys=function(){return this._arr.map((function(e){return e.key}))},i.prototype.has=function(e){return r.has(this._keyIndices,e)},i.prototype.priority=function(e){var t=this._keyIndices[e];if(void 0!==t)return this._arr[t].priority},i.prototype.min=function(){if(0===this.size())throw new Error("Queue underflow");return this._arr[0].key},i.prototype.add=function(e,t){var n=this._keyIndices;if(e=String(e),!r.has(n,e)){var i=this._arr,o=i.length;return n[e]=o,i.push({key:e,priority:t}),this._decrease(o),!0}return!1},i.prototype.removeMin=function(){this._swap(0,this._arr.length-1);var e=this._arr.pop();return delete this._keyIndices[e.key],this._heapify(0),e.key},i.prototype.decrease=function(e,t){var n=this._keyIndices[e];if(t>this._arr[n].priority)throw new Error("New priority is greater than current priority. Key: "+e+" Old: "+this._arr[n].priority+" New: "+t);this._arr[n].priority=t,this._decrease(n)},i.prototype._heapify=function(e){var t=this._arr,n=2*e,r=n+1,i=e;n>1].priority @@ -38,38 +190,61 @@ * Released under MIT license * Based on Underscore.js 1.8.3 * Copyright Jeremy Ashkenas, DocumentCloud and Investigative Reporters & Editors - */(function(){var o="Expected a function",a="__lodash_placeholder__",s=[["ary",128],["bind",1],["bindKey",2],["curry",8],["curryRight",16],["flip",512],["partial",32],["partialRight",64],["rearg",256]],l="[object Arguments]",c="[object Array]",u="[object Boolean]",d="[object Date]",p="[object Error]",f="[object Function]",h="[object GeneratorFunction]",g="[object Map]",m="[object Number]",v="[object Object]",b="[object RegExp]",y="[object Set]",x="[object String]",w="[object Symbol]",k="[object WeakMap]",A="[object ArrayBuffer]",E="[object DataView]",S="[object Float32Array]",$="[object Float64Array]",C="[object Int8Array]",_="[object Int16Array]",O="[object Int32Array]",T="[object Uint8Array]",j="[object Uint16Array]",P="[object Uint32Array]",D=/\b__p \+= '';/g,R=/\b(__p \+=) '' \+/g,I=/(__e\(.*?\)|\b__t\)) \+\n'';/g,N=/&(?:amp|lt|gt|quot|#39);/g,M=/[&<>"']/g,z=RegExp(N.source),L=RegExp(M.source),B=/<%-([\s\S]+?)%>/g,F=/<%([\s\S]+?)%>/g,q=/<%=([\s\S]+?)%>/g,V=/\.|\[(?:[^[\]]*|(["'])(?:(?!\1)[^\\]|\\.)*?\1)\]/,U=/^\w*$/,H=/[^.[\]]+|\[(?:(-?\d+(?:\.\d+)?)|(["'])((?:(?!\2)[^\\]|\\.)*?)\2)\]|(?=(?:\.|\[\])(?:\.|\[\]|$))/g,G=/[\\^$.*+?()[\]{}|]/g,W=RegExp(G.source),Y=/^\s+/,X=/\s/,Z=/\{(?:\n\/\* \[wrapped with .+\] \*\/)?\n?/,Q=/\{\n\/\* \[wrapped with (.+)\] \*/,J=/,? & /,K=/[^\x00-\x2f\x3a-\x40\x5b-\x60\x7b-\x7f]+/g,ee=/[()=,{}\[\]\/\s]/,te=/\\(\\)?/g,ne=/\$\{([^\\}]*(?:\\.[^\\}]*)*)\}/g,re=/\w*$/,ie=/^[-+]0x[0-9a-f]+$/i,oe=/^0b[01]+$/i,ae=/^\[object .+?Constructor\]$/,se=/^0o[0-7]+$/i,le=/^(?:0|[1-9]\d*)$/,ce=/[\xc0-\xd6\xd8-\xf6\xf8-\xff\u0100-\u017f]/g,ue=/($^)/,de=/['\n\r\u2028\u2029\\]/g,pe="\\u0300-\\u036f\\ufe20-\\ufe2f\\u20d0-\\u20ff",fe="\\xac\\xb1\\xd7\\xf7\\x00-\\x2f\\x3a-\\x40\\x5b-\\x60\\x7b-\\xbf\\u2000-\\u206f \\t\\x0b\\f\\xa0\\ufeff\\n\\r\\u2028\\u2029\\u1680\\u180e\\u2000\\u2001\\u2002\\u2003\\u2004\\u2005\\u2006\\u2007\\u2008\\u2009\\u200a\\u202f\\u205f\\u3000",he="[\\ud800-\\udfff]",ge="["+fe+"]",me="["+pe+"]",ve="\\d+",be="[\\u2700-\\u27bf]",ye="[a-z\\xdf-\\xf6\\xf8-\\xff]",xe="[^\\ud800-\\udfff"+fe+ve+"\\u2700-\\u27bfa-z\\xdf-\\xf6\\xf8-\\xffA-Z\\xc0-\\xd6\\xd8-\\xde]",we="\\ud83c[\\udffb-\\udfff]",ke="[^\\ud800-\\udfff]",Ae="(?:\\ud83c[\\udde6-\\uddff]){2}",Ee="[\\ud800-\\udbff][\\udc00-\\udfff]",Se="[A-Z\\xc0-\\xd6\\xd8-\\xde]",$e="(?:"+ye+"|"+xe+")",Ce="(?:"+Se+"|"+xe+")",_e="(?:"+me+"|"+we+")"+"?",Oe="[\\ufe0e\\ufe0f]?"+_e+("(?:\\u200d(?:"+[ke,Ae,Ee].join("|")+")[\\ufe0e\\ufe0f]?"+_e+")*"),Te="(?:"+[be,Ae,Ee].join("|")+")"+Oe,je="(?:"+[ke+me+"?",me,Ae,Ee,he].join("|")+")",Pe=RegExp("['’]","g"),De=RegExp(me,"g"),Re=RegExp(we+"(?="+we+")|"+je+Oe,"g"),Ie=RegExp([Se+"?"+ye+"+(?:['’](?:d|ll|m|re|s|t|ve))?(?="+[ge,Se,"$"].join("|")+")",Ce+"+(?:['’](?:D|LL|M|RE|S|T|VE))?(?="+[ge,Se+$e,"$"].join("|")+")",Se+"?"+$e+"+(?:['’](?:d|ll|m|re|s|t|ve))?",Se+"+(?:['’](?:D|LL|M|RE|S|T|VE))?","\\d*(?:1ST|2ND|3RD|(?![123])\\dTH)(?=\\b|[a-z_])","\\d*(?:1st|2nd|3rd|(?![123])\\dth)(?=\\b|[A-Z_])",ve,Te].join("|"),"g"),Ne=RegExp("[\\u200d\\ud800-\\udfff"+pe+"\\ufe0e\\ufe0f]"),Me=/[a-z][A-Z]|[A-Z]{2}[a-z]|[0-9][a-zA-Z]|[a-zA-Z][0-9]|[^a-zA-Z0-9 ]/,ze=["Array","Buffer","DataView","Date","Error","Float32Array","Float64Array","Function","Int8Array","Int16Array","Int32Array","Map","Math","Object","Promise","RegExp","Set","String","Symbol","TypeError","Uint8Array","Uint8ClampedArray","Uint16Array","Uint32Array","WeakMap","_","clearTimeout","isFinite","parseInt","setTimeout"],Le=-1,Be={};Be[S]=Be[$]=Be[C]=Be[_]=Be[O]=Be[T]=Be["[object Uint8ClampedArray]"]=Be[j]=Be[P]=!0,Be[l]=Be[c]=Be[A]=Be[u]=Be[E]=Be[d]=Be[p]=Be[f]=Be[g]=Be[m]=Be[v]=Be[b]=Be[y]=Be[x]=Be[k]=!1;var Fe={};Fe[l]=Fe[c]=Fe[A]=Fe[E]=Fe[u]=Fe[d]=Fe[S]=Fe[$]=Fe[C]=Fe[_]=Fe[O]=Fe[g]=Fe[m]=Fe[v]=Fe[b]=Fe[y]=Fe[x]=Fe[w]=Fe[T]=Fe["[object Uint8ClampedArray]"]=Fe[j]=Fe[P]=!0,Fe[p]=Fe[f]=Fe[k]=!1;var qe={"\\":"\\","'":"'","\n":"n","\r":"r","\u2028":"u2028","\u2029":"u2029"},Ve=parseFloat,Ue=parseInt,He="object"==typeof e&&e&&e.Object===Object&&e,Ge="object"==typeof self&&self&&self.Object===Object&&self,We=He||Ge||Function("return this")(),Ye=t&&!t.nodeType&&t,Xe=Ye&&"object"==typeof r&&r&&!r.nodeType&&r,Ze=Xe&&Xe.exports===Ye,Qe=Ze&&He.process,Je=function(){try{var e=Xe&&Xe.require&&Xe.require("util").types;return e||Qe&&Qe.binding&&Qe.binding("util")}catch(e){}}(),Ke=Je&&Je.isArrayBuffer,et=Je&&Je.isDate,tt=Je&&Je.isMap,nt=Je&&Je.isRegExp,rt=Je&&Je.isSet,it=Je&&Je.isTypedArray;function ot(e,t,n){switch(n.length){case 0:return e.call(t);case 1:return e.call(t,n[0]);case 2:return e.call(t,n[0],n[1]);case 3:return e.call(t,n[0],n[1],n[2])}return e.apply(t,n)}function at(e,t,n,r){for(var i=-1,o=null==e?0:e.length;++i-1}function pt(e,t,n){for(var r=-1,i=null==e?0:e.length;++r-1;);return n}function It(e,t){for(var n=e.length;n--&&wt(t,e[n],0)>-1;);return n}function Nt(e,t){for(var n=e.length,r=0;n--;)e[n]===t&&++r;return r}var Mt=$t({"À":"A","Á":"A","Â":"A","Ã":"A","Ä":"A","Å":"A","à":"a","á":"a","â":"a","ã":"a","ä":"a","å":"a","Ç":"C","ç":"c","Ð":"D","ð":"d","È":"E","É":"E","Ê":"E","Ë":"E","è":"e","é":"e","ê":"e","ë":"e","Ì":"I","Í":"I","Î":"I","Ï":"I","ì":"i","í":"i","î":"i","ï":"i","Ñ":"N","ñ":"n","Ò":"O","Ó":"O","Ô":"O","Õ":"O","Ö":"O","Ø":"O","ò":"o","ó":"o","ô":"o","õ":"o","ö":"o","ø":"o","Ù":"U","Ú":"U","Û":"U","Ü":"U","ù":"u","ú":"u","û":"u","ü":"u","Ý":"Y","ý":"y","ÿ":"y","Æ":"Ae","æ":"ae","Þ":"Th","þ":"th","ß":"ss","Ā":"A","Ă":"A","Ą":"A","ā":"a","ă":"a","ą":"a","Ć":"C","Ĉ":"C","Ċ":"C","Č":"C","ć":"c","ĉ":"c","ċ":"c","č":"c","Ď":"D","Đ":"D","ď":"d","đ":"d","Ē":"E","Ĕ":"E","Ė":"E","Ę":"E","Ě":"E","ē":"e","ĕ":"e","ė":"e","ę":"e","ě":"e","Ĝ":"G","Ğ":"G","Ġ":"G","Ģ":"G","ĝ":"g","ğ":"g","ġ":"g","ģ":"g","Ĥ":"H","Ħ":"H","ĥ":"h","ħ":"h","Ĩ":"I","Ī":"I","Ĭ":"I","Į":"I","İ":"I","ĩ":"i","ī":"i","ĭ":"i","į":"i","ı":"i","Ĵ":"J","ĵ":"j","Ķ":"K","ķ":"k","ĸ":"k","Ĺ":"L","Ļ":"L","Ľ":"L","Ŀ":"L","Ł":"L","ĺ":"l","ļ":"l","ľ":"l","ŀ":"l","ł":"l","Ń":"N","Ņ":"N","Ň":"N","Ŋ":"N","ń":"n","ņ":"n","ň":"n","ŋ":"n","Ō":"O","Ŏ":"O","Ő":"O","ō":"o","ŏ":"o","ő":"o","Ŕ":"R","Ŗ":"R","Ř":"R","ŕ":"r","ŗ":"r","ř":"r","Ś":"S","Ŝ":"S","Ş":"S","Š":"S","ś":"s","ŝ":"s","ş":"s","š":"s","Ţ":"T","Ť":"T","Ŧ":"T","ţ":"t","ť":"t","ŧ":"t","Ũ":"U","Ū":"U","Ŭ":"U","Ů":"U","Ű":"U","Ų":"U","ũ":"u","ū":"u","ŭ":"u","ů":"u","ű":"u","ų":"u","Ŵ":"W","ŵ":"w","Ŷ":"Y","ŷ":"y","Ÿ":"Y","Ź":"Z","Ż":"Z","Ž":"Z","ź":"z","ż":"z","ž":"z","IJ":"IJ","ij":"ij","Œ":"Oe","œ":"oe","ʼn":"'n","ſ":"s"}),zt=$t({"&":"&","<":"<",">":">",'"':""","'":"'"});function Lt(e){return"\\"+qe[e]}function Bt(e){return Ne.test(e)}function Ft(e){var t=-1,n=Array(e.size);return e.forEach((function(e,r){n[++t]=[r,e]})),n}function qt(e,t){return function(n){return e(t(n))}}function Vt(e,t){for(var n=-1,r=e.length,i=0,o=[];++n",""":'"',"'":"'"});var Zt=function e(t){var n,r=(t=null==t?We:Zt.defaults(We.Object(),t,Zt.pick(We,ze))).Array,i=t.Date,X=t.Error,pe=t.Function,fe=t.Math,he=t.Object,ge=t.RegExp,me=t.String,ve=t.TypeError,be=r.prototype,ye=pe.prototype,xe=he.prototype,we=t["__core-js_shared__"],ke=ye.toString,Ae=xe.hasOwnProperty,Ee=0,Se=(n=/[^.]+$/.exec(we&&we.keys&&we.keys.IE_PROTO||""))?"Symbol(src)_1."+n:"",$e=xe.toString,Ce=ke.call(he),_e=We._,Oe=ge("^"+ke.call(Ae).replace(G,"\\$&").replace(/hasOwnProperty|(function).*?(?=\\\()| for .+?(?=\\\])/g,"$1.*?")+"$"),Te=Ze?t.Buffer:void 0,je=t.Symbol,Re=t.Uint8Array,Ne=Te?Te.allocUnsafe:void 0,qe=qt(he.getPrototypeOf,he),He=he.create,Ge=xe.propertyIsEnumerable,Ye=be.splice,Xe=je?je.isConcatSpreadable:void 0,Qe=je?je.iterator:void 0,Je=je?je.toStringTag:void 0,bt=function(){try{var e=eo(he,"defineProperty");return e({},"",{}),e}catch(e){}}(),$t=t.clearTimeout!==We.clearTimeout&&t.clearTimeout,Qt=i&&i.now!==We.Date.now&&i.now,Jt=t.setTimeout!==We.setTimeout&&t.setTimeout,Kt=fe.ceil,en=fe.floor,tn=he.getOwnPropertySymbols,nn=Te?Te.isBuffer:void 0,rn=t.isFinite,on=be.join,an=qt(he.keys,he),sn=fe.max,ln=fe.min,cn=i.now,un=t.parseInt,dn=fe.random,pn=be.reverse,fn=eo(t,"DataView"),hn=eo(t,"Map"),gn=eo(t,"Promise"),mn=eo(t,"Set"),vn=eo(t,"WeakMap"),bn=eo(he,"create"),yn=vn&&new vn,xn={},wn=_o(fn),kn=_o(hn),An=_o(gn),En=_o(mn),Sn=_o(vn),$n=je?je.prototype:void 0,Cn=$n?$n.valueOf:void 0,_n=$n?$n.toString:void 0;function On(e){if(Ha(e)&&!Ra(e)&&!(e instanceof Dn)){if(e instanceof Pn)return e;if(Ae.call(e,"__wrapped__"))return Oo(e)}return new Pn(e)}var Tn=function(){function e(){}return function(t){if(!Ua(t))return{};if(He)return He(t);e.prototype=t;var n=new e;return e.prototype=void 0,n}}();function jn(){}function Pn(e,t){this.__wrapped__=e,this.__actions__=[],this.__chain__=!!t,this.__index__=0,this.__values__=void 0}function Dn(e){this.__wrapped__=e,this.__actions__=[],this.__dir__=1,this.__filtered__=!1,this.__iteratees__=[],this.__takeCount__=4294967295,this.__views__=[]}function Rn(e){var t=-1,n=null==e?0:e.length;for(this.clear();++t=t?e:t)),e}function Qn(e,t,n,r,i,o){var a,s=1&t,c=2&t,p=4&t;if(n&&(a=i?n(e,r,i,o):n(e)),void 0!==a)return a;if(!Ua(e))return e;var k=Ra(e);if(k){if(a=function(e){var t=e.length,n=new e.constructor(t);t&&"string"==typeof e[0]&&Ae.call(e,"index")&&(n.index=e.index,n.input=e.input);return n}(e),!s)return bi(e,a)}else{var D=ro(e),R=D==f||D==h;if(za(e))return pi(e,s);if(D==v||D==l||R&&!i){if(a=c||R?{}:oo(e),!s)return c?function(e,t){return yi(e,no(e),t)}(e,function(e,t){return e&&yi(t,ks(t),e)}(a,e)):function(e,t){return yi(e,to(e),t)}(e,Wn(a,e))}else{if(!Fe[D])return i?e:{};a=function(e,t,n){var r=e.constructor;switch(t){case A:return fi(e);case u:case d:return new r(+e);case E:return function(e,t){var n=t?fi(e.buffer):e.buffer;return new e.constructor(n,e.byteOffset,e.byteLength)}(e,n);case S:case $:case C:case _:case O:case T:case"[object Uint8ClampedArray]":case j:case P:return hi(e,n);case g:return new r;case m:case x:return new r(e);case b:return function(e){var t=new e.constructor(e.source,re.exec(e));return t.lastIndex=e.lastIndex,t}(e);case y:return new r;case w:return i=e,Cn?he(Cn.call(i)):{}}var i}(e,D,s)}}o||(o=new zn);var I=o.get(e);if(I)return I;o.set(e,a),Za(e)?e.forEach((function(r){a.add(Qn(r,t,n,r,e,o))})):Ga(e)&&e.forEach((function(r,i){a.set(i,Qn(r,t,n,i,e,o))}));var N=k?void 0:(p?c?Wi:Gi:c?ks:ws)(e);return st(N||e,(function(r,i){N&&(r=e[i=r]),Un(a,i,Qn(r,t,n,i,e,o))})),a}function Jn(e,t,n){var r=n.length;if(null==e)return!r;for(e=he(e);r--;){var i=n[r],o=t[i],a=e[i];if(void 0===a&&!(i in e)||!o(a))return!1}return!0}function Kn(e,t,n){if("function"!=typeof e)throw new ve(o);return wo((function(){e.apply(void 0,n)}),t)}function er(e,t,n,r){var i=-1,o=dt,a=!0,s=e.length,l=[],c=t.length;if(!s)return l;n&&(t=ft(t,jt(n))),r?(o=pt,a=!1):t.length>=200&&(o=Dt,a=!1,t=new Mn(t));e:for(;++i-1},In.prototype.set=function(e,t){var n=this.__data__,r=Hn(n,e);return r<0?(++this.size,n.push([e,t])):n[r][1]=t,this},Nn.prototype.clear=function(){this.size=0,this.__data__={hash:new Rn,map:new(hn||In),string:new Rn}},Nn.prototype.delete=function(e){var t=Ji(this,e).delete(e);return this.size-=t?1:0,t},Nn.prototype.get=function(e){return Ji(this,e).get(e)},Nn.prototype.has=function(e){return Ji(this,e).has(e)},Nn.prototype.set=function(e,t){var n=Ji(this,e),r=n.size;return n.set(e,t),this.size+=n.size==r?0:1,this},Mn.prototype.add=Mn.prototype.push=function(e){return this.__data__.set(e,"__lodash_hash_undefined__"),this},Mn.prototype.has=function(e){return this.__data__.has(e)},zn.prototype.clear=function(){this.__data__=new In,this.size=0},zn.prototype.delete=function(e){var t=this.__data__,n=t.delete(e);return this.size=t.size,n},zn.prototype.get=function(e){return this.__data__.get(e)},zn.prototype.has=function(e){return this.__data__.has(e)},zn.prototype.set=function(e,t){var n=this.__data__;if(n instanceof In){var r=n.__data__;if(!hn||r.length<199)return r.push([e,t]),this.size=++n.size,this;n=this.__data__=new Nn(r)}return n.set(e,t),this.size=n.size,this};var tr=ki(cr),nr=ki(ur,!0);function rr(e,t){var n=!0;return tr(e,(function(e,r,i){return n=!!t(e,r,i)})),n}function ir(e,t,n){for(var r=-1,i=e.length;++r0&&n(s)?t>1?ar(s,t-1,n,r,i):ht(i,s):r||(i[i.length]=s)}return i}var sr=Ai(),lr=Ai(!0);function cr(e,t){return e&&sr(e,t,ws)}function ur(e,t){return e&&lr(e,t,ws)}function dr(e,t){return ut(t,(function(t){return Fa(e[t])}))}function pr(e,t){for(var n=0,r=(t=li(t,e)).length;null!=e&&nt}function mr(e,t){return null!=e&&Ae.call(e,t)}function vr(e,t){return null!=e&&t in he(e)}function br(e,t,n){for(var i=n?pt:dt,o=e[0].length,a=e.length,s=a,l=r(a),c=1/0,u=[];s--;){var d=e[s];s&&t&&(d=ft(d,jt(t))),c=ln(d.length,c),l[s]=!n&&(t||o>=120&&d.length>=120)?new Mn(s&&d):void 0}d=e[0];var p=-1,f=l[0];e:for(;++p=s)return l;var c=n[r];return l*("desc"==c?-1:1)}}return e.index-t.index}(e,t,n)}))}function Rr(e,t,n){for(var r=-1,i=t.length,o={};++r-1;)s!==e&&Ye.call(s,l,1),Ye.call(e,l,1);return e}function Nr(e,t){for(var n=e?t.length:0,r=n-1;n--;){var i=t[n];if(n==r||i!==o){var o=i;so(i)?Ye.call(e,i,1):ei(e,i)}}return e}function Mr(e,t){return e+en(dn()*(t-e+1))}function zr(e,t){var n="";if(!e||t<1||t>9007199254740991)return n;do{t%2&&(n+=e),(t=en(t/2))&&(e+=e)}while(t);return n}function Lr(e,t){return ko(mo(e,t,Ws),e+"")}function Br(e){return Bn(Ts(e))}function Fr(e,t){var n=Ts(e);return So(n,Zn(t,0,n.length))}function qr(e,t,n,r){if(!Ua(e))return e;for(var i=-1,o=(t=li(t,e)).length,a=o-1,s=e;null!=s&&++io?0:o+t),(n=n>o?o:n)<0&&(n+=o),o=t>n?0:n-t>>>0,t>>>=0;for(var a=r(o);++i>>1,a=e[o];null!==a&&!Ja(a)&&(n?a<=t:a=200){var c=t?null:zi(e);if(c)return Ut(c);a=!1,i=Dt,l=new Mn}else l=t?[]:s;e:for(;++r=r?e:Gr(e,t,n)}var di=$t||function(e){return We.clearTimeout(e)};function pi(e,t){if(t)return e.slice();var n=e.length,r=Ne?Ne(n):new e.constructor(n);return e.copy(r),r}function fi(e){var t=new e.constructor(e.byteLength);return new Re(t).set(new Re(e)),t}function hi(e,t){var n=t?fi(e.buffer):e.buffer;return new e.constructor(n,e.byteOffset,e.length)}function gi(e,t){if(e!==t){var n=void 0!==e,r=null===e,i=e==e,o=Ja(e),a=void 0!==t,s=null===t,l=t==t,c=Ja(t);if(!s&&!c&&!o&&e>t||o&&a&&l&&!s&&!c||r&&a&&l||!n&&l||!i)return 1;if(!r&&!o&&!c&&e1?n[i-1]:void 0,a=i>2?n[2]:void 0;for(o=e.length>3&&"function"==typeof o?(i--,o):void 0,a&&lo(n[0],n[1],a)&&(o=i<3?void 0:o,i=1),t=he(t);++r-1?i[o?t[a]:a]:void 0}}function _i(e){return Hi((function(t){var n=t.length,r=n,i=Pn.prototype.thru;for(e&&t.reverse();r--;){var a=t[r];if("function"!=typeof a)throw new ve(o);if(i&&!s&&"wrapper"==Xi(a))var s=new Pn([],!0)}for(r=s?r:n;++r1&&y.reverse(),d&&cs))return!1;var c=o.get(e),u=o.get(t);if(c&&u)return c==t&&u==e;var d=-1,p=!0,f=2&n?new Mn:void 0;for(o.set(e,t),o.set(t,e);++d-1&&e%1==0&&e1?"& ":"")+t[r],t=t.join(n>2?", ":" "),e.replace(Z,"{\n/* [wrapped with "+t+"] */\n")}(r,function(e,t){return st(s,(function(n){var r="_."+n[0];t&n[1]&&!dt(e,r)&&e.push(r)})),e.sort()}(function(e){var t=e.match(Q);return t?t[1].split(J):[]}(r),n)))}function Eo(e){var t=0,n=0;return function(){var r=cn(),i=16-(r-n);if(n=r,i>0){if(++t>=800)return arguments[0]}else t=0;return e.apply(void 0,arguments)}}function So(e,t){var n=-1,r=e.length,i=r-1;for(t=void 0===t?r:t;++n1?e[t-1]:void 0;return n="function"==typeof n?(e.pop(),n):void 0,Xo(e,n)}));function na(e){var t=On(e);return t.__chain__=!0,t}function ra(e,t){return t(e)}var ia=Hi((function(e){var t=e.length,n=t?e[0]:0,r=this.__wrapped__,i=function(t){return Xn(t,e)};return!(t>1||this.__actions__.length)&&r instanceof Dn&&so(n)?((r=r.slice(n,+n+(t?1:0))).__actions__.push({func:ra,args:[i],thisArg:void 0}),new Pn(r,this.__chain__).thru((function(e){return t&&!e.length&&e.push(void 0),e}))):this.thru(i)}));var oa=xi((function(e,t,n){Ae.call(e,n)?++e[n]:Yn(e,n,1)}));var aa=Ci(Do),sa=Ci(Ro);function la(e,t){return(Ra(e)?st:tr)(e,Qi(t,3))}function ca(e,t){return(Ra(e)?lt:nr)(e,Qi(t,3))}var ua=xi((function(e,t,n){Ae.call(e,n)?e[n].push(t):Yn(e,n,[t])}));var da=Lr((function(e,t,n){var i=-1,o="function"==typeof t,a=Na(e)?r(e.length):[];return tr(e,(function(e){a[++i]=o?ot(t,e,n):yr(e,t,n)})),a})),pa=xi((function(e,t,n){Yn(e,n,t)}));function fa(e,t){return(Ra(e)?ft:_r)(e,Qi(t,3))}var ha=xi((function(e,t,n){e[n?0:1].push(t)}),(function(){return[[],[]]}));var ga=Lr((function(e,t){if(null==e)return[];var n=t.length;return n>1&&lo(e,t[0],t[1])?t=[]:n>2&&lo(t[0],t[1],t[2])&&(t=[t[0]]),Dr(e,ar(t,1),[])})),ma=Qt||function(){return We.Date.now()};function va(e,t,n){return t=n?void 0:t,Bi(e,128,void 0,void 0,void 0,void 0,t=e&&null==t?e.length:t)}function ba(e,t){var n;if("function"!=typeof t)throw new ve(o);return e=is(e),function(){return--e>0&&(n=t.apply(this,arguments)),e<=1&&(t=void 0),n}}var ya=Lr((function(e,t,n){var r=1;if(n.length){var i=Vt(n,Zi(ya));r|=32}return Bi(e,r,t,n,i)})),xa=Lr((function(e,t,n){var r=3;if(n.length){var i=Vt(n,Zi(xa));r|=32}return Bi(t,r,e,n,i)}));function wa(e,t,n){var r,i,a,s,l,c,u=0,d=!1,p=!1,f=!0;if("function"!=typeof e)throw new ve(o);function h(t){var n=r,o=i;return r=i=void 0,u=t,s=e.apply(o,n)}function g(e){return u=e,l=wo(v,t),d?h(e):s}function m(e){var n=e-c;return void 0===c||n>=t||n<0||p&&e-u>=a}function v(){var e=ma();if(m(e))return b(e);l=wo(v,function(e){var n=t-(e-c);return p?ln(n,a-(e-u)):n}(e))}function b(e){return l=void 0,f&&r?h(e):(r=i=void 0,s)}function y(){var e=ma(),n=m(e);if(r=arguments,i=this,c=e,n){if(void 0===l)return g(c);if(p)return di(l),l=wo(v,t),h(c)}return void 0===l&&(l=wo(v,t)),s}return t=as(t)||0,Ua(n)&&(d=!!n.leading,a=(p="maxWait"in n)?sn(as(n.maxWait)||0,t):a,f="trailing"in n?!!n.trailing:f),y.cancel=function(){void 0!==l&&di(l),u=0,r=c=i=l=void 0},y.flush=function(){return void 0===l?s:b(ma())},y}var ka=Lr((function(e,t){return Kn(e,1,t)})),Aa=Lr((function(e,t,n){return Kn(e,as(t)||0,n)}));function Ea(e,t){if("function"!=typeof e||null!=t&&"function"!=typeof t)throw new ve(o);var n=function(){var r=arguments,i=t?t.apply(this,r):r[0],o=n.cache;if(o.has(i))return o.get(i);var a=e.apply(this,r);return n.cache=o.set(i,a)||o,a};return n.cache=new(Ea.Cache||Nn),n}function Sa(e){if("function"!=typeof e)throw new ve(o);return function(){var t=arguments;switch(t.length){case 0:return!e.call(this);case 1:return!e.call(this,t[0]);case 2:return!e.call(this,t[0],t[1]);case 3:return!e.call(this,t[0],t[1],t[2])}return!e.apply(this,t)}}Ea.Cache=Nn;var $a=ci((function(e,t){var n=(t=1==t.length&&Ra(t[0])?ft(t[0],jt(Qi())):ft(ar(t,1),jt(Qi()))).length;return Lr((function(r){for(var i=-1,o=ln(r.length,n);++i=t})),Da=xr(function(){return arguments}())?xr:function(e){return Ha(e)&&Ae.call(e,"callee")&&!Ge.call(e,"callee")},Ra=r.isArray,Ia=Ke?jt(Ke):function(e){return Ha(e)&&hr(e)==A};function Na(e){return null!=e&&Va(e.length)&&!Fa(e)}function Ma(e){return Ha(e)&&Na(e)}var za=nn||al,La=et?jt(et):function(e){return Ha(e)&&hr(e)==d};function Ba(e){if(!Ha(e))return!1;var t=hr(e);return t==p||"[object DOMException]"==t||"string"==typeof e.message&&"string"==typeof e.name&&!Ya(e)}function Fa(e){if(!Ua(e))return!1;var t=hr(e);return t==f||t==h||"[object AsyncFunction]"==t||"[object Proxy]"==t}function qa(e){return"number"==typeof e&&e==is(e)}function Va(e){return"number"==typeof e&&e>-1&&e%1==0&&e<=9007199254740991}function Ua(e){var t=typeof e;return null!=e&&("object"==t||"function"==t)}function Ha(e){return null!=e&&"object"==typeof e}var Ga=tt?jt(tt):function(e){return Ha(e)&&ro(e)==g};function Wa(e){return"number"==typeof e||Ha(e)&&hr(e)==m}function Ya(e){if(!Ha(e)||hr(e)!=v)return!1;var t=qe(e);if(null===t)return!0;var n=Ae.call(t,"constructor")&&t.constructor;return"function"==typeof n&&n instanceof n&&ke.call(n)==Ce}var Xa=nt?jt(nt):function(e){return Ha(e)&&hr(e)==b};var Za=rt?jt(rt):function(e){return Ha(e)&&ro(e)==y};function Qa(e){return"string"==typeof e||!Ra(e)&&Ha(e)&&hr(e)==x}function Ja(e){return"symbol"==typeof e||Ha(e)&&hr(e)==w}var Ka=it?jt(it):function(e){return Ha(e)&&Va(e.length)&&!!Be[hr(e)]};var es=Ii(Cr),ts=Ii((function(e,t){return e<=t}));function ns(e){if(!e)return[];if(Na(e))return Qa(e)?Wt(e):bi(e);if(Qe&&e[Qe])return function(e){for(var t,n=[];!(t=e.next()).done;)n.push(t.value);return n}(e[Qe]());var t=ro(e);return(t==g?Ft:t==y?Ut:Ts)(e)}function rs(e){return e?(e=as(e))===1/0||e===-1/0?17976931348623157e292*(e<0?-1:1):e==e?e:0:0===e?e:0}function is(e){var t=rs(e),n=t%1;return t==t?n?t-n:t:0}function os(e){return e?Zn(is(e),0,4294967295):0}function as(e){if("number"==typeof e)return e;if(Ja(e))return NaN;if(Ua(e)){var t="function"==typeof e.valueOf?e.valueOf():e;e=Ua(t)?t+"":t}if("string"!=typeof e)return 0===e?e:+e;e=Tt(e);var n=oe.test(e);return n||se.test(e)?Ue(e.slice(2),n?2:8):ie.test(e)?NaN:+e}function ss(e){return yi(e,ks(e))}function ls(e){return null==e?"":Jr(e)}var cs=wi((function(e,t){if(fo(t)||Na(t))yi(t,ws(t),e);else for(var n in t)Ae.call(t,n)&&Un(e,n,t[n])})),us=wi((function(e,t){yi(t,ks(t),e)})),ds=wi((function(e,t,n,r){yi(t,ks(t),e,r)})),ps=wi((function(e,t,n,r){yi(t,ws(t),e,r)})),fs=Hi(Xn);var hs=Lr((function(e,t){e=he(e);var n=-1,r=t.length,i=r>2?t[2]:void 0;for(i&&lo(t[0],t[1],i)&&(r=1);++n1),t})),yi(e,Wi(e),n),r&&(n=Qn(n,7,Vi));for(var i=t.length;i--;)ei(n,t[i]);return n}));var $s=Hi((function(e,t){return null==e?{}:function(e,t){return Rr(e,t,(function(t,n){return vs(e,n)}))}(e,t)}));function Cs(e,t){if(null==e)return{};var n=ft(Wi(e),(function(e){return[e]}));return t=Qi(t),Rr(e,n,(function(e,n){return t(e,n[0])}))}var _s=Li(ws),Os=Li(ks);function Ts(e){return null==e?[]:Pt(e,ws(e))}var js=Si((function(e,t,n){return t=t.toLowerCase(),e+(n?Ps(t):t)}));function Ps(e){return Bs(ls(e).toLowerCase())}function Ds(e){return(e=ls(e))&&e.replace(ce,Mt).replace(De,"")}var Rs=Si((function(e,t,n){return e+(n?"-":"")+t.toLowerCase()})),Is=Si((function(e,t,n){return e+(n?" ":"")+t.toLowerCase()})),Ns=Ei("toLowerCase");var Ms=Si((function(e,t,n){return e+(n?"_":"")+t.toLowerCase()}));var zs=Si((function(e,t,n){return e+(n?" ":"")+Bs(t)}));var Ls=Si((function(e,t,n){return e+(n?" ":"")+t.toUpperCase()})),Bs=Ei("toUpperCase");function Fs(e,t,n){return e=ls(e),void 0===(t=n?void 0:t)?function(e){return Me.test(e)}(e)?function(e){return e.match(Ie)||[]}(e):function(e){return e.match(K)||[]}(e):e.match(t)||[]}var qs=Lr((function(e,t){try{return ot(e,void 0,t)}catch(e){return Ba(e)?e:new X(e)}})),Vs=Hi((function(e,t){return st(t,(function(t){t=Co(t),Yn(e,t,ya(e[t],e))})),e}));function Us(e){return function(){return e}}var Hs=_i(),Gs=_i(!0);function Ws(e){return e}function Ys(e){return Er("function"==typeof e?e:Qn(e,1))}var Xs=Lr((function(e,t){return function(n){return yr(n,e,t)}})),Zs=Lr((function(e,t){return function(n){return yr(e,n,t)}}));function Qs(e,t,n){var r=ws(t),i=dr(t,r);null!=n||Ua(t)&&(i.length||!r.length)||(n=t,t=e,e=this,i=dr(t,ws(t)));var o=!(Ua(n)&&"chain"in n&&!n.chain),a=Fa(e);return st(i,(function(n){var r=t[n];e[n]=r,a&&(e.prototype[n]=function(){var t=this.__chain__;if(o||t){var n=e(this.__wrapped__),i=n.__actions__=bi(this.__actions__);return i.push({func:r,args:arguments,thisArg:e}),n.__chain__=t,n}return r.apply(e,ht([this.value()],arguments))})})),e}function Js(){}var Ks=Pi(ft),el=Pi(ct),tl=Pi(vt);function nl(e){return co(e)?St(Co(e)):function(e){return function(t){return pr(t,e)}}(e)}var rl=Ri(),il=Ri(!0);function ol(){return[]}function al(){return!1}var sl=ji((function(e,t){return e+t}),0),ll=Mi("ceil"),cl=ji((function(e,t){return e/t}),1),ul=Mi("floor");var dl,pl=ji((function(e,t){return e*t}),1),fl=Mi("round"),hl=ji((function(e,t){return e-t}),0);return On.after=function(e,t){if("function"!=typeof t)throw new ve(o);return e=is(e),function(){if(--e<1)return t.apply(this,arguments)}},On.ary=va,On.assign=cs,On.assignIn=us,On.assignInWith=ds,On.assignWith=ps,On.at=fs,On.before=ba,On.bind=ya,On.bindAll=Vs,On.bindKey=xa,On.castArray=function(){if(!arguments.length)return[];var e=arguments[0];return Ra(e)?e:[e]},On.chain=na,On.chunk=function(e,t,n){t=(n?lo(e,t,n):void 0===t)?1:sn(is(t),0);var i=null==e?0:e.length;if(!i||t<1)return[];for(var o=0,a=0,s=r(Kt(i/t));oi?0:i+n),(r=void 0===r||r>i?i:is(r))<0&&(r+=i),r=n>r?0:os(r);n>>0)?(e=ls(e))&&("string"==typeof t||null!=t&&!Xa(t))&&!(t=Jr(t))&&Bt(e)?ui(Wt(e),0,n):e.split(t,n):[]},On.spread=function(e,t){if("function"!=typeof e)throw new ve(o);return t=null==t?0:sn(is(t),0),Lr((function(n){var r=n[t],i=ui(n,0,t);return r&&ht(i,r),ot(e,this,i)}))},On.tail=function(e){var t=null==e?0:e.length;return t?Gr(e,1,t):[]},On.take=function(e,t,n){return e&&e.length?Gr(e,0,(t=n||void 0===t?1:is(t))<0?0:t):[]},On.takeRight=function(e,t,n){var r=null==e?0:e.length;return r?Gr(e,(t=r-(t=n||void 0===t?1:is(t)))<0?0:t,r):[]},On.takeRightWhile=function(e,t){return e&&e.length?ni(e,Qi(t,3),!1,!0):[]},On.takeWhile=function(e,t){return e&&e.length?ni(e,Qi(t,3)):[]},On.tap=function(e,t){return t(e),e},On.throttle=function(e,t,n){var r=!0,i=!0;if("function"!=typeof e)throw new ve(o);return Ua(n)&&(r="leading"in n?!!n.leading:r,i="trailing"in n?!!n.trailing:i),wa(e,t,{leading:r,maxWait:t,trailing:i})},On.thru=ra,On.toArray=ns,On.toPairs=_s,On.toPairsIn=Os,On.toPath=function(e){return Ra(e)?ft(e,Co):Ja(e)?[e]:bi($o(ls(e)))},On.toPlainObject=ss,On.transform=function(e,t,n){var r=Ra(e),i=r||za(e)||Ka(e);if(t=Qi(t,4),null==n){var o=e&&e.constructor;n=i?r?new o:[]:Ua(e)&&Fa(o)?Tn(qe(e)):{}}return(i?st:cr)(e,(function(e,r,i){return t(n,e,r,i)})),n},On.unary=function(e){return va(e,1)},On.union=Ho,On.unionBy=Go,On.unionWith=Wo,On.uniq=function(e){return e&&e.length?Kr(e):[]},On.uniqBy=function(e,t){return e&&e.length?Kr(e,Qi(t,2)):[]},On.uniqWith=function(e,t){return t="function"==typeof t?t:void 0,e&&e.length?Kr(e,void 0,t):[]},On.unset=function(e,t){return null==e||ei(e,t)},On.unzip=Yo,On.unzipWith=Xo,On.update=function(e,t,n){return null==e?e:ti(e,t,si(n))},On.updateWith=function(e,t,n,r){return r="function"==typeof r?r:void 0,null==e?e:ti(e,t,si(n),r)},On.values=Ts,On.valuesIn=function(e){return null==e?[]:Pt(e,ks(e))},On.without=Zo,On.words=Fs,On.wrap=function(e,t){return Ca(si(t),e)},On.xor=Qo,On.xorBy=Jo,On.xorWith=Ko,On.zip=ea,On.zipObject=function(e,t){return oi(e||[],t||[],Un)},On.zipObjectDeep=function(e,t){return oi(e||[],t||[],qr)},On.zipWith=ta,On.entries=_s,On.entriesIn=Os,On.extend=us,On.extendWith=ds,Qs(On,On),On.add=sl,On.attempt=qs,On.camelCase=js,On.capitalize=Ps,On.ceil=ll,On.clamp=function(e,t,n){return void 0===n&&(n=t,t=void 0),void 0!==n&&(n=(n=as(n))==n?n:0),void 0!==t&&(t=(t=as(t))==t?t:0),Zn(as(e),t,n)},On.clone=function(e){return Qn(e,4)},On.cloneDeep=function(e){return Qn(e,5)},On.cloneDeepWith=function(e,t){return Qn(e,5,t="function"==typeof t?t:void 0)},On.cloneWith=function(e,t){return Qn(e,4,t="function"==typeof t?t:void 0)},On.conformsTo=function(e,t){return null==t||Jn(e,t,ws(t))},On.deburr=Ds,On.defaultTo=function(e,t){return null==e||e!=e?t:e},On.divide=cl,On.endsWith=function(e,t,n){e=ls(e),t=Jr(t);var r=e.length,i=n=void 0===n?r:Zn(is(n),0,r);return(n-=t.length)>=0&&e.slice(n,i)==t},On.eq=Ta,On.escape=function(e){return(e=ls(e))&&L.test(e)?e.replace(M,zt):e},On.escapeRegExp=function(e){return(e=ls(e))&&W.test(e)?e.replace(G,"\\$&"):e},On.every=function(e,t,n){var r=Ra(e)?ct:rr;return n&&lo(e,t,n)&&(t=void 0),r(e,Qi(t,3))},On.find=aa,On.findIndex=Do,On.findKey=function(e,t){return yt(e,Qi(t,3),cr)},On.findLast=sa,On.findLastIndex=Ro,On.findLastKey=function(e,t){return yt(e,Qi(t,3),ur)},On.floor=ul,On.forEach=la,On.forEachRight=ca,On.forIn=function(e,t){return null==e?e:sr(e,Qi(t,3),ks)},On.forInRight=function(e,t){return null==e?e:lr(e,Qi(t,3),ks)},On.forOwn=function(e,t){return e&&cr(e,Qi(t,3))},On.forOwnRight=function(e,t){return e&&ur(e,Qi(t,3))},On.get=ms,On.gt=ja,On.gte=Pa,On.has=function(e,t){return null!=e&&io(e,t,mr)},On.hasIn=vs,On.head=No,On.identity=Ws,On.includes=function(e,t,n,r){e=Na(e)?e:Ts(e),n=n&&!r?is(n):0;var i=e.length;return n<0&&(n=sn(i+n,0)),Qa(e)?n<=i&&e.indexOf(t,n)>-1:!!i&&wt(e,t,n)>-1},On.indexOf=function(e,t,n){var r=null==e?0:e.length;if(!r)return-1;var i=null==n?0:is(n);return i<0&&(i=sn(r+i,0)),wt(e,t,i)},On.inRange=function(e,t,n){return t=rs(t),void 0===n?(n=t,t=0):n=rs(n),function(e,t,n){return e>=ln(t,n)&&e=-9007199254740991&&e<=9007199254740991},On.isSet=Za,On.isString=Qa,On.isSymbol=Ja,On.isTypedArray=Ka,On.isUndefined=function(e){return void 0===e},On.isWeakMap=function(e){return Ha(e)&&ro(e)==k},On.isWeakSet=function(e){return Ha(e)&&"[object WeakSet]"==hr(e)},On.join=function(e,t){return null==e?"":on.call(e,t)},On.kebabCase=Rs,On.last=Bo,On.lastIndexOf=function(e,t,n){var r=null==e?0:e.length;if(!r)return-1;var i=r;return void 0!==n&&(i=(i=is(n))<0?sn(r+i,0):ln(i,r-1)),t==t?function(e,t,n){for(var r=n+1;r--;)if(e[r]===t)return r;return r}(e,t,i):xt(e,At,i,!0)},On.lowerCase=Is,On.lowerFirst=Ns,On.lt=es,On.lte=ts,On.max=function(e){return e&&e.length?ir(e,Ws,gr):void 0},On.maxBy=function(e,t){return e&&e.length?ir(e,Qi(t,2),gr):void 0},On.mean=function(e){return Et(e,Ws)},On.meanBy=function(e,t){return Et(e,Qi(t,2))},On.min=function(e){return e&&e.length?ir(e,Ws,Cr):void 0},On.minBy=function(e,t){return e&&e.length?ir(e,Qi(t,2),Cr):void 0},On.stubArray=ol,On.stubFalse=al,On.stubObject=function(){return{}},On.stubString=function(){return""},On.stubTrue=function(){return!0},On.multiply=pl,On.nth=function(e,t){return e&&e.length?Pr(e,is(t)):void 0},On.noConflict=function(){return We._===this&&(We._=_e),this},On.noop=Js,On.now=ma,On.pad=function(e,t,n){e=ls(e);var r=(t=is(t))?Gt(e):0;if(!t||r>=t)return e;var i=(t-r)/2;return Di(en(i),n)+e+Di(Kt(i),n)},On.padEnd=function(e,t,n){e=ls(e);var r=(t=is(t))?Gt(e):0;return t&&rt){var r=e;e=t,t=r}if(n||e%1||t%1){var i=dn();return ln(e+i*(t-e+Ve("1e-"+((i+"").length-1))),t)}return Mr(e,t)},On.reduce=function(e,t,n){var r=Ra(e)?gt:Ct,i=arguments.length<3;return r(e,Qi(t,4),n,i,tr)},On.reduceRight=function(e,t,n){var r=Ra(e)?mt:Ct,i=arguments.length<3;return r(e,Qi(t,4),n,i,nr)},On.repeat=function(e,t,n){return t=(n?lo(e,t,n):void 0===t)?1:is(t),zr(ls(e),t)},On.replace=function(){var e=arguments,t=ls(e[0]);return e.length<3?t:t.replace(e[1],e[2])},On.result=function(e,t,n){var r=-1,i=(t=li(t,e)).length;for(i||(i=1,e=void 0);++r9007199254740991)return[];var n=4294967295,r=ln(e,4294967295);e-=4294967295;for(var i=Ot(r,t=Qi(t));++n=o)return e;var s=n-Gt(r);if(s<1)return r;var l=a?ui(a,0,s).join(""):e.slice(0,s);if(void 0===i)return l+r;if(a&&(s+=l.length-s),Xa(i)){if(e.slice(s).search(i)){var c,u=l;for(i.global||(i=ge(i.source,ls(re.exec(i))+"g")),i.lastIndex=0;c=i.exec(u);)var d=c.index;l=l.slice(0,void 0===d?s:d)}}else if(e.indexOf(Jr(i),s)!=s){var p=l.lastIndexOf(i);p>-1&&(l=l.slice(0,p))}return l+r},On.unescape=function(e){return(e=ls(e))&&z.test(e)?e.replace(N,Xt):e},On.uniqueId=function(e){var t=++Ee;return ls(e)+t},On.upperCase=Ls,On.upperFirst=Bs,On.each=la,On.eachRight=ca,On.first=No,Qs(On,(dl={},cr(On,(function(e,t){Ae.call(On.prototype,t)||(dl[t]=e)})),dl),{chain:!1}),On.VERSION="4.17.21",st(["bind","bindKey","curry","curryRight","partial","partialRight"],(function(e){On[e].placeholder=On})),st(["drop","take"],(function(e,t){Dn.prototype[e]=function(n){n=void 0===n?1:sn(is(n),0);var r=this.__filtered__&&!t?new Dn(this):this.clone();return r.__filtered__?r.__takeCount__=ln(n,r.__takeCount__):r.__views__.push({size:ln(n,4294967295),type:e+(r.__dir__<0?"Right":"")}),r},Dn.prototype[e+"Right"]=function(t){return this.reverse()[e](t).reverse()}})),st(["filter","map","takeWhile"],(function(e,t){var n=t+1,r=1==n||3==n;Dn.prototype[e]=function(e){var t=this.clone();return t.__iteratees__.push({iteratee:Qi(e,3),type:n}),t.__filtered__=t.__filtered__||r,t}})),st(["head","last"],(function(e,t){var n="take"+(t?"Right":"");Dn.prototype[e]=function(){return this[n](1).value()[0]}})),st(["initial","tail"],(function(e,t){var n="drop"+(t?"":"Right");Dn.prototype[e]=function(){return this.__filtered__?new Dn(this):this[n](1)}})),Dn.prototype.compact=function(){return this.filter(Ws)},Dn.prototype.find=function(e){return this.filter(e).head()},Dn.prototype.findLast=function(e){return this.reverse().find(e)},Dn.prototype.invokeMap=Lr((function(e,t){return"function"==typeof e?new Dn(this):this.map((function(n){return yr(n,e,t)}))})),Dn.prototype.reject=function(e){return this.filter(Sa(Qi(e)))},Dn.prototype.slice=function(e,t){e=is(e);var n=this;return n.__filtered__&&(e>0||t<0)?new Dn(n):(e<0?n=n.takeRight(-e):e&&(n=n.drop(e)),void 0!==t&&(n=(t=is(t))<0?n.dropRight(-t):n.take(t-e)),n)},Dn.prototype.takeRightWhile=function(e){return this.reverse().takeWhile(e).reverse()},Dn.prototype.toArray=function(){return this.take(4294967295)},cr(Dn.prototype,(function(e,t){var n=/^(?:filter|find|map|reject)|While$/.test(t),r=/^(?:head|last)$/.test(t),i=On[r?"take"+("last"==t?"Right":""):t],o=r||/^find/.test(t);i&&(On.prototype[t]=function(){var t=this.__wrapped__,a=r?[1]:arguments,s=t instanceof Dn,l=a[0],c=s||Ra(t),u=function(e){var t=i.apply(On,ht([e],a));return r&&d?t[0]:t};c&&n&&"function"==typeof l&&1!=l.length&&(s=c=!1);var d=this.__chain__,p=!!this.__actions__.length,f=o&&!d,h=s&&!p;if(!o&&c){t=h?t:new Dn(this);var g=e.apply(t,a);return g.__actions__.push({func:ra,args:[u],thisArg:void 0}),new Pn(g,d)}return f&&h?e.apply(this,a):(g=this.thru(u),f?r?g.value()[0]:g.value():g)})})),st(["pop","push","shift","sort","splice","unshift"],(function(e){var t=be[e],n=/^(?:push|sort|unshift)$/.test(e)?"tap":"thru",r=/^(?:pop|shift)$/.test(e);On.prototype[e]=function(){var e=arguments;if(r&&!this.__chain__){var i=this.value();return t.apply(Ra(i)?i:[],e)}return this[n]((function(n){return t.apply(Ra(n)?n:[],e)}))}})),cr(Dn.prototype,(function(e,t){var n=On[t];if(n){var r=n.name+"";Ae.call(xn,r)||(xn[r]=[]),xn[r].push({name:t,func:n})}})),xn[Oi(void 0,2).name]=[{name:"wrapper",func:void 0}],Dn.prototype.clone=function(){var e=new Dn(this.__wrapped__);return e.__actions__=bi(this.__actions__),e.__dir__=this.__dir__,e.__filtered__=this.__filtered__,e.__iteratees__=bi(this.__iteratees__),e.__takeCount__=this.__takeCount__,e.__views__=bi(this.__views__),e},Dn.prototype.reverse=function(){if(this.__filtered__){var e=new Dn(this);e.__dir__=-1,e.__filtered__=!0}else(e=this.clone()).__dir__*=-1;return e},Dn.prototype.value=function(){var e=this.__wrapped__.value(),t=this.__dir__,n=Ra(e),r=t<0,i=n?e.length:0,o=function(e,t,n){var r=-1,i=n.length;for(;++r=this.__values__.length;return{done:e,value:e?void 0:this.__values__[this.__index__++]}},On.prototype.plant=function(e){for(var t,n=this;n instanceof jn;){var r=Oo(n);r.__index__=0,r.__values__=void 0,t?i.__wrapped__=r:t=r;var i=r;n=n.__wrapped__}return i.__wrapped__=e,t},On.prototype.reverse=function(){var e=this.__wrapped__;if(e instanceof Dn){var t=e;return this.__actions__.length&&(t=new Dn(this)),(t=t.reverse()).__actions__.push({func:ra,args:[Uo],thisArg:void 0}),new Pn(t,this.__chain__)}return this.thru(Uo)},On.prototype.toJSON=On.prototype.valueOf=On.prototype.value=function(){return ri(this.__wrapped__,this.__actions__)},On.prototype.first=On.prototype.head,Qe&&(On.prototype[Qe]=function(){return this}),On}();We._=Zt,void 0===(i=function(){return Zt}.call(t,n,t,r))||(r.exports=i)}).call(this)}).call(this,n(35),n(80)(e))},function(e,t,n){const r=n(21);function i(e){return!r.isNull(e)&&!r.isUndefined(e)}function o(e,t,n,a){a||(a=1);var s=e.predecessors(t);if(!s||0==n)return[];var l=s.concat(s.reduce((function(t,r){return a>=n&&i(n)?t:t.concat(o(e,r,n,a+1))}),[]));return r.uniq(l)}function a(e,t,n,o){o||(o=1);var s=e.successors(t);if(!s||0==n)return[];var l=s.concat(s.reduce((function(t,r){return o>=n&&i(n)?t:t.concat(a(e,r,n,o+1))}),[]));return r.uniq(l)}e.exports={selectAt:function(e,t){var n=[t],i=r.union([t],a(e,t));return r.each(i,(function(t){var i=o(e,t);n=r.union(n,i,[t])})),n},ancestorNodes:o,descendentNodes:a}},function(e,t){var n={utf8:{stringToBytes:function(e){return n.bin.stringToBytes(unescape(encodeURIComponent(e)))},bytesToString:function(e){return decodeURIComponent(escape(n.bin.bytesToString(e)))}},bin:{stringToBytes:function(e){for(var t=[],n=0;n\n/* TODO */\n.section-target {\n top: -8em;\n}\n\n.noflex {\n flex: 0 0 160px !important;\n}\n\n.highlight {\n color: #24292e;\n background-color: white;\n}\n\n\n\n
\n \n
\n
\n
\n
\n \n
\n\n
\n
\n
\n
Description
\n
\n
\n
\n
This {{ model.resource_type }} is not currently documented
\n
\n
\n
\n
\n
\n
\n
\n
Columns
\n \n
\n
\n\n
\n
\n
\n
Referenced By
\n \n
\n
\n\n
\n
\n
\n \n
\n
\n
\n
\n
\n')}]),e.exports=n},function(e,t,n){"use strict";n.r(t);var r=function(e){return function(e){return!!e&&"object"==typeof e}(e)&&!function(e){var t=Object.prototype.toString.call(e);return"[object RegExp]"===t||"[object Date]"===t||function(e){return e.$$typeof===i}(e)}(e)};var i="function"==typeof Symbol&&Symbol.for?Symbol.for("react.element"):60103;function o(e,t){return!1!==t.clone&&t.isMergeableObject(e)?s((n=e,Array.isArray(n)?[]:{}),e,t):e;var n}function a(e,t,n){return e.concat(t).map((function(e){return o(e,n)}))}function s(e,t,n){(n=n||{}).arrayMerge=n.arrayMerge||a,n.isMergeableObject=n.isMergeableObject||r;var i=Array.isArray(t);return i===Array.isArray(e)?i?n.arrayMerge(e,t,n):function(e,t,n){var r={};return n.isMergeableObject(e)&&Object.keys(e).forEach((function(t){r[t]=o(e[t],n)})),Object.keys(t).forEach((function(i){n.isMergeableObject(t[i])&&e[i]?r[i]=s(e[i],t[i],n):r[i]=o(t[i],n)})),r}(e,t,n):o(t,n)}s.all=function(e,t){if(!Array.isArray(e))throw new Error("first argument should be an array");return e.reduce((function(e,n){return s(e,n,t)}),{})};var l=s;const c=n(8),u=(n(33),n(202)),{getQuoteChar:d}=n(430);c.module("dbt").factory("project",["$q","$http",function(e,t){var n={manifest:"MANIFEST.JSON INLINE DATA",catalog:"CATALOG.JSON INLINE DATA"},r={project:{},tree:{project:[],database:[],sources:[]},files:{manifest:{},catalog:{}},loaded:e.defer()};function i(e,t){return u.each(t.sources,(function(e,n){t.nodes[n]=e})),u.each(e.nodes,(function(e,n){var r=t.nodes[n];if(r){var i,o,a,s=u.keys(r.columns),l=e.columns,c=(i=s,o=l,a={},u.each(o,(function(e,t){var n=u.find(i,(function(e){return e.toLowerCase()==t.toLowerCase()}));n?a[n]=e:a[t]=e})),a);e.columns=c}})),l(t,e)}function o(e,r){return e in n&&"object"==typeof n[e]?{label:e,data:n[e]}:t({method:"GET",url:r}).then((function(t){return{label:e,data:t.data}}),(function(t){console.error(t),alert("dbt Docs was unable to load the "+e+" file at path: \n "+r+"\n\nError: "+t.statusText+" ("+t.status+")\n\nThe dbt Docs site may not work as expected if this file cannot be found.Please try again, and contact support if this error persists.")}))}return r.find_by_id=function(e,t){r.ready((function(){if(e){var n=r.node(e);t(n)}}))},r.node=function(e){return u.find(r.project.nodes,{unique_id:e})},r.loadProject=function(){var t="?cb="+(new Date).getTime(),n=[o("manifest","manifest.json"+t),o("catalog","catalog.json"+t)];e.all(n).then((function(e){u.each(e,(function(e){e?r.files[e.label]=e.data:console.error("FILE FAILED TO LOAD!")})),u.each(r.files.manifest.nodes,(function(e){"model"==e.resource_type&&null!=e.version?e.label=e.name+"_v"+e.version:e.label=e.name})),u.each(r.files.manifest.sources,(function(e){e.label=e.source_name+"."+e.name,r.files.manifest.nodes[e.unique_id]=e})),u.each(r.files.manifest.exposures,(function(e){e.label||(e.label=e.name),r.files.manifest.nodes[e.unique_id]=e})),u.each(r.files.manifest.metrics,(function(e){r.files.manifest.nodes[e.unique_id]=e})),u.each(r.files.manifest.semantic_models,(function(e){r.files.manifest.nodes[e.unique_id]=e,e.label=e.name})),u.each(r.files.manifest.saved_queries,(function(e){r.files.manifest.nodes[e.unique_id]=e,e.label=e.name})),u.each(r.files.manifest.unit_tests,(function(e){r.files.manifest.nodes[e.unique_id]=e,e.label=e.name}));var t=r.files.manifest.metadata.adapter_type,n=function(e,t){var n=e||[],r={};u.each(n,(function(e){r[e.package_name]||(r[e.package_name]={}),r[e.package_name][e.name]=e}));e=[];return u.each(r,(function(n,r){if("dbt"!=r&&r!="dbt_"+t){var i=function(e,t){var n={};u.each(e,(function(e){e.macro_sql.match(/{{\s*adapter_macro\([^)]+\)\s+}}/)&&(e.impls={"Adapter Macro":e.macro_sql},e.is_adapter_macro=!0,n[e.name]=e)}));var r=["postgres","redshift","bigquery","snowflake","spark","presto","default"],i=u.values(n),o=u.filter(e,(function(e){var t=e.name.split("__"),i=t.shift(),o=t.join("__");return!(r.indexOf(i)>=0&&n[o])||(n[o].impls[i]=e.macro_sql,e.is_adapter_macro_impl=!0,!1)}));return i.concat(o)}(n);e=e.concat(i)}})),u.keyBy(e,"unique_id")}(r.files.manifest.macros,t);r.files.manifest.macros=n;var o=i(r.files.manifest,r.files.catalog),a=o.nodes,s=u.keyBy(a,"name"),l=u.filter(o.nodes,{resource_type:"test"});u.each(l,(function(e){if(e.hasOwnProperty("test_metadata")){var t,n={test_name:t=e.test_metadata.namespace?e.test_metadata.namespace+"."+e.test_metadata.name:e.test_metadata.name};if("not_null"==e.test_metadata.name)n.short="N",n.label="Not Null";else if("unique"==e.test_metadata.name)n.short="U",n.label="Unique";else if("relationships"==e.test_metadata.name){var r=e.refs[0],i=s[r];i&&e.test_metadata.kwargs.field&&(n.fk_field=e.test_metadata.kwargs.field,n.fk_model=i),n.short="F",n.label="Foreign Key"}else if("accepted_values"==e.test_metadata.name){if(Array.isArray(e.test_metadata.kwargs.values))var a=e.test_metadata.kwargs.values.join(", ");else a=JSON.stringify(e.test_metadata.kwargs.values);n.short="A",n.label="Accepted Values: "+a}else{var l=u.omit(e.test_metadata.kwargs,"column_name");n.short="+",n.label=t+"("+JSON.stringify(l)+")"}var c=e.depends_on.nodes,p=e.column_name||e.test_metadata.kwargs.column_name||e.test_metadata.kwargs.arg;if(c.length&&p){if("relationships"==e.test_metadata.name)var f=c[c.length-1];else f=c[0];var h=o.nodes[f],g=d(o.metadata),m=u.find(h.columns,(function(e,t){let n=p;return p.startsWith(g)&&p.endsWith(g)&&(n=p.substring(1,p.length-1)),t.toLowerCase()==n.toLowerCase()}));m&&(m.tests=m.tests||[],m.tests.push(n))}}})),r.project=o;var c=u.filter(r.project.macros,(function(e){return!e.is_adapter_macro_impl})),p=u.filter(r.project.nodes,(function(e){return u.includes(["model","source","seed","snapshot","analysis","exposure","metric","semantic_model","saved_query"],e.resource_type)}));r.project.searchable=u.filter(p.concat(c),(function(e){return!e.docs||e.docs.show})),r.loaded.resolve()}))},r.ready=function(e){r.loaded.promise.then((function(){e(r.project)}))},r.search=function(e){if(0==e.length)return u.map(r.project.searchable,(function(e){return{model:e,matches:[]}}));var t=[];return u.each(r.project.searchable,(function(n){var r=function(e,t){var n=[],r={name:"string",description:"string",raw_code:"string",columns:"object",column_description:"n/a",tags:"array",arguments:"array",label:"string"};let i=u.words(e.toLowerCase());for(var o in r)if("column_description"===o)for(var a in t.columns)null!=t.columns[a].description&&i.every(e=>-1!=t.columns[a].description.toLowerCase().indexOf(e))&&n.push({key:o,value:e});else{if(!t[o])continue;if("string"===r[o]&&i.every(e=>-1!=t[o].toLowerCase().indexOf(e)))n.push({key:o,value:e});else if("object"===r[o])for(var a in t[o])null!=t[o][a].name&&i.every(e=>-1!=t[o][a].name.toLowerCase().indexOf(e))&&n.push({key:o,value:e});else if("array"===r[o])for(var s of t[o])i.every(e=>-1!=JSON.stringify(s).toLowerCase().indexOf(e))&&n.push({key:o,value:e})}return n}(e,n);r.length&&t.push({model:n,matches:r})})),t},r.getModelTree=function(e,t){r.loaded.promise.then((function(){var n=u.values(r.project.macros),i=u.filter(r.project.nodes,(function(e){if("test"==e.resource_type&&!e.hasOwnProperty("test_metadata"))return!0;return u.includes(["snapshot","source","seed","model","analysis","exposure","metric","semantic_model","saved_query"],e.resource_type)}));r.tree.database=function(e,t){var n={},r=u.filter(e,(function(e){return!!u.get(e,["docs","show"],!0)&&(-1!=u.indexOf(["source","snapshot","seed"],e.resource_type)||("model"==e.resource_type?"ephemeral"!=e.config.materialized:void 0))})),i=u.sortBy(r,(function(e){return e.database+"."+e.schema+"."+(e.identifier||e.alias||e.name)})),o=u.groupBy(i,"database");return u.each(o,(function(e,r){var i={type:"database",name:r,active:!1,items:[]};n[r]=i;var o=u.groupBy(e,"schema");u.each(o,(function(e,n){n={type:"schema",name:n,active:!1,items:[]};i.items.push(n),u.each(e,(function(e){var r=e.unique_id==t;r&&(i.active=!0,n.active=!0),n.items.push({type:"table",name:e.identifier||e.alias||e.name,node:e,active:r,unique_id:e.unique_id,node_type:"model"})}))}))})),n}(i,e),r.tree.groups=function(e,t){var n={};u.each(e,(function(e){const r=u.get(e,["docs","show"],!0);if(!(e.resource_type in["source","exposure","seed","macro"])&&r&&"private"!==e.access){if("model"==e.resource_type&&null!=e.version)var i=e.name+"_v"+e.version;else i=e.name;var o="protected"===e.access?i+" (protected)":i,a=e.group,s=e.unique_id==t;n[a]?s&&(n[a].active=!0):n[a]={type:"group",name:a,active:s,items:[]},n[a].items.push({type:"file",name:o,node:e,active:s,unique_id:e.unique_id,node_type:"model"})}}));n=u.sortBy(u.values(n),"name");return u.each(n,(function(e){e.items=u.sortBy(e.items,"name")})),n}(i,e),r.tree.project=function(e,t,n){var r={};e=e||[],t=t||[];return u.each(e.concat(t),(function(e){var t=u.get(e,["docs","show"],!0);if("source"!=e.resource_type&&"exposure"!=e.resource_type&&"metric"!=e.resource_type&&"semantic_model"!=e.resource_type&&"saved_query"!=e.resource_type&&t){if(-1!=e.original_file_path.indexOf("\\"))var i=e.original_file_path.split("\\");else i=e.original_file_path.split("/");var o=[e.package_name].concat(i),a=e.unique_id==n,s=u.initial(o);if("macro"==e.resource_type)var l=e.name;else l=u.last(o);if("model"==e.resource_type&&null!=e.version)var c=e.name+"_v"+e.version;else c=e.name;var d=r;u.each(s,(function(e){d[e]?a&&(d[e].active=!0):d[e]={type:"folder",name:e,active:a,items:{}},d=d[e].items})),d[l]={type:"file",name:c,node:e,active:a,unique_id:e.unique_id,node_type:e.resource_type}}})),function e(t){var n=[],r=u.values(t);return u.each(r,(function(t){if(t.items){var r=e(t.items),i=u.sortBy(r,"name");t.items=i}n.push(t)})),n}(r)}(i,n,e);var o=u.values(r.project.sources);r.tree.sources=function(e,t){var n={};u.each(e,(function(e){var r=e.source_name,i=e.name,o=e.unique_id==t;n[r]?o&&(n[r].active=!0):n[r]={type:"folder",name:r,active:o,items:[]},n[r].items.push({type:"file",name:i,node:e,active:o,unique_id:e.unique_id,node_type:"source"})}));n=u.sortBy(u.values(n),"name");return u.each(n,(function(e){e.items=u.sortBy(e.items,"name")})),n}(o,e);var a=u.values(r.project.exposures);r.tree.exposures=function(e,t){var n={};u.each(e,(function(e){e.name;var r=e.type||"Uncategorized";r=function(e){var t={ml:"ML"};return t.hasOwnProperty(e)?t[e]:e.charAt(0).toUpperCase()+e.slice(1)}(r);var i=e.unique_id==t;n[r]?i&&(n[r].active=!0):n[r]={type:"folder",name:r,active:i,items:[]},n[r].items.push({type:"file",name:e.label,node:e,active:i,unique_id:e.unique_id,node_type:"exposure"})}));n=u.sortBy(u.values(n),"name");return u.each(n,(function(e){e.items=u.sortBy(e.items,"name")})),n}(a,e);var s=u.values(r.project.metrics);r.tree.metrics=function(e,t){var n={};u.each(e,(function(e){e.name;var r=e.package_name,i=e.unique_id==t;n[r]?i&&(n[r].active=!0):n[r]={type:"folder",name:r,active:i,items:[]},n[r].items.push({type:"file",name:e.label,node:e,active:i,unique_id:e.unique_id,node_type:"metric"})}));n=u.sortBy(u.values(n),"name");return u.each(n,(function(e){n.items=u.sortBy(n.items,"name")})),n}(s,e);var l=u.values(r.project.semantic_models);r.tree.semantic_models=function(e,t){var n={};u.each(e,(function(e){e.name;var r=e.package_name,i=e.unique_id==t;n[r]?i&&(n[r].active=!0):n[r]={type:"folder",name:r,active:i,items:[]},n[r].items.push({type:"file",name:e.name,node:e,active:i,unique_id:e.unique_id,node_type:"semantic_model"})}));n=u.sortBy(u.values(n),"name");return u.each(n,(function(e){n.items=u.sortBy(n.items,"name")})),n}(l,e);var c=u.values(r.project.saved_queries);r.tree.saved_queries=function(e,t){var n={};u.each(e,(function(e){e.name;var r=e.package_name,i=e.unique_id==t;n[r]?i&&(n[r].active=!0):n[r]={type:"folder",name:r,active:i,items:[]},n[r].items.push({type:"file",name:e.name,node:e,active:i,unique_id:e.unique_id,node_type:"saved_query"})}));n=u.sortBy(u.values(n),"name");return u.each(n,(function(e){n.items=u.sortBy(n.items,"name")})),n}(c,e),t(r.tree)}))},r.updateSelectedInTree=function(e,t){var n=!1;return u.each(t,(function(t){if(t.node&&t.node.unique_id==e)t.active=!0,n=!0;else if(t.node&&t.node.unique_id!=e)t.active=!1;else{r.updateSelectedInTree(e,t.items)&&(t.active=!0,n=!0)}})),n},r.updateSelected=function(e){return r.updateSelectedInTree(e,r.tree.project),r.updateSelectedInTree(e,r.tree.database),r.updateSelectedInTree(e,r.tree.groups),r.updateSelectedInTree(e,r.tree.sources),r.updateSelectedInTree(e,r.tree.exposures),r.updateSelectedInTree(e,r.tree.metrics),r.updateSelectedInTree(e,r.tree.semantic_models),r.updateSelectedInTree(e,r.tree.saved_queries),r.tree},r.caseColumn=function(e){return"snowflake"==r.project.metadata.adapter_type&&e.toUpperCase()==e?e.toLowerCase():e},r.init=function(){r.loadProject()},r}])},function(e,t,n){const r=n(8);n(209),n(230),n(445),n(459),n(460),n(480),n(481),n(482),r.module("dbt").run(["$rootScope","$state","$stateParams",function(e,t,n){e.$state=t,e.$stateParams=n}])},function(e,t){ -/** - * @license AngularJS v1.8.3 - * (c) 2010-2020 Google LLC. http://angularjs.org - * License: MIT - */ -!function(e){"use strict";var t={objectMaxDepth:5,urlErrorParamsEnabled:!0};function n(e){if(!L(e))return t;z(e.objectMaxDepth)&&(t.objectMaxDepth=r(e.objectMaxDepth)?e.objectMaxDepth:NaN),z(e.urlErrorParamsEnabled)&&Z(e.urlErrorParamsEnabled)&&(t.urlErrorParamsEnabled=e.urlErrorParamsEnabled)}function r(e){return q(e)&&e>0}function i(e,n){n=n||Error;var r="https://errors.angularjs.org/1.8.3/",i=r.replace(".","\\.")+"[\\s\\S]*",o=new RegExp(i,"g");return function(){var i,a,s=arguments[0],l=arguments[1],c="["+(e?e+":":"")+s+"] ",u=de(arguments,2).map((function(e){return Ve(e,t.objectMaxDepth)}));if(c+=l.replace(/\{\d+\}/g,(function(e){var t=+e.slice(1,-1);return t=0&&t-1 in e||"function"==typeof e.item)}function k(e,t,n){var r,i;if(e)if(G(e))for(r in e)"prototype"!==r&&"length"!==r&&"name"!==r&&e.hasOwnProperty(r)&&t.call(n,e[r],r,e);else if(U(e)||w(e)){var o="object"!=typeof e;for(r=0,i=e.length;r=0&&e.splice(n,1),n}function oe(e,t,n){var i,o,a=[],s=[];if(n=r(n)?n:NaN,t){if((o=t)&&q(o.length)&&J.test(m.call(o))||(i=t,"[object ArrayBuffer]"===m.call(i)))throw b("cpta","Can't copy! TypedArray destination cannot be mutated.");if(e===t)throw b("cpi","Can't copy! Source and destination are identical.");return U(t)?t.length=0:k(t,(function(e,n){"$$hashKey"!==n&&delete t[n]})),a.push(e),s.push(t),l(e,t,n)}return c(e,n);function l(e,t,n){if(--n<0)return"...";var r,i=t.$$hashKey;if(U(e))for(var o=0,a=e.length;o2?de(arguments,2):[];return!G(t)||t instanceof RegExp?t:n.length?function(){return arguments.length?t.apply(e,ue(n,arguments,0)):t.apply(e,n)}:function(){return arguments.length?t.apply(e,arguments):t.call(e)}}function fe(t,n){var r=n;return"string"==typeof t&&"$"===t.charAt(0)&&"$"===t.charAt(1)?r=void 0:Y(n)?r="$WINDOW":n&&e.document===n?r="$DOCUMENT":X(n)&&(r="$SCOPE"),r}function he(e,t){if(!M(e))return q(t)||(t=t?2:null),JSON.stringify(e,fe,t)}function ge(e){return F(e)?JSON.parse(e):e}var me=/:/g;function ve(e,t){e=e.replace(me,"");var n=Date.parse("Jan 01, 1970 00:00:00 "+e)/6e4;return j(n)?t:n}function be(e,t){return(e=new Date(e.getTime())).setMinutes(e.getMinutes()+t),e}function ye(e,t,n){n=n?-1:1;var r=e.getTimezoneOffset();return be(e,n*(ve(t,r)-r))}function xe(e){e=a(e).clone().empty();var t=a("
").append(e).html();try{return e[0].nodeType===Fe?d(t):t.match(/^(<[^>]+>)/)[1].replace(/^<([\w-]+)/,(function(e,t){return"<"+d(t)}))}catch(e){return d(t)}}function we(e){try{return decodeURIComponent(e)}catch(e){}}function ke(e){var t={};return k((e||"").split("&"),(function(e){var n,r,i;e&&(r=e=e.replace(/\+/g,"%20"),-1!==(n=e.indexOf("="))&&(r=e.substring(0,n),i=e.substring(n+1)),z(r=we(r))&&(i=!z(i)||we(i),u.call(t,r)?U(t[r])?t[r].push(i):t[r]=[t[r],i]:t[r]=i))})),t}function Ae(e){return Ee(e,!0).replace(/%26/gi,"&").replace(/%3D/gi,"=").replace(/%2B/gi,"+")}function Ee(e,t){return encodeURIComponent(e).replace(/%40/gi,"@").replace(/%3A/gi,":").replace(/%24/g,"$").replace(/%2C/gi,",").replace(/%3B/gi,";").replace(/%20/g,t?"%20":"+")}var Se=["ng-","data-ng-","ng:","x-ng-"];var $e=function(t){var n=t.currentScript;if(!n)return!0;if(!(n instanceof e.HTMLScriptElement||n instanceof e.SVGScriptElement))return!1;var r=n.attributes;return[r.getNamedItem("src"),r.getNamedItem("href"),r.getNamedItem("xlink:href")].every((function(e){if(!e)return!0;if(!e.value)return!1;var n=t.createElement("a");if(n.href=e.value,t.location.origin===n.origin)return!0;switch(n.protocol){case"http:":case"https:":case"ftp:":case"blob:":case"file:":case"data:":return!0;default:return!1}}))}(e.document);function Ce(t,n){var r,i,o={};if(k(Se,(function(e){var n=e+"app";!r&&t.hasAttribute&&t.hasAttribute(n)&&(r=t,i=t.getAttribute(n))})),k(Se,(function(e){var n,o=e+"app";!r&&(n=t.querySelector("["+o.replace(":","\\:")+"]"))&&(r=n,i=n.getAttribute(o))})),r){if(!$e)return void e.console.error("AngularJS: disabling automatic bootstrap. +*/var r = function (e, t) { this.recycle(e, t) }; function i() { return !1 } function o() { return !0 } r.prototype = { instanceString: function () { return "event" }, recycle: function (e, t) { if (this.isImmediatePropagationStopped = this.isPropagationStopped = this.isDefaultPrevented = i, null != e && e.preventDefault ? (this.type = e.type, this.isDefaultPrevented = e.defaultPrevented ? o : i) : null != e && e.type ? t = e : this.type = e, null != t && (this.originalEvent = t.originalEvent, this.type = null != t.type ? t.type : this.type, this.cy = t.cy, this.target = t.target, this.position = t.position, this.renderedPosition = t.renderedPosition, this.namespace = t.namespace, this.layout = t.layout), null != this.cy && null != this.position && null == this.renderedPosition) { var n = this.position, r = this.cy.zoom(), a = this.cy.pan(); this.renderedPosition = { x: n.x * r + a.x, y: n.y * r + a.y } } this.timeStamp = e && e.timeStamp || Date.now() }, preventDefault: function () { this.isDefaultPrevented = o; var e = this.originalEvent; e && e.preventDefault && e.preventDefault() }, stopPropagation: function () { this.isPropagationStopped = o; var e = this.originalEvent; e && e.stopPropagation && e.stopPropagation() }, stopImmediatePropagation: function () { this.isImmediatePropagationStopped = o, this.stopPropagation() }, isDefaultPrevented: i, isPropagationStopped: i, isImmediatePropagationStopped: i }, e.exports = r + }, function (e, t, n) { "use strict"; var r = n(1); e.exports = function (e, t) { var n = e.cy().hasCompoundNodes(); function i(e) { var t = e.pstyle("z-compound-depth"); return "auto" === t.value ? n ? e.zDepth() : 0 : "bottom" === t.value ? -1 : "top" === t.value ? r.MAX_INT : 0 } var o = i(e) - i(t); if (0 !== o) return o; function a(e) { return "auto" === e.pstyle("z-index-compare").value && e.isNode() ? 1 : 0 } var s = a(e) - a(t); if (0 !== s) return s; var l = e.pstyle("z-index").value - t.pstyle("z-index").value; return 0 !== l ? l : e.poolIndex() - t.poolIndex() } }, function (e, t, n) { "use strict"; var r = n(0), i = n(1), o = n(6), a = function e(t) { if (!(this instanceof e)) return new e(t); r.core(t) ? (this._private = { cy: t, coreStyle: {} }, this.length = 0, this.resetToDefault()) : i.error("A style must have a core reference") }, s = a.prototype; s.instanceString = function () { return "style" }, s.clear = function () { for (var e = 0; e < this.length; e++)this[e] = void 0; return this.length = 0, this._private.newStyle = !0, this }, s.resetToDefault = function () { return this.clear(), this.addDefaultStylesheet(), this }, s.core = function () { return this._private.coreStyle }, s.selector = function (e) { var t = "core" === e ? null : new o(e), n = this.length++; return this[n] = { selector: t, properties: [], mappedProperties: [], index: n }, this }, s.css = function () { var e = this, t = arguments; switch (t.length) { case 1: for (var n = t[0], r = 0; r < e.properties.length; r++) { var o = e.properties[r], a = n[o.name]; void 0 === a && (a = n[i.dash2camel(o.name)]), void 0 !== a && this.cssRule(o.name, a) } break; case 2: this.cssRule(t[0], t[1]) }return this }, s.style = s.css, s.cssRule = function (e, t) { var n = this.parse(e, t); if (n) { var r = this.length - 1; this[r].properties.push(n), this[r].properties[n.name] = n, n.name.match(/pie-(\d+)-background-size/) && n.value && (this._private.hasPie = !0), n.mapped && this[r].mappedProperties.push(n), !this[r].selector && (this._private.coreStyle[n.name] = n) } return this }, s.append = function (e) { return r.stylesheet(e) ? e.appendToStyle(this) : r.array(e) ? this.appendFromJson(e) : r.string(e) && this.appendFromString(e), this }, a.fromJson = function (e, t) { var n = new a(e); return n.fromJson(t), n }, a.fromString = function (e, t) { return new a(e).fromString(t) }, [n(85), n(86), n(87), n(88), n(89), n(90), n(91), n(92)].forEach((function (e) { i.extend(s, e) })), a.types = s.types, a.properties = s.properties, e.exports = a }, function (e, t, n) { "use strict"; var r = n(1); e.exports = { setupDequeueing: function (e) { return function () { var t = this, n = this.renderer; if (!t.dequeueingSetup) { t.dequeueingSetup = !0; var i = r.debounce((function () { n.redrawHint("eles", !0), n.redrawHint("drag", !0), n.redraw() }), e.deqRedrawThreshold), o = e.priority || r.noop; n.beforeRender((function (o, a) { for (var s = r.performanceNow(), l = n.averageRedrawTime, c = n.lastRedrawTime, u = [], d = n.cy.extent(), p = n.getPixelRatio(); ;) { var f = r.performanceNow(), h = f - s, g = f - a; if (c < 1e3 / 60) { var m = 1e3 / 60 - (o ? l : 0); if (g >= e.deqFastCost * m) break } else if (o) { if (h >= e.deqCost * c || h >= e.deqAvgCost * l) break } else if (g >= e.deqNoDrawCost * (1e3 / 60)) break; var v = e.deq(t, p, d); if (!(v.length > 0)) break; for (var b = 0; b < v.length; b++)u.push(v[b]) } u.length > 0 && (e.onDeqd(t, u), !o && e.shouldRedraw(t, u, p, d) && i()) }), o(t)) } } } } }, function (e, t, n) { "use strict"; var r = n(0), i = n(12), o = n(94), a = n(136), s = function (e) { return void 0 === e && (e = {}), r.plainObject(e) ? new i(e) : r.string(e) ? o.apply(o, arguments) : void 0 }; s.use = function (e) { var t = Array.prototype.slice.call(arguments, 1); return t.unshift(s), e.apply(null, t), this }, s.version = n(137), s.stylesheet = s.Stylesheet = a, e.exports = s }, function (e, t, n) { "use strict"; var r = n(0); e.exports = { hex2tuple: function (e) { if ((4 === e.length || 7 === e.length) && "#" === e[0]) { var t = void 0, n = void 0, r = void 0; return 4 === e.length ? (t = parseInt(e[1] + e[1], 16), n = parseInt(e[2] + e[2], 16), r = parseInt(e[3] + e[3], 16)) : (t = parseInt(e[1] + e[2], 16), n = parseInt(e[3] + e[4], 16), r = parseInt(e[5] + e[6], 16)), [t, n, r] } }, hsl2tuple: function (e) { var t = void 0, n = void 0, r = void 0, i = void 0, o = void 0, a = void 0, s = void 0, l = void 0; function c(e, t, n) { return n < 0 && (n += 1), n > 1 && (n -= 1), n < 1 / 6 ? e + 6 * (t - e) * n : n < .5 ? t : n < 2 / 3 ? e + (t - e) * (2 / 3 - n) * 6 : e } var u = new RegExp("^" + this.regex.hsla + "$").exec(e); if (u) { if ((n = parseInt(u[1])) < 0 ? n = (360 - -1 * n % 360) % 360 : n > 360 && (n %= 360), n /= 360, (r = parseFloat(u[2])) < 0 || r > 100) return; if (r /= 100, (i = parseFloat(u[3])) < 0 || i > 100) return; if (i /= 100, void 0 !== (o = u[4]) && ((o = parseFloat(o)) < 0 || o > 1)) return; if (0 === r) a = s = l = Math.round(255 * i); else { var d = i < .5 ? i * (1 + r) : i + r - i * r, p = 2 * i - d; a = Math.round(255 * c(p, d, n + 1 / 3)), s = Math.round(255 * c(p, d, n)), l = Math.round(255 * c(p, d, n - 1 / 3)) } t = [a, s, l, o] } return t }, rgb2tuple: function (e) { var t = void 0, n = new RegExp("^" + this.regex.rgba + "$").exec(e); if (n) { t = []; for (var r = [], i = 1; i <= 3; i++) { var o = n[i]; if ("%" === o[o.length - 1] && (r[i] = !0), o = parseFloat(o), r[i] && (o = o / 100 * 255), o < 0 || o > 255) return; t.push(Math.floor(o)) } var a = r[1] || r[2] || r[3], s = r[1] && r[2] && r[3]; if (a && !s) return; var l = n[4]; if (void 0 !== l) { if ((l = parseFloat(l)) < 0 || l > 1) return; t.push(l) } } return t }, colorname2tuple: function (e) { return this.colors[e.toLowerCase()] }, color2tuple: function (e) { return (r.array(e) ? e : null) || this.colorname2tuple(e) || this.hex2tuple(e) || this.rgb2tuple(e) || this.hsl2tuple(e) }, colors: { transparent: [0, 0, 0, 0], aliceblue: [240, 248, 255], antiquewhite: [250, 235, 215], aqua: [0, 255, 255], aquamarine: [127, 255, 212], azure: [240, 255, 255], beige: [245, 245, 220], bisque: [255, 228, 196], black: [0, 0, 0], blanchedalmond: [255, 235, 205], blue: [0, 0, 255], blueviolet: [138, 43, 226], brown: [165, 42, 42], burlywood: [222, 184, 135], cadetblue: [95, 158, 160], chartreuse: [127, 255, 0], chocolate: [210, 105, 30], coral: [255, 127, 80], cornflowerblue: [100, 149, 237], cornsilk: [255, 248, 220], crimson: [220, 20, 60], cyan: [0, 255, 255], darkblue: [0, 0, 139], darkcyan: [0, 139, 139], darkgoldenrod: [184, 134, 11], darkgray: [169, 169, 169], darkgreen: [0, 100, 0], darkgrey: [169, 169, 169], darkkhaki: [189, 183, 107], darkmagenta: [139, 0, 139], darkolivegreen: [85, 107, 47], darkorange: [255, 140, 0], darkorchid: [153, 50, 204], darkred: [139, 0, 0], darksalmon: [233, 150, 122], darkseagreen: [143, 188, 143], darkslateblue: [72, 61, 139], darkslategray: [47, 79, 79], darkslategrey: [47, 79, 79], darkturquoise: [0, 206, 209], darkviolet: [148, 0, 211], deeppink: [255, 20, 147], deepskyblue: [0, 191, 255], dimgray: [105, 105, 105], dimgrey: [105, 105, 105], dodgerblue: [30, 144, 255], firebrick: [178, 34, 34], floralwhite: [255, 250, 240], forestgreen: [34, 139, 34], fuchsia: [255, 0, 255], gainsboro: [220, 220, 220], ghostwhite: [248, 248, 255], gold: [255, 215, 0], goldenrod: [218, 165, 32], gray: [128, 128, 128], grey: [128, 128, 128], green: [0, 128, 0], greenyellow: [173, 255, 47], honeydew: [240, 255, 240], hotpink: [255, 105, 180], indianred: [205, 92, 92], indigo: [75, 0, 130], ivory: [255, 255, 240], khaki: [240, 230, 140], lavender: [230, 230, 250], lavenderblush: [255, 240, 245], lawngreen: [124, 252, 0], lemonchiffon: [255, 250, 205], lightblue: [173, 216, 230], lightcoral: [240, 128, 128], lightcyan: [224, 255, 255], lightgoldenrodyellow: [250, 250, 210], lightgray: [211, 211, 211], lightgreen: [144, 238, 144], lightgrey: [211, 211, 211], lightpink: [255, 182, 193], lightsalmon: [255, 160, 122], lightseagreen: [32, 178, 170], lightskyblue: [135, 206, 250], lightslategray: [119, 136, 153], lightslategrey: [119, 136, 153], lightsteelblue: [176, 196, 222], lightyellow: [255, 255, 224], lime: [0, 255, 0], limegreen: [50, 205, 50], linen: [250, 240, 230], magenta: [255, 0, 255], maroon: [128, 0, 0], mediumaquamarine: [102, 205, 170], mediumblue: [0, 0, 205], mediumorchid: [186, 85, 211], mediumpurple: [147, 112, 219], mediumseagreen: [60, 179, 113], mediumslateblue: [123, 104, 238], mediumspringgreen: [0, 250, 154], mediumturquoise: [72, 209, 204], mediumvioletred: [199, 21, 133], midnightblue: [25, 25, 112], mintcream: [245, 255, 250], mistyrose: [255, 228, 225], moccasin: [255, 228, 181], navajowhite: [255, 222, 173], navy: [0, 0, 128], oldlace: [253, 245, 230], olive: [128, 128, 0], olivedrab: [107, 142, 35], orange: [255, 165, 0], orangered: [255, 69, 0], orchid: [218, 112, 214], palegoldenrod: [238, 232, 170], palegreen: [152, 251, 152], paleturquoise: [175, 238, 238], palevioletred: [219, 112, 147], papayawhip: [255, 239, 213], peachpuff: [255, 218, 185], peru: [205, 133, 63], pink: [255, 192, 203], plum: [221, 160, 221], powderblue: [176, 224, 230], purple: [128, 0, 128], red: [255, 0, 0], rosybrown: [188, 143, 143], royalblue: [65, 105, 225], saddlebrown: [139, 69, 19], salmon: [250, 128, 114], sandybrown: [244, 164, 96], seagreen: [46, 139, 87], seashell: [255, 245, 238], sienna: [160, 82, 45], silver: [192, 192, 192], skyblue: [135, 206, 235], slateblue: [106, 90, 205], slategray: [112, 128, 144], slategrey: [112, 128, 144], snow: [255, 250, 250], springgreen: [0, 255, 127], steelblue: [70, 130, 180], tan: [210, 180, 140], teal: [0, 128, 128], thistle: [216, 191, 216], tomato: [255, 99, 71], turquoise: [64, 224, 208], violet: [238, 130, 238], wheat: [245, 222, 179], white: [255, 255, 255], whitesmoke: [245, 245, 245], yellow: [255, 255, 0], yellowgreen: [154, 205, 50] } } }, function (e, t, n) { "use strict"; var r = n(0); e.exports = { mapEmpty: function (e) { return null == e || 0 === Object.keys(e).length }, pushMap: function (e) { var t = this.getMap(e); null == t ? this.setMap(this.extend({}, e, { value: [e.value] })) : t.push(e.value) }, setMap: function (e) { for (var t = e.map, n = e.keys, i = n.length, o = 0; o < i; o++) { var a = n[o]; r.plainObject(a) && this.error("Tried to set map with object key"), o < n.length - 1 ? (null == t[a] && (t[a] = {}), t = t[a]) : t[a] = e.value } }, getMap: function (e) { for (var t = e.map, n = e.keys, i = n.length, o = 0; o < i; o++) { var a = n[o]; if (r.plainObject(a) && this.error("Tried to get map with object key"), null == (t = t[a])) return t } return t }, deleteMap: function (e) { for (var t = e.map, n = e.keys, i = n.length, o = e.keepChildren, a = 0; a < i; a++) { var s = n[a]; if (r.plainObject(s) && this.error("Tried to delete map with object key"), a === e.keys.length - 1) if (o) for (var l = Object.keys(t), c = 0; c < l.length; c++) { var u = l[c]; o[u] || (t[u] = void 0) } else t[s] = void 0; else t = t[s] } } } }, function (e, t, n) { "use strict"; var r = "(?:[-+]?(?:(?:\\d+|\\d*\\.\\d+)(?:[Ee][+-]?\\d+)?))"; e.exports = { regex: { number: r, rgba: "rgb[a]?\\(((?:[-+]?(?:(?:\\d+|\\d*\\.\\d+)(?:[Ee][+-]?\\d+)?))[%]?)\\s*,\\s*((?:[-+]?(?:(?:\\d+|\\d*\\.\\d+)(?:[Ee][+-]?\\d+)?))[%]?)\\s*,\\s*((?:[-+]?(?:(?:\\d+|\\d*\\.\\d+)(?:[Ee][+-]?\\d+)?))[%]?)(?:\\s*,\\s*((?:[-+]?(?:(?:\\d+|\\d*\\.\\d+)(?:[Ee][+-]?\\d+)?))))?\\)", rgbaNoBackRefs: "rgb[a]?\\((?:(?:[-+]?(?:(?:\\d+|\\d*\\.\\d+)(?:[Ee][+-]?\\d+)?))[%]?)\\s*,\\s*(?:(?:[-+]?(?:(?:\\d+|\\d*\\.\\d+)(?:[Ee][+-]?\\d+)?))[%]?)\\s*,\\s*(?:(?:[-+]?(?:(?:\\d+|\\d*\\.\\d+)(?:[Ee][+-]?\\d+)?))[%]?)(?:\\s*,\\s*(?:(?:[-+]?(?:(?:\\d+|\\d*\\.\\d+)(?:[Ee][+-]?\\d+)?))))?\\)", hsla: "hsl[a]?\\(((?:[-+]?(?:(?:\\d+|\\d*\\.\\d+)(?:[Ee][+-]?\\d+)?)))\\s*,\\s*((?:[-+]?(?:(?:\\d+|\\d*\\.\\d+)(?:[Ee][+-]?\\d+)?))[%])\\s*,\\s*((?:[-+]?(?:(?:\\d+|\\d*\\.\\d+)(?:[Ee][+-]?\\d+)?))[%])(?:\\s*,\\s*((?:[-+]?(?:(?:\\d+|\\d*\\.\\d+)(?:[Ee][+-]?\\d+)?))))?\\)", hslaNoBackRefs: "hsl[a]?\\((?:(?:[-+]?(?:(?:\\d+|\\d*\\.\\d+)(?:[Ee][+-]?\\d+)?)))\\s*,\\s*(?:(?:[-+]?(?:(?:\\d+|\\d*\\.\\d+)(?:[Ee][+-]?\\d+)?))[%])\\s*,\\s*(?:(?:[-+]?(?:(?:\\d+|\\d*\\.\\d+)(?:[Ee][+-]?\\d+)?))[%])(?:\\s*,\\s*(?:(?:[-+]?(?:(?:\\d+|\\d*\\.\\d+)(?:[Ee][+-]?\\d+)?))))?\\)", hex3: "\\#[0-9a-fA-F]{3}", hex6: "\\#[0-9a-fA-F]{6}" } } }, function (e, t, n) { "use strict"; var r = n(13), i = n(0); e.exports = { camel2dash: r((function (e) { return e.replace(/([A-Z])/g, (function (e) { return "-" + e.toLowerCase() })) })), dash2camel: r((function (e) { return e.replace(/(-\w)/g, (function (e) { return e[1].toUpperCase() })) })), prependCamel: r((function (e, t) { return e + t[0].toUpperCase() + t.substring(1) }), (function (e, t) { return e + "$" + t })), capitalize: function (e) { return i.emptyString(e) ? e : e.charAt(0).toUpperCase() + e.substring(1) } } }, function (e, t, n) { "use strict"; var r = n(3), i = r ? r.performance : null, o = {}, a = i && i.now ? function () { return i.now() } : function () { return Date.now() }, s = function () { if (r) { if (r.requestAnimationFrame) return function (e) { r.requestAnimationFrame(e) }; if (r.mozRequestAnimationFrame) return function (e) { r.mozRequestAnimationFrame(e) }; if (r.webkitRequestAnimationFrame) return function (e) { r.webkitRequestAnimationFrame(e) }; if (r.msRequestAnimationFrame) return function (e) { r.msRequestAnimationFrame(e) } } return function (e) { e && setTimeout((function () { e(a()) }), 1e3 / 60) } }(); o.requestAnimationFrame = function (e) { s(e) }, o.performanceNow = a, o.debounce = n(26), o.now = function () { return Date.now() }, e.exports = o }, function (t, n) { t.exports = e }, function (e, t, n) { "use strict"; function r(e, t) { return e < t ? -1 : e > t ? 1 : 0 } e.exports = { sort: { ascending: r, descending: function (e, t) { return -1 * r(e, t) } } } }, function (e, t, n) { "use strict"; function r() { this._obj = {} } var i = r.prototype; i.set = function (e, t) { this._obj[e] = t }, i.delete = function (e) { this._obj[e] = null }, i.has = function (e) { return null != this._obj[e] }, i.get = function (e) { return this._obj[e] }, e.exports = r }, function (e, t, n) { "use strict"; var r = n(1), i = {};[n(30), n(31), n(33), n(34), n(35), n(36), n(37), n(38), n(39), n(40), n(41)].forEach((function (e) { r.extend(i, e) })), e.exports = i }, function (e, t, n) { "use strict"; var r = n(0), i = function (e) { return e = { bfs: e.bfs || !e.dfs, dfs: e.dfs || !e.bfs }, function (t, n, i) { var o; r.plainObject(t) && !r.elementOrCollection(t) && (t = (o = t).roots || o.root, n = o.visit, i = o.directed), i = 2 !== arguments.length || r.fn(n) ? i : n, n = r.fn(n) ? n : function () { }; for (var a, s = this._private.cy, l = t = r.string(t) ? this.filter(t) : t, c = [], u = [], d = {}, p = {}, f = {}, h = 0, g = this.nodes(), m = this.edges(), v = 0; v < l.length; v++)l[v].isNode() && (c.unshift(l[v]), e.bfs && (f[l[v].id()] = !0, u.push(l[v])), p[l[v].id()] = 0); for (; 0 !== c.length;) { if (l = e.bfs ? c.shift() : c.pop(), e.dfs) { if (f[l.id()]) continue; f[l.id()] = !0, u.push(l) } var b, y = p[l.id()], x = d[l.id()], w = null == x ? void 0 : x.connectedNodes().not(l)[0]; if (!0 === (b = n(l, x, w, h++, y))) { a = l; break } if (!1 === b) break; var k = l.connectedEdges(i ? function (e) { return e.data("source") === l.id() } : void 0).intersect(m); for (v = 0; v < k.length; v++) { var A = k[v], E = A.connectedNodes((function (e) { return e.id() !== l.id() })).intersect(g); 0 === E.length || f[E.id()] || (E = E[0], c.push(E), e.bfs && (f[E.id()] = !0, u.push(E)), d[E.id()] = A, p[E.id()] = p[l.id()] + 1) } } var S = []; for (v = 0; v < u.length; v++) { var $ = u[v], C = d[$.id()]; C && S.push(C), S.push($) } return { path: s.collection(S, { unique: !0 }), found: s.collection(a) } } }, o = { breadthFirstSearch: i({ bfs: !0 }), depthFirstSearch: i({ dfs: !0 }) }; o.bfs = o.breadthFirstSearch, o.dfs = o.depthFirstSearch, e.exports = o }, function (e, t, n) { "use strict"; var r = n(0), i = n(9), o = { dijkstra: function (e, t, n) { var o; r.plainObject(e) && !r.elementOrCollection(e) && (e = (o = e).root, t = o.weight, n = o.directed); var a = this._private.cy; t = r.fn(t) ? t : function () { return 1 }; for (var s = r.string(e) ? this.filter(e)[0] : e[0], l = {}, c = {}, u = {}, d = this.edges().filter((function (e) { return !e.isLoop() })), p = this.nodes(), f = function (e) { return l[e.id()] }, h = function (e, t) { l[e.id()] = t, g.updateItem(e) }, g = new i((function (e, t) { return f(e) - f(t) })), m = 0; m < p.length; m++) { var v = p[m]; l[v.id()] = v.same(s) ? 0 : 1 / 0, g.push(v) } for (var b = function (e, r) { for (var i, o = (n ? e.edgesTo(r) : e.edgesWith(r)).intersect(d), a = 1 / 0, s = 0; s < o.length; s++) { var l = o[s], c = t(l); (c < a || !i) && (a = c, i = l) } return { edge: i, dist: a } }; g.size() > 0;) { var y = g.pop(), x = f(y), w = y.id(); if (u[w] = x, x !== 1 / 0) { var k = y.neighborhood().intersect(p); for (m = 0; m < k.length; m++) { var A = k[m], E = A.id(), S = b(y, A), $ = x + S.dist; $ < f(A) && (h(A, $), c[E] = { node: y, edge: S.edge }) } } } return { distanceTo: function (e) { var t = r.string(e) ? p.filter(e)[0] : e[0]; return u[t.id()] }, pathTo: function (e) { var t = r.string(e) ? p.filter(e)[0] : e[0], n = [], i = t; if (t.length > 0) for (n.unshift(t); c[i.id()];) { var o = c[i.id()]; n.unshift(o.edge), n.unshift(o.node), i = o.node } return a.collection(n) } } } }; e.exports = o }, function (e, t) { e.exports = n }, function (e, t, n) { "use strict"; var r = n(0), i = { kruskal: function (e) { var t = this.cy(); function n(e) { for (var t = 0; t < o.length; t++) { var n = o[t]; if (n.anySame(e)) return { eles: n, index: t } } } e = r.fn(e) ? e : function () { return 1 }; for (var i = t.collection(t, []), o = [], a = this.nodes(), s = 0; s < a.length; s++)o.push(a[s].collection()); var l = this.edges().toArray().sort((function (t, n) { return e(t) - e(n) })); for (s = 0; s < l.length; s++) { var c = l[s], u = c.source()[0], d = c.target()[0], p = n(u), f = n(d); p.index !== f.index && (i = i.add(c), o[p.index] = p.eles.add(f.eles), o.splice(f.index, 1)) } return a.add(i) } }; e.exports = i }, function (e, t, n) { "use strict"; var r = n(0), i = { aStar: function (e) { e = e || {}; var t = function e(t, n, r, o) { if (t == n) return o.unshift(i.getElementById(n)), o; if (n in r) { var a = r[n], s = g[n]; return o.unshift(i.getElementById(s)), o.unshift(i.getElementById(n)), e(t, a, r, o) } }, n = function (e, t) { if (0 !== e.length) { for (var n = 0, r = t[e[0]], i = 1; i < e.length; i++) { var o = t[e[i]]; o < r && (r = o, n = i) } return n } }, i = this._private.cy; if (null != e && null != e.root) { var o = r.string(e.root) ? this.filter(e.root)[0] : e.root[0]; if (null != e.goal) { var a = r.string(e.goal) ? this.filter(e.goal)[0] : e.goal[0]; if (null != e.heuristic && r.fn(e.heuristic)) var s = e.heuristic; else s = function () { return 0 }; if (null != e.weight && r.fn(e.weight)) var l = e.weight; else l = function (e) { return 1 }; if (null != e.directed) var c = e.directed; else c = !1; var u = o.id(), d = a.id(), p = [], f = [u], h = {}, g = {}, m = {}, v = {}; m[u] = 0, v[u] = s(o); for (var b = 0; f.length > 0;) { var y = n(f, v), x = i.getElementById(f[y]), w = x.id(); if (b++, w == d) { var k = t(u, d, h, []); return { found: !0, distance: m[w], path: this.spawn(k), steps: b } } p.push(w), f.splice(y, 1); for (var A = x._private.edges, E = 0; E < A.length; E++) { var S = A[E]; if (this.hasElementWithId(S.id()) && (!c || S.data("source") === w)) { var $ = S.source(), C = S.target(), _ = $.id() !== w ? $ : C, O = _.id(); if (this.hasElementWithId(O) && -1 == p.indexOf(O)) { var T = m[w] + l(S); -1 != f.indexOf(O) ? T < m[O] && (m[O] = T, v[O] = T + s(_), h[O] = w) : (m[O] = T, v[O] = T + s(_), f.push(O), h[O] = w, g[O] = S.id()) } } } } return { found: !1, distance: void 0, path: void 0, steps: b } } } } }; e.exports = i }, function (e, t, n) { "use strict"; var r = n(0), i = { floydWarshall: function (e) { e = e || {}; var t = this.cy(); if (null != e.weight && r.fn(e.weight)) var n = e.weight; else n = function (e) { return 1 }; if (null != e.directed) var i = e.directed; else i = !1; for (var o = this.edges().stdFilter((function (e) { return !e.isLoop() })), a = this.nodes(), s = a.length, l = {}, c = 0; c < s; c++)l[a[c].id()] = c; var u = []; for (c = 0; c < s; c++) { for (var d = new Array(s), p = 0; p < s; p++)d[p] = c == p ? 0 : 1 / 0; u.push(d) } var f = [], h = [], g = function (e) { for (var t = 0; t < s; t++) { for (var n = new Array(s), r = 0; r < s; r++)n[r] = void 0; e.push(n) } }; for (g(f), g(h), c = 0; c < o.length; c++) { var m = l[o[c].source().id()], v = l[o[c].target().id()], b = n(o[c]); u[m][v] > b && (u[m][v] = b, f[m][v] = v, h[m][v] = o[c]) } if (!i) for (c = 0; c < o.length; c++)m = l[o[c].target().id()], v = l[o[c].source().id()], b = n(o[c]), u[m][v] > b && (u[m][v] = b, f[m][v] = v, h[m][v] = o[c]); for (var y = 0; y < s; y++)for (c = 0; c < s; c++)for (p = 0; p < s; p++)u[c][y] + u[y][p] < u[c][p] && (u[c][p] = u[c][y] + u[y][p], f[c][p] = f[c][y]); var x = []; for (c = 0; c < s; c++)x.push(a[c].id()); return { distance: function (e, n) { if (r.string(e)) var i = t.filter(e)[0].id(); else i = e.id(); if (r.string(n)) var o = t.filter(n)[0].id(); else o = n.id(); return u[l[i]][l[o]] }, path: function (e, n) { if (r.string(e)) var i = t.filter(e)[0].id(); else i = e.id(); if (r.string(n)) var o = t.filter(n)[0].id(); else o = n.id(); var a = function (e, n, r, i, o) { if (e === n) return t.getElementById(i[e]); if (void 0 !== r[e][n]) { for (var a = [t.getElementById(i[e])], s = e; e !== n;) { s = e, e = r[e][n]; var l = o[s][e]; a.push(l), a.push(t.getElementById(i[e])) } return a } }(l[i], l[o], f, x, h); return t.collection(a) } } } }; e.exports = i }, function (e, t, n) { "use strict"; var r = n(0), i = n(1), o = { bellmanFord: function (e) { var t = this; if (null != (e = e || {}).weight && r.fn(e.weight)) var n = e.weight; else n = function (e) { return 1 }; if (null != e.directed) var o = e.directed; else o = !1; if (null != e.root) { if (r.string(e.root)) var a = this.filter(e.root)[0]; else a = e.root[0]; for (var s = this._private.cy, l = this.edges().stdFilter((function (e) { return !e.isLoop() })), c = this.nodes(), u = c.length, d = {}, p = 0; p < u; p++)d[c[p].id()] = p; var f = [], h = [], g = []; for (p = 0; p < u; p++)c[p].id() === a.id() ? f[p] = 0 : f[p] = 1 / 0, h[p] = void 0; var m = !1; for (p = 1; p < u; p++) { m = !1; for (var v = 0; v < l.length; v++) { var b, y = d[l[v].source().id()], x = d[l[v].target().id()], w = n(l[v]); (b = f[y] + w) < f[x] && (f[x] = b, h[x] = y, g[x] = l[v], m = !0), o || (b = f[x] + w) < f[y] && (f[y] = b, h[y] = x, g[y] = l[v], m = !0) } if (!m) break } if (m) for (v = 0; v < l.length; v++)if (y = d[l[v].source().id()], x = d[l[v].target().id()], w = n(l[v]), f[y] + w < f[x]) return i.error("Graph contains a negative weight cycle for Bellman-Ford"), { pathTo: void 0, distanceTo: void 0, hasNegativeWeightCycle: !0 }; var k = []; for (p = 0; p < u; p++)k.push(c[p].id()); return { distanceTo: function (e) { if (r.string(e)) var t = s.filter(e)[0].id(); else t = e.id(); return f[d[t]] }, pathTo: function (e) { if (r.string(e)) var n = s.filter(e)[0].id(); else n = e.id(); var i = function (e, t, n, r, i, o) { for (; ;) { if (i.push(s.getElementById(r[n])), i.push(o[n]), t === n) return i; var a = e[n]; if (void 0 === a) return; n = a } }(h, d[a.id()], d[n], k, [], g); return null != i && i.reverse(), t.spawn(i) }, hasNegativeWeightCycle: !1 } } } }; e.exports = o }, function (e, t, n) { "use strict"; var r = n(1), i = { kargerStein: function (e) { e = e || {}; var t = function e(t, n, r, i) { return r <= i ? n : e(t, function (e, t, n) { for (var r = n[e], i = r[1], o = r[2], a = t[i], s = t[o], l = n.filter((function (e) { return !(t[e[1]] === a && t[e[2]] === s || t[e[1]] === s && t[e[2]] === a) })), c = 0; c < l.length; c++) { var u = l[c]; u[1] === s ? (l[c] = u.slice(0), l[c][1] = a) : u[2] === s && (l[c] = u.slice(0), l[c][2] = a) } for (c = 0; c < t.length; c++)t[c] === s && (t[c] = a); return l }(Math.floor(Math.random() * n.length), t, n), r - 1, i) }, n = this._private.cy, i = this.edges().stdFilter((function (e) { return !e.isLoop() })), o = this.nodes(), a = o.length, s = i.length, l = Math.ceil(Math.pow(Math.log(a) / Math.LN2, 2)), c = Math.floor(a / Math.sqrt(2)); if (!(a < 2)) { for (var u = {}, d = 0; d < a; d++)u[o[d].id()] = d; var p = []; for (d = 0; d < s; d++) { var f = i[d]; p.push([d, u[f.source().id()], u[f.target().id()]]) } var h, g = 1 / 0, m = []; for (d = 0; d < a; d++)m.push(d); for (var v = 0; v <= l; v++) { var b = m.slice(0), y = t(b, p, a, c), x = b.slice(0), w = t(b, y, c, 2), k = t(x, y, c, 2); w.length <= k.length && w.length < g ? (g = w.length, h = [w, b]) : k.length <= w.length && k.length < g && (g = k.length, h = [k, x]) } var A = h[0].map((function (e) { return i[e[0]] })), E = [], S = [], $ = h[1][0]; for (d = 0; d < h[1].length; d++)h[1][d] === $ ? E.push(o[d]) : S.push(o[d]); return { cut: this.spawn(n, A), partition1: this.spawn(E), partition2: this.spawn(S) } } r.error("At least 2 nodes are required for Karger-Stein algorithm") } }; e.exports = i }, function (e, t, n) { "use strict"; var r = n(0), i = { pageRank: function (e) { var t = function (e) { for (var t = e.length, n = 0, r = 0; r < t; r++)n += e[r]; for (r = 0; r < t; r++)e[r] = e[r] / n }; if (null != (e = e || {}) && null != e.dampingFactor) var n = e.dampingFactor; else n = .8; if (null != e && null != e.precision) var i = e.precision; else i = 1e-6; if (null != e && null != e.iterations) var o = e.iterations; else o = 200; if (null != e && null != e.weight && r.fn(e.weight)) var a = e.weight; else a = function (e) { return 1 }; for (var s = this._private.cy, l = this.edges().stdFilter((function (e) { return !e.isLoop() })), c = this.nodes(), u = c.length, d = l.length, p = {}, f = 0; f < u; f++)p[c[f].id()] = f; var h = [], g = [], m = (1 - n) / u; for (f = 0; f < u; f++) { for (var v = [], b = 0; b < u; b++)v.push(0); h.push(v), g.push(0) } for (f = 0; f < d; f++) { var y = l[f], x = p[y.source().id()], w = p[y.target().id()], k = a(y); h[w][x] += k, g[x] += k } var A = 1 / u + m; for (b = 0; b < u; b++)if (0 === g[b]) for (f = 0; f < u; f++)h[f][b] = A; else for (f = 0; f < u; f++)h[f][b] = h[f][b] / g[b] + m; var E, S = [], $ = []; for (f = 0; f < u; f++)S.push(1), $.push(0); for (var C = 0; C < o; C++) { var _ = $.slice(0); for (f = 0; f < u; f++)for (b = 0; b < u; b++)_[f] += h[f][b] * S[b]; t(_), E = S, S = _; var O = 0; for (f = 0; f < u; f++)O += Math.pow(E[f] - S[f], 2); if (O < i) break } return { rank: function (e) { if (r.string(e)) var t = s.filter(e)[0].id(); else t = e.id(); return S[p[t]] } } } }; e.exports = i }, function (e, t, n) { "use strict"; var r = n(0), i = n(1), o = { degreeCentralityNormalized: function (e) { e = e || {}; var t = this.cy(); if (null != e.directed) var n = e.directed; else n = !1; var o = this.nodes(), a = o.length; if (n) { var s = {}, l = {}, c = 0, u = 0; for (f = 0; f < a; f++)h = o[f], g = this.degreeCentrality(i.extend({}, e, { root: h })), c < g.indegree && (c = g.indegree), u < g.outdegree && (u = g.outdegree), s[h.id()] = g.indegree, l[h.id()] = g.outdegree; return { indegree: function (e) { return 0 == c ? 0 : (e = r.string(e) ? t.filter(e)[0].id() : e.id(), s[e] / c) }, outdegree: function (e) { return 0 == u ? 0 : (e = r.string(e) ? t.filter(e)[0].id() : e.id(), l[e] / u) } } } for (var d = {}, p = 0, f = 0; f < a; f++) { var h = o[f], g = this.degreeCentrality(i.extend({}, e, { root: h })); p < g.degree && (p = g.degree), d[h.id()] = g.degree } return { degree: function (e) { return 0 == p ? 0 : (e = r.string(e) ? t.filter(e)[0].id() : e.id(), d[e] / p) } } }, degreeCentrality: function (e) { if (null != (e = e || {}) && null != e.root) { var t = r.string(e.root) ? this.filter(e.root)[0] : e.root[0]; if (null != e.weight && r.fn(e.weight)) var n = e.weight; else n = function (e) { return 1 }; if (null != e.directed) var i = e.directed; else i = !1; if (null != e.alpha && r.number(e.alpha)) var o = e.alpha; else o = 0; if (i) { var a = t.connectedEdges('edge[target = "' + t.id() + '"]').intersection(this), s = t.connectedEdges('edge[source = "' + t.id() + '"]').intersection(this), l = a.length, c = s.length, u = 0, d = 0; for (g = 0; g < a.length; g++)u += n(a[g]); for (g = 0; g < s.length; g++)d += n(s[g]); return { indegree: Math.pow(l, 1 - o) * Math.pow(u, o), outdegree: Math.pow(c, 1 - o) * Math.pow(d, o) } } for (var p = t.connectedEdges().intersection(this), f = p.length, h = 0, g = 0; g < p.length; g++)h += n(p[g]); return { degree: Math.pow(f, 1 - o) * Math.pow(h, o) } } } }; o.dc = o.degreeCentrality, o.dcn = o.degreeCentralityNormalised = o.degreeCentralityNormalized, e.exports = o }, function (e, t, n) { "use strict"; var r = n(0), i = { closenessCentralityNormalized: function (e) { e = e || {}; var t = this.cy(), n = e.harmonic; void 0 === n && (n = !0); for (var i = {}, o = 0, a = this.nodes(), s = this.floydWarshall({ weight: e.weight, directed: e.directed }), l = 0; l < a.length; l++) { for (var c = 0, u = 0; u < a.length; u++)if (l != u) { var d = s.distance(a[l], a[u]); c += n ? 1 / d : d } n || (c = 1 / c), o < c && (o = c), i[a[l].id()] = c } return { closeness: function (e) { return 0 == o ? 0 : (e = r.string(e) ? t.filter(e)[0].id() : e.id(), i[e] / o) } } }, closenessCentrality: function (e) { if (null != (e = e || {}).root) { if (r.string(e.root)) var t = this.filter(e.root)[0]; else t = e.root[0]; if (null != e.weight && r.fn(e.weight)) var n = e.weight; else n = function () { return 1 }; if (null != e.directed && r.bool(e.directed)) var i = e.directed; else i = !1; var o = e.harmonic; void 0 === o && (o = !0); for (var a = this.dijkstra({ root: t, weight: n, directed: i }), s = 0, l = this.nodes(), c = 0; c < l.length; c++)if (l[c].id() != t.id()) { var u = a.distanceTo(l[c]); s += o ? 1 / u : u } return o ? s : 1 / s } } }; i.cc = i.closenessCentrality, i.ccn = i.closenessCentralityNormalised = i.closenessCentralityNormalized, e.exports = i }, function (e, t, n) { "use strict"; var r = n(0), i = n(9), o = { betweennessCentrality: function (e) { var t, n; e = e || {}, r.fn(e.weight) ? (n = e.weight, t = !0) : t = !1; for (var o = null != e.directed && e.directed, a = this._private.cy, s = this.nodes(), l = {}, c = {}, u = 0, d = function (e, t) { c[e] = t, t > u && (u = t) }, p = function (e) { return c[e] }, f = 0; f < s.length; f++)l[w = (k = s[f]).id()] = o ? k.outgoers().nodes() : k.openNeighborhood().nodes(), d(w, 0); for (var h = 0; h < s.length; h++) { var g = s[h].id(), m = [], v = {}, b = {}, y = {}, x = new i((function (e, t) { return y[e] - y[t] })); for (f = 0; f < s.length; f++) { var w; v[w = s[f].id()] = [], b[w] = 0, y[w] = 1 / 0 } for (b[g] = 1, y[g] = 0, x.push(g); !x.empty();) { var k = x.pop(); if (m.push(k), t) for (var A = 0; A < l[k].length; A++) { var E = l[k][A], S = a.getElementById(k), $ = n(S.edgesTo(E).length > 0 ? S.edgesTo(E)[0] : E.edgesTo(S)[0]); E = E.id(), y[E] > y[k] + $ && (y[E] = y[k] + $, x.nodes.indexOf(E) < 0 ? x.push(E) : x.updateItem(E), b[E] = 0, v[E] = []), y[E] == y[k] + $ && (b[E] = b[E] + b[k], v[E].push(k)) } else for (A = 0; A < l[k].length; A++)E = l[k][A].id(), y[E] == 1 / 0 && (x.push(E), y[E] = y[k] + 1), y[E] == y[k] + 1 && (b[E] = b[E] + b[k], v[E].push(k)) } var C = {}; for (f = 0; f < s.length; f++)C[s[f].id()] = 0; for (; m.length > 0;)for (E = m.pop(), A = 0; A < v[E].length; A++)C[k = v[E][A]] = C[k] + b[k] / b[E] * (1 + C[E]), E != s[h].id() && d(E, p(E) + C[E]) } var _ = { betweenness: function (e) { return e = r.string(e) ? a.filter(e).id() : e.id(), p(e) }, betweennessNormalized: function (e) { return 0 == u ? 0 : (e = r.string(e) ? a.filter(e).id() : e.id(), p(e) / u) } }; return _.betweennessNormalised = _.betweennessNormalized, _ } }; o.bc = o.betweennessCentrality, e.exports = o }, function (e, t, n) { "use strict"; var r = n(4), i = { animate: r.animate(), animation: r.animation(), animated: r.animated(), clearQueue: r.clearQueue(), delay: r.delay(), delayAnimation: r.delayAnimation(), stop: r.stop() }; e.exports = i }, function (e, t, n) { "use strict"; var r = n(1), i = n(44), o = n(2), a = n(0), s = { animated: function () { return function () { var e = void 0 !== this.length ? this : [this]; if (!(this._private.cy || this).styleEnabled()) return !1; var t = e[0]; return t ? t._private.animation.current.length > 0 : void 0 } }, clearQueue: function () { return function () { var e = void 0 !== this.length ? this : [this]; if (!(this._private.cy || this).styleEnabled()) return this; for (var t = 0; t < e.length; t++)e[t]._private.animation.queue = []; return this } }, delay: function () { return function (e, t) { return (this._private.cy || this).styleEnabled() ? this.animate({ delay: e, duration: e, complete: t }) : this } }, delayAnimation: function () { return function (e, t) { return (this._private.cy || this).styleEnabled() ? this.animation({ delay: e, duration: e, complete: t }) : this } }, animation: function () { return function (e, t) { var n = void 0 !== this.length, s = n ? this : [this], l = this._private.cy || this, c = !n, u = !c; if (!l.styleEnabled()) return this; var d = l.style(); if (e = r.assign({}, e, t), 0 === Object.keys(e).length) return new i(s[0], e); switch (void 0 === e.duration && (e.duration = 400), e.duration) { case "slow": e.duration = 600; break; case "fast": e.duration = 200 }if (u && (e.style = d.getPropsList(e.style || e.css), e.css = void 0), u && null != e.renderedPosition) { var p = e.renderedPosition, f = l.pan(), h = l.zoom(); e.position = o.renderedToModelPosition(p, h, f) } if (c && null != e.panBy) { var g = e.panBy, m = l.pan(); e.pan = { x: m.x + g.x, y: m.y + g.y } } var v = e.center || e.centre; if (c && null != v) { var b = l.getCenterPan(v.eles, e.zoom); null != b && (e.pan = b) } if (c && null != e.fit) { var y = e.fit, x = l.getFitViewport(y.eles || y.boundingBox, y.padding); null != x && (e.pan = x.pan, e.zoom = x.zoom) } if (c && a.plainObject(e.zoom)) { var w = l.getZoomedViewport(e.zoom); null != w && (w.zoomed && (e.zoom = w.zoom), w.panned && (e.pan = w.pan)) } return new i(s[0], e) } }, animate: function () { return function (e, t) { var n = void 0 !== this.length ? this : [this]; if (!(this._private.cy || this).styleEnabled()) return this; t && (e = r.extend({}, e, t)); for (var i = 0; i < n.length; i++) { var o = n[i], a = o.animated() && (void 0 === e.queue || e.queue); o.animation(e, a ? { queue: !0 } : void 0).play() } return this } }, stop: function () { return function (e, t) { var n = void 0 !== this.length ? this : [this], r = this._private.cy || this; if (!r.styleEnabled()) return this; for (var i = 0; i < n.length; i++) { for (var o = n[i]._private, a = o.animation.current, s = 0; s < a.length; s++) { var l = a[s]._private; t && (l.duration = 0) } e && (o.animation.queue = []), t || (o.animation.current = []) } return r.notify({ eles: this, type: "draw" }), this } } }; e.exports = s }, function (e, t, n) { "use strict"; var r = n(1), i = n(0), o = n(5), a = function (e, t, n) { var o = this._private = r.extend({ duration: 1e3 }, t, n); o.target = e, o.style = o.style || o.css, o.started = !1, o.playing = !1, o.hooked = !1, o.applying = !1, o.progress = 0, o.completes = [], o.frames = [], o.complete && i.fn(o.complete) && o.completes.push(o.complete), this.length = 1, this[0] = this }, s = a.prototype; r.extend(s, { instanceString: function () { return "animation" }, hook: function () { var e = this._private; if (!e.hooked) { var t = e.target._private.animation; (e.queue ? t.queue : t.current).push(this), i.elementOrCollection(e.target) && e.target.cy().addToAnimationPool(e.target), e.hooked = !0 } return this }, play: function () { var e = this._private; return 1 === e.progress && (e.progress = 0), e.playing = !0, e.started = !1, e.stopped = !1, this.hook(), this }, playing: function () { return this._private.playing }, apply: function () { var e = this._private; return e.applying = !0, e.started = !1, e.stopped = !1, this.hook(), this }, applying: function () { return this._private.applying }, pause: function () { var e = this._private; return e.playing = !1, e.started = !1, this }, stop: function () { var e = this._private; return e.playing = !1, e.started = !1, e.stopped = !0, this }, rewind: function () { return this.progress(0) }, fastforward: function () { return this.progress(1) }, time: function (e) { var t = this._private; return void 0 === e ? t.progress * t.duration : this.progress(e / t.duration) }, progress: function (e) { var t = this._private, n = t.playing; return void 0 === e ? t.progress : (n && this.pause(), t.progress = e, t.started = !1, n && this.play(), this) }, completed: function () { return 1 === this._private.progress }, reverse: function () { var e = this._private, t = e.playing; t && this.pause(), e.progress = 1 - e.progress, e.started = !1; var n = function (t, n) { var r = e[t]; null != r && (e[t] = e[n], e[n] = r) }; if (n("zoom", "startZoom"), n("pan", "startPan"), n("position", "startPosition"), e.style) for (var r = 0; r < e.style.length; r++) { var i = e.style[r], o = i.name, a = e.startStyle[o]; e.startStyle[o] = i, e.style[r] = a } return t && this.play(), this }, promise: function (e) { var t = this._private, n = void 0; switch (e) { case "frame": n = t.frames; break; default: case "complete": case "completed": n = t.completes }return new o((function (e, t) { n.push((function () { e() })) })) } }), s.complete = s.completed, e.exports = a }, function (e, t, n) { "use strict"; var r = n(1), i = n(0), o = { data: function (e) { return e = r.extend({}, { field: "data", bindingEvent: "data", allowBinding: !1, allowSetting: !1, allowGetting: !1, settingEvent: "data", settingTriggersEvent: !1, triggerFnName: "trigger", immutableKeys: {}, updateStyle: !1, beforeGet: function (e) { }, beforeSet: function (e, t) { }, onSet: function (e) { }, canSet: function (e) { return !0 } }, e), function (t, n) { var r = e, o = void 0 !== this.length, a = o ? this : [this], s = o ? this[0] : this; if (i.string(t)) { if (r.allowGetting && void 0 === n) { var l = void 0; return s && (r.beforeGet(s), l = s._private[r.field][t]), l } if (r.allowSetting && void 0 !== n && !r.immutableKeys[t]) { var c = function (e, t, n) { return t in e ? Object.defineProperty(e, t, { value: n, enumerable: !0, configurable: !0, writable: !0 }) : e[t] = n, e }({}, t, n); r.beforeSet(this, c); for (var u = 0, d = a.length; u < d; u++) { var p = a[u]; r.canSet(p) && (p._private[r.field][t] = n) } r.updateStyle && this.updateStyle(), r.onSet(this), r.settingTriggersEvent && this[r.triggerFnName](r.settingEvent) } } else if (r.allowSetting && i.plainObject(t)) { var f = t, h = void 0, g = void 0, m = Object.keys(f); r.beforeSet(this, f); for (var v = 0; v < m.length; v++)if (g = f[h = m[v]], !r.immutableKeys[h]) for (var b = 0; b < a.length; b++) { var y = a[b]; r.canSet(y) && (y._private[r.field][h] = g) } r.updateStyle && this.updateStyle(), r.onSet(this), r.settingTriggersEvent && this[r.triggerFnName](r.settingEvent) } else if (r.allowBinding && i.fn(t)) { var x = t; this.on(r.bindingEvent, x) } else if (r.allowGetting && void 0 === t) { var w = void 0; return s && (r.beforeGet(s), w = s._private[r.field]), w } return this } }, removeData: function (e) { return e = r.extend({}, { field: "data", event: "data", triggerFnName: "trigger", triggerEvent: !1, immutableKeys: {} }, e), function (t) { var n = e, r = void 0 !== this.length ? this : [this]; if (i.string(t)) { for (var o = t.split(/\s+/), a = o.length, s = 0; s < a; s++) { var l = o[s]; if (!i.emptyString(l) && !n.immutableKeys[l]) for (var c = 0, u = r.length; c < u; c++)r[c]._private[n.field][l] = void 0 } n.triggerEvent && this[n.triggerFnName](n.event) } else if (void 0 === t) { for (var d = 0, p = r.length; d < p; d++)for (var f = r[d]._private[n.field], h = Object.keys(f), g = 0; g < h.length; g++) { var m = h[g]; !n.immutableKeys[m] && (f[m] = void 0) } n.triggerEvent && this[n.triggerFnName](n.event) } return this } } }; e.exports = o }, function (e, t, n) { "use strict"; var r = n(5), i = { eventAliasesOn: function (e) { var t = e; t.addListener = t.listen = t.bind = t.on, t.unlisten = t.unbind = t.off = t.removeListener, t.trigger = t.emit, t.pon = t.promiseOn = function (e, t) { var n = this, i = Array.prototype.slice.call(arguments, 0); return new r((function (e, t) { var r = i.concat([function (t) { n.off.apply(n, o), e(t) }]), o = r.concat([]); n.on.apply(n, r) })) } } }; e.exports = i }, function (e, t, n) { "use strict"; var r = n(8), i = { classes: function (e) { e = (e || "").match(/\S+/g) || []; for (var t = this, n = [], i = new r(e), o = function (e) { var o = t[e], a = o._private, s = a.classes, l = !1; i.forEach((function (e) { s.has(e) || (l = !0) })), l || s.forEach((function (e) { i.has(e) || (l = !0) })), l && (a.classes = new r(i), n.push(o)) }, a = 0; a < t.length; a++)o(a); return n.length > 0 && this.spawn(n).updateStyle().emit("class"), t }, addClass: function (e) { return this.toggleClass(e, !0) }, hasClass: function (e) { var t = this[0]; return null != t && t._private.classes.has(e) }, toggleClass: function (e, t) { for (var n = e.match(/\S+/g) || [], r = [], i = 0, o = this.length; i < o; i++)for (var a = this[i], s = !1, l = 0; l < n.length; l++) { var c = n[l], u = a._private.classes, d = u.has(c); t || void 0 === t && !d ? (u.add(c), d || s || (r.push(a), s = !0)) : (u.delete(c), d && !s && (r.push(a), s = !0)) } return r.length > 0 && this.spawn(r).updateStyle().emit("class"), this }, removeClass: function (e) { return this.toggleClass(e, !1) }, flashClass: function (e, t) { var n = this; if (null == t) t = 250; else if (0 === t) return n; return n.addClass(e), setTimeout((function () { n.removeClass(e) }), t), n } }; e.exports = i }, function (e, t, n) { "use strict"; n(0); var r = n(6), i = { allAre: function (e) { var t = new r(e); return this.every((function (e) { return t.matches(e) })) }, is: function (e) { var t = new r(e); return this.some((function (e) { return t.matches(e) })) }, some: function (e, t) { for (var n = 0; n < this.length; n++)if (t ? e.apply(t, [this[n], n, this]) : e(this[n], n, this)) return !0; return !1 }, every: function (e, t) { for (var n = 0; n < this.length; n++)if (!(t ? e.apply(t, [this[n], n, this]) : e(this[n], n, this))) return !1; return !0 }, same: function (e) { return e = this.cy().collection(e), this.length === e.length && this.every((function (t) { return e.hasElementWithId(t.id()) })) }, anySame: function (e) { return e = this.cy().collection(e), this.some((function (t) { return e.hasElementWithId(t.id()) })) }, allAreNeighbors: function (e) { e = this.cy().collection(e); var t = this.neighborhood(); return e.every((function (e) { return t.hasElementWithId(e.id()) })) }, contains: function (e) { e = this.cy().collection(e); var t = this; return e.every((function (e) { return t.hasElementWithId(e.id()) })) } }; i.allAreNeighbours = i.allAreNeighbors, i.has = i.contains, e.exports = i }, function (e, t, n) { "use strict"; var r = n(1), i = n(50), o = n(10), a = function (e) { for (var t = void 0, n = void 0, r = void 0, o = 0; o < i.length; o++) { var a = i[o], s = a.name, l = e.match(a.regexObj); if (null != l) { n = l, t = a, r = s; var c = l[0]; e = e.substring(c.length); break } } return { expr: t, match: n, name: r, remaining: e } }; e.exports = { parse: function (e) { var t = this._private.selectorText = e, n = this[0] = o(); for (this.length = 1, t = function (e) { var t = e.match(/^\s+/); if (t) { var n = t[0]; e = e.substring(n.length) } return e }(t); ;) { var i = a(t); if (null == i.expr) return r.error("The selector `" + e + "`is invalid"), !1; var s = i.match.slice(1), l = i.expr.populate(this, n, s); if (!1 === l) return !1; if (null != l && (n = l), (t = i.remaining).match(/^\s*$/)) break } for (var c = 0; c < this.length; c++) { var u = this[c]; if (null != u.subject) { for (; u.subject !== u;)if (null != u.parent) { var d = u.parent, p = u; p.parent = null, d.child = p, u = d } else { if (null == u.ancestor) return u.source || u.target || u.connectedNodes ? (r.error("The selector `" + this.text() + "` can not contain a subject selector that applies to the source or target of an edge selector"), !1) : (r.error("When adjusting references for the selector `" + this.text() + "`, neither parent nor ancestor was found"), !1); var f = u.ancestor, h = u; h.ancestor = null, f.descendant = h, u = f } this[c] = u.subject } } return !0 } } }, function (e, t, n) { "use strict"; var r = function (e, t) { if (Array.isArray(e)) return e; if (Symbol.iterator in Object(e)) return function (e, t) { var n = [], r = !0, i = !1, o = void 0; try { for (var a, s = e[Symbol.iterator](); !(r = (a = s.next()).done) && (n.push(a.value), !t || n.length !== t); r = !0); } catch (e) { i = !0, o = e } finally { try { !r && s.return && s.return() } finally { if (i) throw o } } return n }(e, t); throw new TypeError("Invalid attempt to destructure non-iterable instance") }, i = n(15).stateSelectorRegex, o = n(51), a = n(1), s = n(10), l = function (e) { return e.replace(new RegExp("\\\\(" + o.metaChar + ")", "g"), (function (e, t) { return t })) }, c = function (e, t, n) { t === e[e.length - 1] && (e[e.length - 1] = n) }, u = [{ name: "group", query: !0, regex: "(" + o.group + ")", populate: function (e, t, n) { var i = r(n, 1)[0]; t.group = "*" === i ? i : i + "s" } }, { name: "state", query: !0, regex: i, populate: function (e, t, n) { var i = r(n, 1)[0]; t.colonSelectors.push(i) } }, { name: "id", query: !0, regex: "\\#(" + o.id + ")", populate: function (e, t, n) { var i = r(n, 1)[0]; t.ids.push(l(i)) } }, { name: "className", query: !0, regex: "\\.(" + o.className + ")", populate: function (e, t, n) { var i = r(n, 1)[0]; t.classes.push(l(i)) } }, { name: "dataExists", query: !0, regex: "\\[\\s*(" + o.variable + ")\\s*\\]", populate: function (e, t, n) { var i = r(n, 1)[0]; t.data.push({ field: l(i) }) } }, { name: "dataCompare", query: !0, regex: "\\[\\s*(" + o.variable + ")\\s*(" + o.comparatorOp + ")\\s*(" + o.value + ")\\s*\\]", populate: function (e, t, n) { var i = r(n, 3), a = i[0], s = i[1], c = i[2]; c = null != new RegExp("^" + o.string + "$").exec(c) ? c.substring(1, c.length - 1) : parseFloat(c), t.data.push({ field: l(a), operator: s, value: c }) } }, { name: "dataBool", query: !0, regex: "\\[\\s*(" + o.boolOp + ")\\s*(" + o.variable + ")\\s*\\]", populate: function (e, t, n) { var i = r(n, 2), o = i[0], a = i[1]; t.data.push({ field: l(a), operator: o }) } }, { name: "metaCompare", query: !0, regex: "\\[\\[\\s*(" + o.meta + ")\\s*(" + o.comparatorOp + ")\\s*(" + o.number + ")\\s*\\]\\]", populate: function (e, t, n) { var i = r(n, 3), o = i[0], a = i[1], s = i[2]; t.meta.push({ field: l(o), operator: a, value: parseFloat(s) }) } }, { name: "nextQuery", separator: !0, regex: o.separator, populate: function (e) { var t = e[e.length++] = s(); return e.currentSubject = null, t } }, { name: "directedEdge", separator: !0, regex: o.directedEdge, populate: function (e, t) { var n = s(), r = t, i = s(); return n.group = "edges", n.target = i, n.source = r, n.subject = e.currentSubject, c(e, t, n), i } }, { name: "undirectedEdge", separator: !0, regex: o.undirectedEdge, populate: function (e, t) { var n = s(), r = t, i = s(); return n.group = "edges", n.connectedNodes = [r, i], n.subject = e.currentSubject, c(e, t, n), i } }, { name: "child", separator: !0, regex: o.child, populate: function (e, t) { var n = s(); return n.parent = t, n.subject = e.currentSubject, c(e, t, n), n } }, { name: "descendant", separator: !0, regex: o.descendant, populate: function (e, t) { var n = s(); return n.ancestor = t, n.subject = e.currentSubject, c(e, t, n), n } }, { name: "subject", modifier: !0, regex: o.subject, populate: function (e, t) { if (null != e.currentSubject && t.subject != t) return a.error("Redefinition of subject in selector `" + e.toString() + "`"), !1; e.currentSubject = t, t.subject = t, e[e.length - 1].subject = t } }]; u.forEach((function (e) { return e.regexObj = new RegExp("^" + e.regex) })), e.exports = u }, function (e, t, n) { "use strict"; var r = { metaChar: "[\\!\\\"\\#\\$\\%\\&\\'\\(\\)\\*\\+\\,\\.\\/\\:\\;\\<\\=\\>\\?\\@\\[\\]\\^\\`\\{\\|\\}\\~]", comparatorOp: "=|\\!=|>|>=|<|<=|\\$=|\\^=|\\*=", boolOp: "\\?|\\!|\\^", string: "\"(?:\\\\\"|[^\"])*\"|'(?:\\\\'|[^'])*'", number: n(1).regex.number, meta: "degree|indegree|outdegree", separator: "\\s*,\\s*", descendant: "\\s+", child: "\\s+>\\s+", subject: "\\$", group: "node|edge|\\*", directedEdge: "\\s+->\\s+", undirectedEdge: "\\s+<->\\s+" }; r.variable = "(?:[\\w-]|(?:\\\\" + r.metaChar + "))+", r.value = r.string + "|" + r.number, r.className = r.variable, r.id = r.variable, function () { var e = void 0, t = void 0, n = void 0; for (e = r.comparatorOp.split("|"), n = 0; n < e.length; n++)t = e[n], r.comparatorOp += "|@" + t; for (e = r.comparatorOp.split("|"), n = 0; n < e.length; n++)(t = e[n]).indexOf("!") >= 0 || "=" !== t && (r.comparatorOp += "|\\!" + t) }(), e.exports = r }, function (e, t, n) { "use strict"; var r = n(15).stateSelectorMatches, i = n(0), o = function (e, t) { for (var n = !0, r = 0; r < e[t.name].length; r++) { var o = e[t.name][r], a = o.operator, s = o.value, l = o.field, c = void 0, u = t.fieldValue(l); if (null != a && null != s) { var d = i.string(u) || i.number(u) ? "" + u : "", p = "" + s, f = !1; a.indexOf("@") >= 0 && (d = d.toLowerCase(), p = p.toLowerCase(), a = a.replace("@", ""), f = !0); var h = !1; a.indexOf("!") >= 0 && (a = a.replace("!", ""), h = !0), f && (s = p.toLowerCase(), u = d.toLowerCase()); var g = !1; switch (a) { case "*=": c = d.indexOf(p) >= 0; break; case "$=": c = d.indexOf(p, d.length - p.length) >= 0; break; case "^=": c = 0 === d.indexOf(p); break; case "=": c = u === s; break; case ">": g = !0, c = u > s; break; case ">=": g = !0, c = u >= s; break; case "<": g = !0, c = u < s; break; case "<=": g = !0, c = u <= s; break; default: c = !1 }!h || null == u && g || (c = !c) } else if (null != a) switch (a) { case "?": c = !!u; break; case "!": c = !u; break; case "^": c = void 0 === u } else c = void 0 !== u; if (!c) { n = !1; break } } return n }, a = function (e, t, n) { if (null != e) { var r = !1; if (!t) return !1; n = n(); for (var i = 0; i < n.length; i++)if (s(e, n[i])) { r = !0; break } return r } return !0 }, s = function (e, t) { if (e.groupOnly) return "*" === e.group || e.group === t.group(); if (null != e.group && "*" != e.group && e.group != t.group()) return !1; var n = t.cy(), i = void 0, s = !0; for (i = 0; i < e.colonSelectors.length; i++) { var l = e.colonSelectors[i]; if (!(s = r(l, t))) break } if (!s) return !1; var c = !0; for (i = 0; i < e.ids.length; i++) { var u = e.ids[i], d = t.id(); if (!(c = c && u == d)) break } if (!c) return !1; var p = !0; for (i = 0; i < e.classes.length; i++) { var f = e.classes[i]; if (!(p = p && t.hasClass(f))) break } if (!p) return !1; if (!o(e, { name: "data", fieldValue: function (e) { return t.data(e) } })) return !1; if (!o(e, { name: "meta", fieldValue: function (e) { return t[e]() } })) return !1; if (null != e.collection && !e.collection.hasElementWithId(t.id())) return !1; if (null != e.filter && t.collection().some(e.filter)) return !1; var h = n.hasCompoundNodes(), g = function () { return t.source() }, m = function () { return t.target() }; if (!a(e.parent, h, (function () { return t.parent() }))) return !1; if (!a(e.ancestor, h, (function () { return t.parents() }))) return !1; if (!a(e.child, h, (function () { return t.children() }))) return !1; if (!a(e.descendant, h, (function () { return t.descendants() }))) return !1; if (!a(e.source, !0, g)) return !1; if (!a(e.target, !0, m)) return !1; if (e.connectedNodes) { var v = e.connectedNodes[0], b = e.connectedNodes[1]; if (a(v, !0, g) && a(b, !0, m)); else if (!a(v, !0, m) || !a(b, !0, g)) return !1 } return !0 }; e.exports = { matches: function (e) { if (this.invalid()) return !1; for (var t = 0; t < this.length; t++) { var n = this[t]; if (s(n, e)) return !0 } return !1 }, filter: function (e) { var t = this, n = e.cy(); if (t.invalid()) return n.collection(); if (1 === t.length && 1 === t[0].length && 1 === t[0].ids.length) return e.getElementById(t[0].ids[0]).collection(); var r = function (e) { for (var n = 0; n < t.length; n++) { var r = t[n]; if (s(r, e)) return !0 } return !1 }; return null == t.text() && (r = function () { return !0 }), e.filter(r) } } }, function (e, t, n) { "use strict"; var r = n(8), i = { parent: function (e) { var t = []; if (1 === this.length) { var n = this[0]._private.parent; if (n) return n } for (var r = 0; r < this.length; r++) { var i = this[r]._private.parent; i && t.push(i) } return this.spawn(t, { unique: !0 }).filter(e) }, parents: function (e) { for (var t = [], n = this.parent(); n.nonempty();) { for (var r = 0; r < n.length; r++) { var i = n[r]; t.push(i) } n = n.parent() } return this.spawn(t, { unique: !0 }).filter(e) }, commonAncestors: function (e) { for (var t = void 0, n = 0; n < this.length; n++) { var r = this[n].parents(); t = (t = t || r).intersect(r) } return t.filter(e) }, orphans: function (e) { return this.stdFilter((function (e) { return e.isOrphan() })).filter(e) }, nonorphans: function (e) { return this.stdFilter((function (e) { return e.isChild() })).filter(e) }, children: function (e) { for (var t = [], n = 0; n < this.length; n++) { var r = this[n]; t = t.concat(r._private.children) } return this.spawn(t, { unique: !0 }).filter(e) }, siblings: function (e) { return this.parent().children().not(this).filter(e) }, isParent: function () { var e = this[0]; if (e) return e.isNode() && 0 !== e._private.children.length }, isChildless: function () { var e = this[0]; if (e) return e.isNode() && 0 === e._private.children.length }, isChild: function () { var e = this[0]; if (e) return e.isNode() && null != e._private.parent }, isOrphan: function () { var e = this[0]; if (e) return e.isNode() && null == e._private.parent }, descendants: function (e) { var t = []; return function e(n) { for (var r = 0; r < n.length; r++) { var i = n[r]; t.push(i), i.children().nonempty() && e(i.children()) } }(this.children()), this.spawn(t, { unique: !0 }).filter(e) } }; function o(e, t, n, i) { for (var o = [], a = new r, s = e.cy().hasCompoundNodes(), l = 0; l < e.length; l++) { var c = e[l]; n ? o.push(c) : s && i(o, a, c) } for (; o.length > 0;) { var u = o.shift(); t(u), a.add(u.id()), s && i(o, a, u) } return e } function a(e, t, n) { if (n.isParent()) for (var r = n._private.children, i = 0; i < r.length; i++) { var o = r[i]; t.has(o.id()) || e.push(o) } } function s(e, t, n) { if (n.isChild()) { var r = n._private.parent; t.has(r.id()) || e.push(r) } } function l(e, t, n) { s(e, t, n), a(e, t, n) } i.forEachDown = function (e) { var t = !(arguments.length > 1 && void 0 !== arguments[1]) || arguments[1]; return o(this, e, t, a) }, i.forEachUp = function (e) { var t = !(arguments.length > 1 && void 0 !== arguments[1]) || arguments[1]; return o(this, e, t, s) }, i.forEachUpAndDown = function (e) { var t = !(arguments.length > 1 && void 0 !== arguments[1]) || arguments[1]; return o(this, e, t, l) }, i.ancestors = i.parents, e.exports = i }, function (e, t, n) { "use strict"; var r, i = n(4), o = void 0; (o = r = { data: i.data({ field: "data", bindingEvent: "data", allowBinding: !0, allowSetting: !0, settingEvent: "data", settingTriggersEvent: !0, triggerFnName: "trigger", allowGetting: !0, immutableKeys: { id: !0, source: !0, target: !0, parent: !0 }, updateStyle: !0 }), removeData: i.removeData({ field: "data", event: "data", triggerFnName: "trigger", triggerEvent: !0, immutableKeys: { id: !0, source: !0, target: !0, parent: !0 }, updateStyle: !0 }), scratch: i.data({ field: "scratch", bindingEvent: "scratch", allowBinding: !0, allowSetting: !0, settingEvent: "scratch", settingTriggersEvent: !0, triggerFnName: "trigger", allowGetting: !0, updateStyle: !0 }), removeScratch: i.removeData({ field: "scratch", event: "scratch", triggerFnName: "trigger", triggerEvent: !0, updateStyle: !0 }), rscratch: i.data({ field: "rscratch", allowBinding: !1, allowSetting: !0, settingTriggersEvent: !1, allowGetting: !0 }), removeRscratch: i.removeData({ field: "rscratch", triggerEvent: !1 }), id: function () { var e = this[0]; if (e) return e._private.data.id } }).attr = o.data, o.removeAttr = o.removeData, e.exports = r }, function (e, t, n) { "use strict"; var r = n(1), i = {}; function o(e) { return function (t) { if (void 0 === t && (t = !0), 0 !== this.length && this.isNode() && !this.removed()) { for (var n = 0, r = this[0], i = r._private.edges, o = 0; o < i.length; o++) { var a = i[o]; !t && a.isLoop() || (n += e(r, a)) } return n } } } function a(e, t) { return function (n) { for (var r = void 0, i = this.nodes(), o = 0; o < i.length; o++) { var a = i[o][e](n); void 0 === a || void 0 !== r && !t(a, r) || (r = a) } return r } } r.extend(i, { degree: o((function (e, t) { return t.source().same(t.target()) ? 2 : 1 })), indegree: o((function (e, t) { return t.target().same(e) ? 1 : 0 })), outdegree: o((function (e, t) { return t.source().same(e) ? 1 : 0 })) }), r.extend(i, { minDegree: a("degree", (function (e, t) { return e < t })), maxDegree: a("degree", (function (e, t) { return e > t })), minIndegree: a("indegree", (function (e, t) { return e < t })), maxIndegree: a("indegree", (function (e, t) { return e > t })), minOutdegree: a("outdegree", (function (e, t) { return e < t })), maxOutdegree: a("outdegree", (function (e, t) { return e > t })) }), r.extend(i, { totalDegree: function (e) { for (var t = 0, n = this.nodes(), r = 0; r < n.length; r++)t += n[r].degree(e); return t } }), e.exports = i }, function (e, t, n) { "use strict"; var r = n(1), i = n(57), o = n(58), a = n(59), s = n(60); e.exports = r.assign({}, i, o, a, s) }, function (e, t, n) { "use strict"; var r, i = n(4), o = n(0), a = n(2), s = void 0, l = function (e, t) { for (var n = 0; n < e.length; n++) { var r = e[n]; if (r.isParent() && !r.locked()) { var i = r._private.position, o = { x: t.x - i.x, y: t.y - i.y }; e.children().shift(o) } } }; (s = r = { position: i.data({ field: "position", bindingEvent: "position", allowBinding: !0, allowSetting: !0, settingEvent: "position", settingTriggersEvent: !0, triggerFnName: "emitAndNotify", allowGetting: !0, validKeys: ["x", "y"], beforeGet: function (e) { e.updateCompoundBounds() }, beforeSet: l, onSet: function (e) { e.dirtyCompoundBoundsCache() }, canSet: function (e) { return !e.locked() } }), silentPosition: i.data({ field: "position", bindingEvent: "position", allowBinding: !1, allowSetting: !0, settingEvent: "position", settingTriggersEvent: !1, triggerFnName: "trigger", allowGetting: !1, validKeys: ["x", "y"], beforeSet: l, onSet: function (e) { e.dirtyCompoundBoundsCache() }, canSet: function (e) { return !e.locked() } }), positions: function (e, t) { if (o.plainObject(e)) t ? this.silentPosition(e) : this.position(e); else if (o.fn(e)) { var n = e, r = this.cy(); r.startBatch(); for (var i = 0; i < this.length; i++) { var a, s = this[i]; (a = n(s, i)) && (t ? s.silentPosition(a) : s.position(a)) } r.endBatch() } return this }, silentPositions: function (e) { return this.positions(e, !0) }, shift: function (e, t) { var n = void 0; if (o.plainObject(e) ? n = e : o.string(e) && o.number(t) && ((n = { x: 0, y: 0 })[e] = t), null != n) for (var r = 0; r < this.length; r++) { var i = this[r], a = i.position(); i.position({ x: a.x + n.x, y: a.y + n.y }) } return this }, renderedPosition: function (e, t) { var n = this[0], r = this.cy(), i = r.zoom(), s = r.pan(), l = o.plainObject(e) ? e : void 0, c = void 0 !== l || void 0 !== t && o.string(e); if (n && n.isNode()) { if (!c) { var u = n.position(); return l = a.modelToRenderedPosition(u, i, s), void 0 === e ? l : l[e] } for (var d = 0; d < this.length; d++) { var p = this[d]; void 0 !== t ? p.position(e, (t - s[e]) / i) : void 0 !== l && p.position(a.renderedToModelPosition(l, i, s)) } } else if (!c) return; return this }, relativePosition: function (e, t) { var n = this[0], r = this.cy(), i = o.plainObject(e) ? e : void 0, a = void 0 !== i || void 0 !== t && o.string(e), s = r.hasCompoundNodes(); if (n && n.isNode()) { if (!a) { var l = n.position(), c = s ? n.parent() : null, u = c && c.length > 0, d = u; u && (c = c[0]); var p = d ? c.position() : { x: 0, y: 0 }; return i = { x: l.x - p.x, y: l.y - p.y }, void 0 === e ? i : i[e] } for (var f = 0; f < this.length; f++) { var h = this[f], g = s ? h.parent() : null, m = g && g.length > 0, v = m; m && (g = g[0]); var b = v ? g.position() : { x: 0, y: 0 }; void 0 !== t ? h.position(e, t + b[e]) : void 0 !== i && h.position({ x: i.x + b.x, y: i.y + b.y }) } } else if (!a) return; return this } }).modelPosition = s.point = s.position, s.modelPositions = s.points = s.positions, s.renderedPoint = s.renderedPosition, s.relativePoint = s.relativePosition, e.exports = r }, function (e, t, n) { "use strict"; var r = n(0), i = n(1), o = n(2), a = void 0, s = void 0; a = s = {}, s.renderedBoundingBox = function (e) { var t = this.boundingBox(e), n = this.cy(), r = n.zoom(), i = n.pan(), o = t.x1 * r + i.x, a = t.x2 * r + i.x, s = t.y1 * r + i.y, l = t.y2 * r + i.y; return { x1: o, x2: a, y1: s, y2: l, w: a - o, h: l - s } }, s.dirtyCompoundBoundsCache = function () { var e = this.cy(); return e.styleEnabled() && e.hasCompoundNodes() ? (this.forEachUp((function (e) { e._private.compoundBoundsClean = !1, e.isParent() && e.emit("bounds") })), this) : this }, s.updateCompoundBounds = function () { var e = this.cy(); if (!e.styleEnabled() || !e.hasCompoundNodes()) return this; if (e.batching()) return this; var t = []; function n(e) { if (e.isParent()) { var n = e._private, r = e.children(), i = "include" === e.pstyle("compound-sizing-wrt-labels").value, o = { width: { val: e.pstyle("min-width").pfValue, left: e.pstyle("min-width-bias-left"), right: e.pstyle("min-width-bias-right") }, height: { val: e.pstyle("min-height").pfValue, top: e.pstyle("min-height-bias-top"), bottom: e.pstyle("min-height-bias-bottom") } }, a = r.boundingBox({ includeLabels: i, includeOverlays: !1, useCache: !1 }), s = n.position; 0 !== a.w && 0 !== a.h || ((a = { w: e.pstyle("width").pfValue, h: e.pstyle("height").pfValue }).x1 = s.x - a.w / 2, a.x2 = s.x + a.w / 2, a.y1 = s.y - a.h / 2, a.y2 = s.y + a.h / 2); var l = o.width.left.value; "px" === o.width.left.units && o.width.val > 0 && (l = 100 * l / o.width.val); var c = o.width.right.value; "px" === o.width.right.units && o.width.val > 0 && (c = 100 * c / o.width.val); var u = o.height.top.value; "px" === o.height.top.units && o.height.val > 0 && (u = 100 * u / o.height.val); var d = o.height.bottom.value; "px" === o.height.bottom.units && o.height.val > 0 && (d = 100 * d / o.height.val); var p = b(o.width.val - a.w, l, c), f = p.biasDiff, h = p.biasComplementDiff, g = b(o.height.val - a.h, u, d), m = g.biasDiff, v = g.biasComplementDiff; n.autoPadding = function (e, t, n, r) { if ("%" !== n.units) return "px" === n.units ? n.pfValue : 0; switch (r) { case "width": return e > 0 ? n.pfValue * e : 0; case "height": return t > 0 ? n.pfValue * t : 0; case "average": return e > 0 && t > 0 ? n.pfValue * (e + t) / 2 : 0; case "min": return e > 0 && t > 0 ? e > t ? n.pfValue * t : n.pfValue * e : 0; case "max": return e > 0 && t > 0 ? e > t ? n.pfValue * e : n.pfValue * t : 0; default: return 0 } }(a.w, a.h, e.pstyle("padding"), e.pstyle("padding-relative-to").value), n.autoWidth = Math.max(a.w, o.width.val), s.x = (-f + a.x1 + a.x2 + h) / 2, n.autoHeight = Math.max(a.h, o.height.val), s.y = (-m + a.y1 + a.y2 + v) / 2, t.push(e) } function b(e, t, n) { var r = 0, i = 0, o = t + n; return e > 0 && o > 0 && (r = t / o * e, i = n / o * e), { biasDiff: r, biasComplementDiff: i } } } for (var r = 0; r < this.length; r++) { var i = this[r], o = i._private; o.compoundBoundsClean || (n(i), e._private.batchingStyle || (o.compoundBoundsClean = !0)) } return this }; var l = function (e) { return e === 1 / 0 || e === -1 / 0 ? 0 : e }, c = function (e, t, n, r, i) { r - t != 0 && i - n != 0 && null != t && null != n && null != r && null != i && (e.x1 = t < e.x1 ? t : e.x1, e.x2 = r > e.x2 ? r : e.x2, e.y1 = n < e.y1 ? n : e.y1, e.y2 = i > e.y2 ? i : e.y2) }, u = function (e, t, n) { return i.getPrefixedProperty(e, t, n) }, d = function (e, t, n) { if (!t.cy().headless()) { var r = t._private.rstyle, i = r.arrowWidth / 2, o = void 0, a = void 0; "none" !== t.pstyle(n + "-arrow-shape").value && ("source" === n ? (o = r.srcX, a = r.srcY) : "target" === n ? (o = r.tgtX, a = r.tgtY) : (o = r.midX, a = r.midY), c(e, o - i, a - i, o + i, a + i)) } }, p = function (e, t, n) { if (!t.cy().headless()) { var r = void 0; r = n ? n + "-" : ""; var i = t._private, o = i.rstyle; if (t.pstyle(r + "label").strValue) { var a = t.pstyle("text-halign"), s = t.pstyle("text-valign"), l = u(o, "labelWidth", n), d = u(o, "labelHeight", n), p = u(o, "labelX", n), f = u(o, "labelY", n), h = t.pstyle(r + "text-margin-x").pfValue, g = t.pstyle(r + "text-margin-y").pfValue, m = t.isEdge(), v = t.pstyle(r + "text-rotation"), b = t.pstyle("text-outline-width").pfValue, y = t.pstyle("text-border-width").pfValue / 2, x = t.pstyle("text-background-padding").pfValue, w = d + 2 * x, k = l + 2 * x, A = k / 2, E = w / 2, S = void 0, $ = void 0, C = void 0, _ = void 0; if (m) S = p - A, $ = p + A, C = f - E, _ = f + E; else { switch (a.value) { case "left": S = p - k, $ = p; break; case "center": S = p - A, $ = p + A; break; case "right": S = p, $ = p + k }switch (s.value) { case "top": C = f - w, _ = f; break; case "center": C = f - E, _ = f + E; break; case "bottom": C = f, _ = f + w } } var O = m && "autorotate" === v.strValue, T = null != v.pfValue && 0 !== v.pfValue; if (O || T) { var j = O ? u(i.rstyle, "labelAngle", n) : v.pfValue, P = Math.cos(j), D = Math.sin(j), R = function (e, t) { return { x: (e -= p) * P - (t -= f) * D + p, y: e * D + t * P + f } }, I = R(S, C), N = R(S, _), M = R($, C), z = R($, _); S = Math.min(I.x, N.x, M.x, z.x), $ = Math.max(I.x, N.x, M.x, z.x), C = Math.min(I.y, N.y, M.y, z.y), _ = Math.max(I.y, N.y, M.y, z.y) } S += h - Math.max(b, y), $ += h + Math.max(b, y), C += g - Math.max(b, y), _ += g + Math.max(b, y), c(e, S, C, $, _) } return e } }, f = function (e) { return e ? "t" : "f" }, h = function (e) { var t = ""; return t += f(e.incudeNodes), t += f(e.includeEdges), t += f(e.includeLabels), t += f(e.includeOverlays) }, g = function (e, t) { var n = e._private, r = void 0, i = e.cy().headless(), a = t === m ? v : h(t); return t.useCache && !i && n.bbCache && n.bbCache[a] ? r = n.bbCache[a] : (r = function (e, t) { var n = e._private.cy, r = n.styleEnabled(), i = n.headless(), a = { x1: 1 / 0, y1: 1 / 0, x2: -1 / 0, y2: -1 / 0 }, s = e._private, u = r ? e.pstyle("display").value : "element", f = e.isNode(), h = e.isEdge(), g = void 0, m = void 0, v = void 0, b = void 0, y = void 0, x = void 0, w = "none" !== u; if (w) { var k = 0; r && t.includeOverlays && 0 !== e.pstyle("overlay-opacity").value && (k = e.pstyle("overlay-padding").value); var A = 0; if (r && (A = e.pstyle("width").pfValue / 2), f && t.includeNodes) { var E = e.position(); y = E.x, x = E.y; var S = e.outerWidth() / 2, $ = e.outerHeight() / 2; c(a, g = y - S - k, v = x - $ - k, m = y + S + k, b = x + $ + k) } else if (h && t.includeEdges) { var C = s.rstyle || {}; if (r && !i && (g = Math.min(C.srcX, C.midX, C.tgtX), m = Math.max(C.srcX, C.midX, C.tgtX), v = Math.min(C.srcY, C.midY, C.tgtY), b = Math.max(C.srcY, C.midY, C.tgtY), c(a, g -= A, v -= A, m += A, b += A)), r && !i && "haystack" === e.pstyle("curve-style").strValue) { var _ = C.haystackPts || []; if (g = _[0].x, v = _[0].y, g > (m = _[1].x)) { var O = g; g = m, m = O } if (v > (b = _[1].y)) { var T = v; v = b, b = T } c(a, g - A, v - A, m + A, b + A) } else { for (var j = C.bezierPts || C.linePts || [], P = 0; P < j.length; P++) { var D = j[P]; g = D.x - A, m = D.x + A, v = D.y - A, b = D.y + A, c(a, g, v, m, b) } if (0 === j.length) { var R = e.source().position(), I = e.target().position(); if ((g = R.x) > (m = I.x)) { var N = g; g = m, m = N } if ((v = R.y) > (b = I.y)) { var M = v; v = b, b = M } c(a, g -= A, v -= A, m += A, b += A) } } } if (r && t.includeEdges && h && (d(a, e, "mid-source"), d(a, e, "mid-target"), d(a, e, "source"), d(a, e, "target")), r && "yes" === e.pstyle("ghost").value) { var z = e.pstyle("ghost-offset-x").pfValue, L = e.pstyle("ghost-offset-y").pfValue; c(a, a.x1 + z, a.y1 + L, a.x2 + z, a.y2 + L) } r && (g = a.x1, m = a.x2, v = a.y1, b = a.y2, c(a, g - k, v - k, m + k, b + k)), r && t.includeLabels && (p(a, e, null), h && (p(a, e, "source"), p(a, e, "target"))) } return a.x1 = l(a.x1), a.y1 = l(a.y1), a.x2 = l(a.x2), a.y2 = l(a.y2), a.w = l(a.x2 - a.x1), a.h = l(a.y2 - a.y1), a.w > 0 && a.h > 0 && w && o.expandBoundingBox(a, 1), a }(e, t), i || (n.bbCache = n.bbCache || {}, n.bbCache[a] = r)), r }, m = { includeNodes: !0, includeEdges: !0, includeLabels: !0, includeOverlays: !0, useCache: !0 }, v = h(m); function b(e) { return { includeNodes: i.default(e.includeNodes, m.includeNodes), includeEdges: i.default(e.includeEdges, m.includeEdges), includeLabels: i.default(e.includeLabels, m.includeLabels), includeOverlays: i.default(e.includeOverlays, m.includeOverlays), useCache: i.default(e.useCache, m.useCache) } } s.boundingBox = function (e) { if (1 === this.length && this[0]._private.bbCache && (void 0 === e || void 0 === e.useCache || !0 === e.useCache)) return e = void 0 === e ? m : b(e), g(this[0], e); var t = { x1: 1 / 0, y1: 1 / 0, x2: -1 / 0, y2: -1 / 0 }, n = b(e = e || i.staticEmptyObject()), r = this.cy().styleEnabled(); r && this.recalculateRenderedStyle(n.useCache), this.updateCompoundBounds(); for (var o, a, s = {}, u = 0; u < this.length; u++) { var d = this[u]; if (r && d.isEdge() && "bezier" === d.pstyle("curve-style").strValue && !s[d.id()]) { for (var p = d.parallelEdges(), f = 0; f < p.length; f++)s[p[f].id()] = !0; p.recalculateRenderedStyle(n.useCache) } o = t, a = g(d, n), c(o, a.x1, a.y1, a.x2, a.y2) } return t.x1 = l(t.x1), t.y1 = l(t.y1), t.x2 = l(t.x2), t.y2 = l(t.y2), t.w = l(t.x2 - t.x1), t.h = l(t.y2 - t.y1), t }, s.boundingBoxAt = function (e) { var t = this.nodes(); if (r.plainObject(e)) { var n = e; e = function () { return n } } for (var i = 0; i < t.length; i++) { var o = t[i], a = o._private, s = a.position, l = e.call(o, o, i); a.bbAtOldPos = { x: s.x, y: s.y }, l && (s.x = l.x, s.y = l.y) } this.emit("dirty"), t.dirtyCompoundBoundsCache().updateCompoundBounds(); for (var c = this.boundingBox({ useCache: !1 }), u = 0; u < t.length; u++) { var d = t[u], p = d._private, f = d._private.position, h = p.bbAtOldPos; f.x = h.x, f.y = h.y } return t.dirtyCompoundBoundsCache(), this.emit("dirty"), c }, a.boundingbox = a.boundingBox, a.renderedBoundingbox = a.renderedBoundingBox, e.exports = s }, function (e, t, n) { "use strict"; var r = n(1), i = void 0, o = void 0; i = o = {}; var a = function (e) { e.uppercaseName = r.capitalize(e.name), e.autoName = "auto" + e.uppercaseName, e.labelName = "label" + e.uppercaseName, e.outerName = "outer" + e.uppercaseName, e.uppercaseOuterName = r.capitalize(e.outerName), i[e.name] = function () { var t = this[0], n = t._private, r = n.cy._private.styleEnabled; if (t) { if (!r) return 1; if (t.isParent()) return t.updateCompoundBounds(), n[e.autoName] || 0; var i = t.pstyle(e.name); switch (i.strValue) { case "label": return t.recalculateRenderedStyle(), n.rstyle[e.labelName] || 0; default: return i.pfValue } } }, i["outer" + e.uppercaseName] = function () { var t = this[0], n = t._private.cy._private.styleEnabled; if (t) return n ? t[e.name]() + t.pstyle("border-width").pfValue + 2 * t.padding() : 1 }, i["rendered" + e.uppercaseName] = function () { var t = this[0]; if (t) return t[e.name]() * this.cy().zoom() }, i["rendered" + e.uppercaseOuterName] = function () { var t = this[0]; if (t) return t[e.outerName]() * this.cy().zoom() } }; a({ name: "width" }), a({ name: "height" }), o.padding = function () { var e = this[0], t = e._private; return e.isParent() ? (e.updateCompoundBounds(), void 0 !== t.autoPadding ? t.autoPadding : e.pstyle("padding").pfValue) : e.pstyle("padding").pfValue }, e.exports = o }, function (e, t, n) { "use strict"; var r = function (e, t) { if (e.isEdge()) return t(e.renderer()) }; e.exports = { controlPoints: function () { var e = this; return r(this, (function (t) { return t.getControlPoints(e) })) }, segmentPoints: function () { var e = this; return r(this, (function (t) { return t.getSegmentPoints(e) })) }, sourceEndpoint: function () { var e = this; return r(this, (function (t) { return t.getSourceEndpoint(e) })) }, targetEndpoint: function () { var e = this; return r(this, (function (t) { return t.getTargetEndpoint(e) })) }, midpoint: function () { var e = this; return r(this, (function (t) { return t.getEdgeMidpoint(e) })) } } }, function (e, t, n) { "use strict"; var r = n(11), i = n(4), o = n(0), a = n(1), s = n(6), l = { qualifierCompare: function (e, t) { return null == e || null == t ? null == e && null == t : e.sameText(t) }, eventMatches: function (e, t, n) { var r = t.qualifier; return null == r || e !== n.target && o.element(n.target) && r.matches(n.target) }, eventFields: function (e) { return { cy: e.cy(), target: e } }, callbackContext: function (e, t, n) { return null != t.qualifier ? n.target : e }, beforeEmit: function (e, t) { t.conf && t.conf.once && t.conf.onceCollection.removeListener(t.event, t.qualifier, t.callback) }, bubble: function () { return !0 }, parent: function (e) { return e.isChild() ? e.parent() : e.cy() } }, c = function (e) { return o.string(e) ? new s(e) : e }, u = { createEmitter: function () { for (var e = 0; e < this.length; e++) { var t = this[e], n = t._private; n.emitter || (n.emitter = new r(a.assign({ context: t }, l))) } return this }, emitter: function () { return this._private.emitter }, on: function (e, t, n) { for (var r = 0; r < this.length; r++)this[r].emitter().on(e, c(t), n); return this }, removeListener: function (e, t, n) { for (var r = 0; r < this.length; r++)this[r].emitter().removeListener(e, c(t), n); return this }, one: function (e, t, n) { for (var r = 0; r < this.length; r++)this[r].emitter().one(e, c(t), n); return this }, once: function (e, t, n) { for (var r = 0; r < this.length; r++)this[r].emitter().on(e, c(t), n, { once: !0, onceCollection: this }) }, emit: function (e, t) { for (var n = 0; n < this.length; n++)this[n].emitter().emit(e, t); return this }, emitAndNotify: function (e, t) { if (0 !== this.length) return this.cy().notify({ type: e, eles: this }), this.emit(e, t), this } }; i.eventAliasesOn(u), e.exports = u }, function (e, t, n) { "use strict"; var r = n(0), i = n(6), o = { nodes: function (e) { return this.filter((function (e) { return e.isNode() })).filter(e) }, edges: function (e) { return this.filter((function (e) { return e.isEdge() })).filter(e) }, filter: function (e, t) { if (void 0 === e) return this; if (r.string(e) || r.elementOrCollection(e)) return new i(e).filter(this); if (r.fn(e)) { for (var n = this.spawn(), o = 0; o < this.length; o++) { var a = this[o]; (t ? e.apply(t, [a, o, this]) : e(a, o, this)) && n.merge(a) } return n } return this.spawn() }, not: function (e) { if (e) { r.string(e) && (e = this.filter(e)); for (var t = [], n = e._private.map, i = 0; i < this.length; i++) { var o = this[i]; n.has(o.id()) || t.push(o) } return this.spawn(t) } return this }, absoluteComplement: function () { return this.cy().mutableElements().not(this) }, intersect: function (e) { if (r.string(e)) { var t = e; return this.filter(t) } for (var n = [], i = e, o = this.length < e.length, a = o ? i._private.map : this._private.map, s = o ? this : i, l = 0; l < s.length; l++) { var c = s[l]._private.data.id, u = a.get(c); u && n.push(u.ele) } return this.spawn(n) }, xor: function (e) { var t = this._private.cy; r.string(e) && (e = t.$(e)); var n = [], i = e, o = function (e, t) { for (var r = 0; r < e.length; r++) { var i = e[r], o = i._private.data.id; t.hasElementWithId(o) || n.push(i) } }; return o(this, i), o(i, this), this.spawn(n) }, diff: function (e) { var t = this._private.cy; r.string(e) && (e = t.$(e)); var n = [], i = [], o = [], a = e, s = function (e, t, n) { for (var r = 0; r < e.length; r++) { var i = e[r], a = i._private.data.id; t.hasElementWithId(a) ? o.push(i) : n.push(i) } }; return s(this, a, n), s(a, this, i), { left: this.spawn(n, { unique: !0 }), right: this.spawn(i, { unique: !0 }), both: this.spawn(o, { unique: !0 }) } }, add: function (e) { var t = this._private.cy; if (!e) return this; if (r.string(e)) { var n = e; e = t.mutableElements().filter(n) } for (var i = [], o = 0; o < this.length; o++)i.push(this[o]); for (var a = this._private.map, s = 0; s < e.length; s++) { var l = !a.has(e[s].id()); l && i.push(e[s]) } return this.spawn(i) }, merge: function (e) { var t = this._private, n = t.cy; if (!e) return this; if (e && r.string(e)) { var i = e; e = n.mutableElements().filter(i) } for (var o = t.map, a = 0; a < e.length; a++) { var s = e[a], l = s._private.data.id; if (o.has(l)) { var c = o.get(l).index; this[c] = s, o.set(l, { ele: s, index: c }) } else { var u = this.length++; this[u] = s, o.set(l, { ele: s, index: u }) } } return this }, unmergeOne: function (e) { e = e[0]; var t = this._private, n = e._private.data.id, r = t.map, i = r.get(n); if (!i) return this; var o = i.index; this[o] = void 0, r.delete(n); var a = o === this.length - 1; if (this.length > 1 && !a) { var s = this.length - 1, l = this[s], c = l._private.data.id; this[s] = void 0, this[o] = l, r.set(c, { ele: l, index: o }) } return this.length--, this }, unmerge: function (e) { var t = this._private.cy; if (!e) return this; if (e && r.string(e)) { var n = e; e = t.mutableElements().filter(n) } for (var i = 0; i < e.length; i++)this.unmergeOne(e[i]); return this }, map: function (e, t) { for (var n = [], r = 0; r < this.length; r++) { var i = this[r], o = t ? e.apply(t, [i, r, this]) : e(i, r, this); n.push(o) } return n }, reduce: function (e, t) { for (var n = t, r = 0; r < this.length; r++)n = e(n, this[r], r, this); return n }, max: function (e, t) { for (var n = -1 / 0, r = void 0, i = 0; i < this.length; i++) { var o = this[i], a = t ? e.apply(t, [o, i, this]) : e(o, i, this); a > n && (n = a, r = o) } return { value: n, ele: r } }, min: function (e, t) { for (var n = 1 / 0, r = void 0, i = 0; i < this.length; i++) { var o = this[i], a = t ? e.apply(t, [o, i, this]) : e(o, i, this); a < n && (n = a, r = o) } return { value: n, ele: r } } }, a = o; a.u = a["|"] = a["+"] = a.union = a.or = a.add, a["\\"] = a["!"] = a["-"] = a.difference = a.relativeComplement = a.subtract = a.not, a.n = a["&"] = a["."] = a.and = a.intersection = a.intersect, a["^"] = a["(+)"] = a["(-)"] = a.symmetricDifference = a.symdiff = a.xor, a.fnFilter = a.filterFn = a.stdFilter = a.filter, a.complement = a.abscomp = a.absoluteComplement, e.exports = o }, function (e, t, n) { "use strict"; e.exports = { isNode: function () { return "nodes" === this.group() }, isEdge: function () { return "edges" === this.group() }, isLoop: function () { return this.isEdge() && this.source().id() === this.target().id() }, isSimple: function () { return this.isEdge() && this.source().id() !== this.target().id() }, group: function () { var e = this[0]; if (e) return e._private.group } } }, function (e, t, n) { "use strict"; var r = n(0), i = n(17), o = n(1), a = { forEach: function (e, t) { if (r.fn(e)) for (var n = 0; n < this.length; n++) { var i = this[n]; if (!1 === (t ? e.apply(t, [i, n, this]) : e(i, n, this))) break } return this }, toArray: function () { for (var e = [], t = 0; t < this.length; t++)e.push(this[t]); return e }, slice: function (e, t) { var n = [], r = this.length; null == t && (t = r), null == e && (e = 0), e < 0 && (e = r + e), t < 0 && (t = r + t); for (var i = e; i >= 0 && i < t && i < r; i++)n.push(this[i]); return this.spawn(n) }, size: function () { return this.length }, eq: function (e) { return this[e] || this.spawn() }, first: function () { return this[0] || this.spawn() }, last: function () { return this[this.length - 1] || this.spawn() }, empty: function () { return 0 === this.length }, nonempty: function () { return !this.empty() }, sort: function (e) { if (!r.fn(e)) return this; var t = this.toArray().sort(e); return this.spawn(t) }, sortByZIndex: function () { return this.sort(i) }, zDepth: function () { var e = this[0]; if (e) { var t = e._private; if ("nodes" === t.group) { var n = t.data.parent ? e.parents().size() : 0; return e.isParent() ? n : o.MAX_INT - 1 } var r = t.source, i = t.target, a = r.zDepth(), s = i.zDepth(); return Math.max(a, s, 0) } } }; a.each = a.forEach, e.exports = a }, function (e, t, n) { "use strict"; var r = n(0), i = n(1), o = n(5), a = n(2), s = { layoutDimensions: function (e) { if ((e = i.assign({ nodeDimensionsIncludeLabels: !0 }, e)).nodeDimensionsIncludeLabels) { var t = this.boundingBox(); return { w: t.w, h: t.h } } return { w: this.outerWidth(), h: this.outerHeight() } }, layoutPositions: function (e, t, n) { var s = this.nodes(), l = this.cy(), c = t.eles, u = function (e, t) { return e.id() + "$" + t }, d = i.memoize(n, u); e.emit({ type: "layoutstart", layout: e }), e.animations = []; var p = t.spacingFactor && 1 !== t.spacingFactor, f = function () { if (!p) return null; for (var e = a.makeBoundingBox(), t = 0; t < s.length; t++) { var n = s[t], r = d(n, t); a.expandBoundingBoxByPoint(e, r.x, r.y) } return e }(), h = i.memoize((function (e, n) { var i = d(e, n), o = e.position(); return r.number(o.x) && r.number(o.y) || e.silentPosition({ x: 0, y: 0 }), p && (i = function (e, t, n) { var r = t.x1 + t.w / 2, i = t.y1 + t.h / 2; return { x: r + (n.x - r) * e, y: i + (n.y - i) * e } }(Math.abs(t.spacingFactor), f, i)), null != t.transform && (i = t.transform(e, i)), i }), u); if (t.animate) { for (var g = 0; g < s.length; g++) { var m = s[g], v = h(m, g); if (null == t.animateFilter || t.animateFilter(m, g)) { var b = m.animation({ position: v, duration: t.animationDuration, easing: t.animationEasing }); e.animations.push(b), b.play() } else m.position(v) } if (t.fit) { var y = l.animation({ fit: { boundingBox: c.boundingBoxAt(h), padding: t.padding }, duration: t.animationDuration, easing: t.animationEasing }); e.animations.push(y), y.play() } else if (void 0 !== t.zoom && void 0 !== t.pan) { var x = l.animation({ zoom: t.zoom, pan: t.pan, duration: t.animationDuration, easing: t.animationEasing }); e.animations.push(x), x.play() } e.one("layoutready", t.ready), e.emit({ type: "layoutready", layout: e }), o.all(e.animations.map((function (e) { return e.promise() }))).then((function () { e.one("layoutstop", t.stop), e.emit({ type: "layoutstop", layout: e }) })) } else s.positions(h), t.fit && l.fit(t.eles, t.padding), null != t.zoom && l.zoom(t.zoom), t.pan && l.pan(t.pan), e.one("layoutready", t.ready), e.emit({ type: "layoutready", layout: e }), e.one("layoutstop", t.stop), e.emit({ type: "layoutstop", layout: e }); return this }, layout: function (e) { return this.cy().makeLayout(i.extend({}, e, { eles: this })) } }; s.createLayout = s.makeLayout = s.layout, e.exports = s }, function (e, t, n) { "use strict"; var r = n(0); function i(e, t, n) { var r, i = n._private, o = i.styleCache = i.styleCache || {}; return null != (r = o[e]) ? r : r = o[e] = t(n) } function o(e, t) { return function (n) { return i(e, t, n) } } function a(e, t) { var n = function (e) { return t.call(e) }; return function () { var t = this[0]; if (t) return i(e, n, t) } } var s = { recalculateRenderedStyle: function (e) { var t = this.cy(), n = t.renderer(), r = t.styleEnabled(); return n && r && n.recalculateRenderedStyle(this, e), this }, dirtyStyleCache: function () { var e = this.cy(), t = function (e) { return e._private.styleCache = {} }; if (e.hasCompoundNodes()) { var n = void 0; (n = this.spawnSelf().merge(this.descendants()).merge(this.parents())).merge(n.connectedEdges()), n.forEach(t) } else this.forEach((function (e) { t(e), e.connectedEdges().forEach(t) })); return this }, updateStyle: function (e) { var t = this._private.cy; if (!t.styleEnabled()) return this; if (t._private.batchingStyle) return t._private.batchStyleEles.merge(this), this; var n = t.hasCompoundNodes(), r = t.style(), i = this; e = !(!e && void 0 !== e), n && (i = this.spawnSelf().merge(this.descendants()).merge(this.parents())); var o = r.apply(i); return o.dirtyStyleCache(), o.dirtyCompoundBoundsCache(), e ? o.emitAndNotify("style") : o.emit("style"), this }, updateMappers: function (e) { var t = this._private.cy, n = t.style(); if (e = !(!e && void 0 !== e), !t.styleEnabled()) return this; var r = n.updateMappers(this); return r.dirtyStyleCache(), r.dirtyCompoundBoundsCache(), e ? r.emitAndNotify("style") : r.emit("style"), this }, parsedStyle: function (e) { var t = this[0], n = t.cy(); if (n.styleEnabled()) return t ? t._private.style[e] || n.style().getDefaultProperty(e) : void 0 }, numericStyle: function (e) { var t = this[0]; if (t.cy().styleEnabled() && t) { var n = t.pstyle(e); return void 0 !== n.pfValue ? n.pfValue : n.value } }, numericStyleUnits: function (e) { var t = this[0]; if (t.cy().styleEnabled()) return t ? t.pstyle(e).units : void 0 }, renderedStyle: function (e) { var t = this.cy(); if (!t.styleEnabled()) return this; var n = this[0]; return n ? t.style().getRenderedStyle(n, e) : void 0 }, style: function (e, t) { var n = this.cy(); if (!n.styleEnabled()) return this; var i = n.style(); if (r.plainObject(e)) { var o = e; i.applyBypass(this, o, !1), this.dirtyStyleCache(), this.dirtyCompoundBoundsCache(), this.emitAndNotify("style") } else if (r.string(e)) { if (void 0 === t) { var a = this[0]; return a ? i.getStylePropertyValue(a, e) : void 0 } i.applyBypass(this, e, t, !1), this.dirtyStyleCache(), this.dirtyCompoundBoundsCache(), this.emitAndNotify("style") } else if (void 0 === e) { var s = this[0]; return s ? i.getRawStyle(s) : void 0 } return this }, removeStyle: function (e) { var t = this.cy(); if (!t.styleEnabled()) return this; var n = t.style(); if (void 0 === e) for (var r = 0; r < this.length; r++) { var i = this[r]; n.removeAllBypasses(i, !1) } else { e = e.split(/\s+/); for (var o = 0; o < this.length; o++) { var a = this[o]; n.removeBypasses(a, e, !1) } } return this.dirtyStyleCache(), this.dirtyCompoundBoundsCache(), this.emitAndNotify("style"), this }, show: function () { return this.css("display", "element"), this }, hide: function () { return this.css("display", "none"), this }, effectiveOpacity: function () { var e = this.cy(); if (!e.styleEnabled()) return 1; var t = e.hasCompoundNodes(), n = this[0]; if (n) { var r = n._private, i = n.pstyle("opacity").value; if (!t) return i; var o = r.data.parent ? n.parents() : null; if (o) for (var a = 0; a < o.length; a++)i *= o[a].pstyle("opacity").value; return i } }, transparent: function () { if (!this.cy().styleEnabled()) return !1; var e = this[0], t = e.cy().hasCompoundNodes(); return e ? t ? 0 === e.effectiveOpacity() : 0 === e.pstyle("opacity").value : void 0 }, backgrounding: function () { return !!this.cy().styleEnabled() && !!this[0]._private.backgrounding } }; function l(e, t) { var n = e._private.data.parent ? e.parents() : null; if (n) for (var r = 0; r < n.length; r++)if (!t(n[r])) return !1; return !0 } function c(e) { var t = e.ok, n = e.edgeOkViaNode || e.ok, r = e.parentOk || e.ok; return function () { var e = this.cy(); if (!e.styleEnabled()) return !0; var i = this[0], o = e.hasCompoundNodes(); if (i) { var a = i._private; if (!t(i)) return !1; if (i.isNode()) return !o || l(i, r); var s = a.source, c = a.target; return n(s) && (!o || l(s, n)) && (s === c || n(c) && (!o || l(c, n))) } } } var u = o("eleTakesUpSpace", (function (e) { return "element" === e.pstyle("display").value && 0 !== e.width() && (!e.isNode() || 0 !== e.height()) })); s.takesUpSpace = a("takesUpSpace", c({ ok: u })); var d = o("eleInteractive", (function (e) { return "yes" === e.pstyle("events").value && "visible" === e.pstyle("visibility").value && u(e) })), p = o("parentInteractive", (function (e) { return "visible" === e.pstyle("visibility").value && u(e) })); s.interactive = a("interactive", c({ ok: d, parentOk: p, edgeOkViaNode: u })), s.noninteractive = function () { var e = this[0]; if (e) return !e.interactive() }; var f = o("eleVisible", (function (e) { return "visible" === e.pstyle("visibility").value && 0 !== e.pstyle("opacity").pfValue && u(e) })), h = u; s.visible = a("visible", c({ ok: f, edgeOkViaNode: h })), s.hidden = function () { var e = this[0]; if (e) return !e.visible() }, s.bypass = s.css = s.style, s.renderedCss = s.renderedStyle, s.removeBypass = s.removeCss = s.removeStyle, s.pstyle = s.parsedStyle, e.exports = s }, function (e, t, n) { "use strict"; var r = {}; function i(e) { return function () { var t = arguments, n = []; if (2 === t.length) { var r = t[0], i = t[1]; this.on(e.event, r, i) } else if (1 === t.length) { var o = t[0]; this.on(e.event, o) } else if (0 === t.length) { for (var a = 0; a < this.length; a++) { var s = this[a], l = !e.ableField || s._private[e.ableField], c = s._private[e.field] != e.value; if (e.overrideAble) { var u = e.overrideAble(s); if (void 0 !== u && (l = u, !u)) return this } l && (s._private[e.field] = e.value, c && n.push(s)) } var d = this.spawn(n); d.updateStyle(), d.emit(e.event) } return this } } function o(e) { r[e.field] = function () { var t = this[0]; if (t) { if (e.overrideField) { var n = e.overrideField(t); if (void 0 !== n) return n } return t._private[e.field] } }, r[e.on] = i({ event: e.on, field: e.field, ableField: e.ableField, overrideAble: e.overrideAble, value: !0 }), r[e.off] = i({ event: e.off, field: e.field, ableField: e.ableField, overrideAble: e.overrideAble, value: !1 }) } o({ field: "locked", overrideField: function (e) { return !!e.cy().autolock() || void 0 }, on: "lock", off: "unlock" }), o({ field: "grabbable", overrideField: function (e) { return !e.cy().autoungrabify() && void 0 }, on: "grabify", off: "ungrabify" }), o({ field: "selected", ableField: "selectable", overrideAble: function (e) { return !e.cy().autounselectify() && void 0 }, on: "select", off: "unselect" }), o({ field: "selectable", overrideField: function (e) { return !e.cy().autounselectify() && void 0 }, on: "selectify", off: "unselectify" }), r.deselect = r.unselect, r.grabbed = function () { var e = this[0]; if (e) return e._private.grabbed }, o({ field: "active", on: "activate", off: "unactivate" }), r.inactive = function () { var e = this[0]; if (e) return !e._private.active }, e.exports = r }, function (e, t, n) { "use strict"; var r = n(1), i = n(0), o = {}, a = function (e, t) { return function (n, r, o, a) { var s = n, l = void 0; if (null == s ? l = "null" : i.elementOrCollection(s) && 1 === s.length && (l = "#" + s.id()), 1 === this.length && l) { var c = this[0]._private, u = c.traversalCache = c.traversalCache || {}, d = u[t] = u[t] || {}, p = d[l]; return p || (d[l] = e.call(this, n, r, o, a)) } return e.call(this, n, r, o, a) } }, s = function (e) { return function (t) { for (var n = [], r = 0; r < this.length; r++) { var i = this[r]; if (i.isNode()) { for (var o = !1, a = i.connectedEdges(), s = 0; s < a.length; s++) { var l = a[s], c = l.source(), u = l.target(); if (e.noIncomingEdges && u === i && c !== i || e.noOutgoingEdges && c === i && u !== i) { o = !0; break } } o || n.push(i) } } return this.spawn(n, { unique: !0 }).filter(t) } }, l = function (e) { return function (t) { for (var n = [], r = 0; r < this.length; r++) { var i = this[r]; if (i.isNode()) for (var o = i.connectedEdges(), a = 0; a < o.length; a++) { var s = o[a], l = s.source(), c = s.target(); e.outgoing && l === i ? (n.push(s), n.push(c)) : e.incoming && c === i && (n.push(s), n.push(l)) } } return this.spawn(n, { unique: !0 }).filter(t) } }, c = function (e) { return function (t) { for (var n = this, r = [], i = {}; ;) { var o = e.outgoing ? n.outgoers() : n.incomers(); if (0 === o.length) break; for (var a = !1, s = 0; s < o.length; s++) { var l = o[s], c = l.id(); i[c] || (i[c] = !0, r.push(l), a = !0) } if (!a) break; n = o } return this.spawn(r, { unique: !0 }).filter(t) } }; function u(e) { return function (t) { for (var n = [], r = 0; r < this.length; r++) { var i = this[r]._private[e.attr]; i && n.push(i) } return this.spawn(n, { unique: !0 }).filter(t) } } function d(e) { return function (t) { var n = [], r = this._private.cy, o = e || {}; i.string(t) && (t = r.$(t)); for (var a = 0; a < t.length; a++)for (var s = t[a]._private.edges, l = 0; l < s.length; l++) { var c = s[l], u = c._private.data, d = this.hasElementWithId(u.source) && t.hasElementWithId(u.target), p = t.hasElementWithId(u.source) && this.hasElementWithId(u.target); if (d || p) { if (o.thisIsSrc || o.thisIsTgt) { if (o.thisIsSrc && !d) continue; if (o.thisIsTgt && !p) continue } n.push(c) } } return this.spawn(n, { unique: !0 }) } } function p(e) { return e = r.extend({}, { codirected: !1 }, e), function (t) { for (var n = [], r = this.edges(), i = e, o = 0; o < r.length; o++)for (var a = r[o]._private, s = a.source, l = s._private.data.id, c = a.data.target, u = s._private.edges, d = 0; d < u.length; d++) { var p = u[d], f = p._private.data, h = f.target, g = f.source, m = h === c && g === l, v = l === h && c === g; (i.codirected && m || !i.codirected && (m || v)) && n.push(p) } return this.spawn(n, { unique: !0 }).filter(t) } } o.clearTraversalCache = function () { for (var e = 0; e < this.length; e++)this[e]._private.traversalCache = null }, r.extend(o, { roots: s({ noIncomingEdges: !0 }), leaves: s({ noOutgoingEdges: !0 }), outgoers: a(l({ outgoing: !0 }), "outgoers"), successors: c({ outgoing: !0 }), incomers: a(l({ incoming: !0 }), "incomers"), predecessors: c({ incoming: !0 }) }), r.extend(o, { neighborhood: a((function (e) { for (var t = [], n = this.nodes(), r = 0; r < n.length; r++)for (var i = n[r], o = i.connectedEdges(), a = 0; a < o.length; a++) { var s = o[a], l = s.source(), c = s.target(), u = i === l ? c : l; u.length > 0 && t.push(u[0]), t.push(s[0]) } return this.spawn(t, { unique: !0 }).filter(e) }), "neighborhood"), closedNeighborhood: function (e) { return this.neighborhood().add(this).filter(e) }, openNeighborhood: function (e) { return this.neighborhood(e) } }), o.neighbourhood = o.neighborhood, o.closedNeighbourhood = o.closedNeighborhood, o.openNeighbourhood = o.openNeighborhood, r.extend(o, { source: a((function (e) { var t = this[0], n = void 0; return t && (n = t._private.source || t.cy().collection()), n && e ? n.filter(e) : n }), "source"), target: a((function (e) { var t = this[0], n = void 0; return t && (n = t._private.target || t.cy().collection()), n && e ? n.filter(e) : n }), "target"), sources: u({ attr: "source" }), targets: u({ attr: "target" }) }), r.extend(o, { edgesWith: a(d(), "edgesWith"), edgesTo: a(d({ thisIsSrc: !0 }), "edgesTo") }), r.extend(o, { connectedEdges: a((function (e) { for (var t = [], n = 0; n < this.length; n++) { var r = this[n]; if (r.isNode()) for (var i = r._private.edges, o = 0; o < i.length; o++) { var a = i[o]; t.push(a) } } return this.spawn(t, { unique: !0 }).filter(e) }), "connectedEdges"), connectedNodes: a((function (e) { for (var t = [], n = 0; n < this.length; n++) { var r = this[n]; r.isEdge() && (t.push(r.source()[0]), t.push(r.target()[0])) } return this.spawn(t, { unique: !0 }).filter(e) }), "connectedNodes"), parallelEdges: a(p(), "parallelEdges"), codirectedEdges: a(p({ codirected: !0 }), "codirectedEdges") }), r.extend(o, { components: function () { var e = this, t = e.cy(), n = e.spawn(), r = e.nodes().spawnSelf(), i = [], o = function (e, t) { n.merge(e), r.unmerge(e), t.merge(e) }; if (r.empty()) return e.spawn(); var a = function () { var n = t.collection(); i.push(n); var a = r[0]; o(a, n), e.bfs({ directed: !1, roots: a, visit: function (e, t, r, i, a) { o(e, n) } }) }; do { a() } while (r.length > 0); return i.map((function (e) { var t = e.connectedEdges().stdFilter((function (t) { return e.anySame(t.source()) && e.anySame(t.target()) })); return e.union(t) })) } }), e.exports = o }, function (e, t, n) { "use strict"; var r = n(0), i = n(1), o = n(7), a = n(14), s = { add: function (e) { var t = void 0, n = this; if (r.elementOrCollection(e)) { var s = e; if (s._private.cy === n) t = s.restore(); else { for (var l = [], c = 0; c < s.length; c++) { var u = s[c]; l.push(u.json()) } t = new o(n, l) } } else if (r.array(e)) t = new o(n, e); else if (r.plainObject(e) && (r.array(e.nodes) || r.array(e.edges))) { for (var d = e, p = [], f = ["nodes", "edges"], h = 0, g = f.length; h < g; h++) { var m = f[h], v = d[m]; if (r.array(v)) for (var b = 0, y = v.length; b < y; b++) { var x = i.extend({ group: m }, v[b]); p.push(x) } } t = new o(n, p) } else t = new a(n, e).collection(); return t }, remove: function (e) { if (r.elementOrCollection(e)); else if (r.string(e)) { var t = e; e = this.$(t) } return e.remove() } }; e.exports = s }, function (e, t, n) { "use strict"; var r = n(4), i = n(1), o = n(71), a = { animate: r.animate(), animation: r.animation(), animated: r.animated(), clearQueue: r.clearQueue(), delay: r.delay(), delayAnimation: r.delayAnimation(), stop: r.stop(), addToAnimationPool: function (e) { this.styleEnabled() && this._private.aniEles.merge(e) }, stopAnimationLoop: function () { this._private.animationsRunning = !1 }, startAnimationLoop: function () { var e = this; if (e._private.animationsRunning = !0, e.styleEnabled()) { var t = e.renderer(); t && t.beforeRender ? t.beforeRender((function (t, n) { o(n, e) }), t.beforeRenderPriorities.animations) : function t() { e._private.animationsRunning && i.requestAnimationFrame((function (n) { o(n, e), t() })) }() } } }; e.exports = a }, function (e, t, n) { "use strict"; var r = n(72), i = n(77); e.exports = function (e, t) { var n = t._private.aniEles, o = []; function a(t, n) { var a = t._private, s = a.animation.current, l = a.animation.queue, c = !1; if (!n && "none" === t.pstyle("display").value) { s = s.splice(0, s.length).concat(l.splice(0, l.length)); for (var u = 0; u < s.length; u++)s[u].stop() } if (0 === s.length) { var d = l.shift(); d && s.push(d) } for (var p = function (e) { for (var t = e.length - 1; t >= 0; t--)(0, e[t])(); e.splice(0, e.length) }, f = s.length - 1; f >= 0; f--) { var h = s[f], g = h._private; g.stopped ? (s.splice(f, 1), g.hooked = !1, g.playing = !1, g.started = !1, p(g.frames)) : (g.playing || g.applying) && (g.playing && g.applying && (g.applying = !1), g.started || i(t, h, e, n), r(t, h, e, n), g.applying && (g.applying = !1), p(g.frames), h.completed() && (s.splice(f, 1), g.hooked = !1, g.playing = !1, g.started = !1, p(g.completes)), c = !0) } return n || 0 !== s.length || 0 !== l.length || o.push(t), c } for (var s = !1, l = 0; l < n.length; l++) { var c = a(n[l]); s = s || c } var u = a(t, !0); (s || u) && (n.length > 0 ? (n.dirtyCompoundBoundsCache(), t.notify({ type: "draw", eles: n })) : t.notify({ type: "draw" })), n.unmerge(o), t.emit("step") } }, function (e, t, n) { "use strict"; var r = n(73), i = n(76), o = n(0); function a(e, t) { return !!(null != e && null != t && (o.number(e) && o.number(t) || e && t)) } e.exports = function (e, t, n, s) { var l = !s, c = e._private, u = t._private, d = u.easing, p = u.startTime, f = (s ? e : e.cy()).style(); if (!u.easingImpl) if (null == d) u.easingImpl = r.linear; else { var h = void 0; h = o.string(d) ? f.parse("transition-timing-function", d).value : d; var g = void 0, m = void 0; o.string(h) ? (g = h, m = []) : (g = h[1], m = h.slice(2).map((function (e) { return +e }))), m.length > 0 ? ("spring" === g && m.push(u.duration), u.easingImpl = r[g].apply(null, m)) : u.easingImpl = r[g] } var v = u.easingImpl, b = void 0; if (b = 0 === u.duration ? 1 : (n - p) / u.duration, u.applying && (b = u.progress), b < 0 ? b = 0 : b > 1 && (b = 1), null == u.delay) { var y = u.startPosition, x = u.position; if (x && l && !e.locked()) { var w = e.position(); a(y.x, x.x) && (w.x = i(y.x, x.x, b, v)), a(y.y, x.y) && (w.y = i(y.y, x.y, b, v)), e.emit("position") } var k = u.startPan, A = u.pan, E = c.pan, S = null != A && s; S && (a(k.x, A.x) && (E.x = i(k.x, A.x, b, v)), a(k.y, A.y) && (E.y = i(k.y, A.y, b, v)), e.emit("pan")); var $ = u.startZoom, C = u.zoom, _ = null != C && s; _ && (a($, C) && (c.zoom = i($, C, b, v)), e.emit("zoom")), (S || _) && e.emit("viewport"); var O = u.style; if (O && O.length > 0 && l) { for (var T = 0; T < O.length; T++) { var j = O[T], P = j.name, D = j, R = u.startStyle[P], I = f.properties[R.name], N = i(R, D, b, v, I); f.overrideBypass(e, P, N) } e.emit("style") } } return u.progress = b, b } }, function (e, t, n) { "use strict"; var r = n(74), i = n(75), o = function (e, t, n, i) { var o = r(e, t, n, i); return function (e, t, n) { return e + (t - e) * o(n) } }, a = { linear: function (e, t, n) { return e + (t - e) * n }, ease: o(.25, .1, .25, 1), "ease-in": o(.42, 0, 1, 1), "ease-out": o(0, 0, .58, 1), "ease-in-out": o(.42, 0, .58, 1), "ease-in-sine": o(.47, 0, .745, .715), "ease-out-sine": o(.39, .575, .565, 1), "ease-in-out-sine": o(.445, .05, .55, .95), "ease-in-quad": o(.55, .085, .68, .53), "ease-out-quad": o(.25, .46, .45, .94), "ease-in-out-quad": o(.455, .03, .515, .955), "ease-in-cubic": o(.55, .055, .675, .19), "ease-out-cubic": o(.215, .61, .355, 1), "ease-in-out-cubic": o(.645, .045, .355, 1), "ease-in-quart": o(.895, .03, .685, .22), "ease-out-quart": o(.165, .84, .44, 1), "ease-in-out-quart": o(.77, 0, .175, 1), "ease-in-quint": o(.755, .05, .855, .06), "ease-out-quint": o(.23, 1, .32, 1), "ease-in-out-quint": o(.86, 0, .07, 1), "ease-in-expo": o(.95, .05, .795, .035), "ease-out-expo": o(.19, 1, .22, 1), "ease-in-out-expo": o(1, 0, 0, 1), "ease-in-circ": o(.6, .04, .98, .335), "ease-out-circ": o(.075, .82, .165, 1), "ease-in-out-circ": o(.785, .135, .15, .86), spring: function (e, t, n) { if (0 === n) return a.linear; var r = i(e, t, n); return function (e, t, n) { return e + (t - e) * r(n) } }, "cubic-bezier": o }; e.exports = a }, function (e, t, n) { + "use strict"; +/*! Bezier curve function generator. Copyright Gaetan Renaudeau. MIT License: http://en.wikipedia.org/wiki/MIT_License */e.exports = function (e, t, n, r) { var i = 4, o = .001, a = 1e-7, s = 10, l = 11, c = 1 / (l - 1), u = "undefined" != typeof Float32Array; if (4 !== arguments.length) return !1; for (var d = 0; d < 4; ++d)if ("number" != typeof arguments[d] || isNaN(arguments[d]) || !isFinite(arguments[d])) return !1; e = Math.min(e, 1), n = Math.min(n, 1), e = Math.max(e, 0), n = Math.max(n, 0); var p = u ? new Float32Array(l) : new Array(l); function f(e, t) { return 1 - 3 * t + 3 * e } function h(e, t) { return 3 * t - 6 * e } function g(e) { return 3 * e } function m(e, t, n) { return ((f(t, n) * e + h(t, n)) * e + g(t)) * e } function v(e, t, n) { return 3 * f(t, n) * e * e + 2 * h(t, n) * e + g(t) } function b(t, r) { for (var o = 0; o < i; ++o) { var a = v(r, e, n); if (0 === a) return r; r -= (m(r, e, n) - t) / a } return r } function y() { for (var t = 0; t < l; ++t)p[t] = m(t * c, e, n) } function x(t, r, i) { var o = void 0, l = void 0, c = 0; do { (o = m(l = r + (i - r) / 2, e, n) - t) > 0 ? i = l : r = l } while (Math.abs(o) > a && ++c < s); return l } function w(t) { for (var r = 0, i = 1, a = l - 1; i !== a && p[i] <= t; ++i)r += c; --i; var s = r + (t - p[i]) / (p[i + 1] - p[i]) * c, u = v(s, e, n); return u >= o ? b(t, s) : 0 === u ? s : x(t, r, r + c) } var k = !1; function A() { k = !0, e === t && n === r || y() } var E = function (i) { return k || A(), e === t && n === r ? i : 0 === i ? 0 : 1 === i ? 1 : m(w(i), t, r) }; E.getControlPoints = function () { return [{ x: e, y: t }, { x: n, y: r }] }; var S = "generateBezier(" + [e, t, n, r] + ")"; return E.toString = function () { return S }, E } + }, function (e, t, n) { + "use strict"; +/*! Runge-Kutta spring physics function generator. Adapted from Framer.js, copyright Koen Bok. MIT License: http://en.wikipedia.org/wiki/MIT_License */var r = function () { function e(e) { return -e.tension * e.x - e.friction * e.v } function t(t, n, r) { var i = { x: t.x + r.dx * n, v: t.v + r.dv * n, tension: t.tension, friction: t.friction }; return { dx: i.v, dv: e(i) } } function n(n, r) { var i = { dx: n.v, dv: e(n) }, o = t(n, .5 * r, i), a = t(n, .5 * r, o), s = t(n, r, a), l = 1 / 6 * (i.dx + 2 * (o.dx + a.dx) + s.dx), c = 1 / 6 * (i.dv + 2 * (o.dv + a.dv) + s.dv); return n.x = n.x + l * r, n.v = n.v + c * r, n } return function e(t, r, i) { var o, a = { x: -1, v: 0, tension: null, friction: null }, s = [0], l = 0, c = void 0, u = void 0; for (t = parseFloat(t) || 500, r = parseFloat(r) || 20, i = i || null, a.tension = t, a.friction = r, c = (o = null !== i) ? (l = e(t, r)) / i * .016 : .016; u = n(u || a, c), s.push(1 + u.x), l += 16, Math.abs(u.x) > 1e-4 && Math.abs(u.v) > 1e-4;); return o ? function (e) { return s[e * (s.length - 1) | 0] } : l } }(); e.exports = r + }, function (e, t, n) { "use strict"; var r = n(0); function i(e, t, n, r, i) { if (1 === r) return n; var o = i(t, n, r); return null == e || ((e.roundValue || e.color) && (o = Math.round(o)), void 0 !== e.min && (o = Math.max(o, e.min)), void 0 !== e.max && (o = Math.min(o, e.max))), o } function o(e, t) { return null != e.pfValue || null != e.value ? null == e.pfValue || null != t && "%" === t.type.units ? e.value : e.pfValue : e } e.exports = function (e, t, n, a, s) { var l = null != s ? s.type : null; n < 0 ? n = 0 : n > 1 && (n = 1); var c = o(e, s), u = o(t, s); if (r.number(c) && r.number(u)) return i(l, c, u, n, a); if (r.array(c) && r.array(u)) { for (var d = [], p = 0; p < u.length; p++) { var f = c[p], h = u[p]; if (null != f && null != h) { var g = i(l, f, h, n, a); d.push(g) } else d.push(h) } return d } } }, function (e, t, n) { "use strict"; e.exports = function (e, t, n, r) { var i = !r, o = e, a = t._private, s = r ? e : e.cy(), l = s.style(); if (i) { var c = o.position(); a.startPosition = a.startPosition || { x: c.x, y: c.y }, a.startStyle = a.startStyle || l.getAnimationStartStyle(o, a.style) } if (r) { var u = s._private.pan; a.startPan = a.startPan || { x: u.x, y: u.y }, a.startZoom = null != a.startZoom ? a.startZoom : s._private.zoom } a.started = !0, a.startTime = n - a.progress * a.duration } }, function (e, t, n) { "use strict"; var r = n(11), i = n(4), o = n(0), a = n(1), s = n(6), l = { qualifierCompare: function (e, t) { return null == e || null == t ? null == e && null == t : e.sameText(t) }, eventMatches: function (e, t, n) { var r = t.qualifier; return null == r || e !== n.target && o.element(n.target) && r.matches(n.target) }, eventFields: function (e) { return { cy: e, target: e } }, callbackContext: function (e, t, n) { return null != t.qualifier ? n.target : e } }, c = function (e) { return o.string(e) ? new s(e) : e }, u = { createEmitter: function () { var e = this._private; return e.emitter || (e.emitter = new r(a.assign({ context: this }, l))), this }, emitter: function () { return this._private.emitter }, on: function (e, t, n) { return this.emitter().on(e, c(t), n), this }, removeListener: function (e, t, n) { return this.emitter().removeListener(e, c(t), n), this }, one: function (e, t, n) { return this.emitter().one(e, c(t), n), this }, once: function (e, t, n) { return this.emitter().one(e, c(t), n), this }, emit: function (e, t) { return this.emitter().emit(e, t), this } }; i.eventAliasesOn(u), e.exports = u }, function (e, t, n) { "use strict"; var r = { png: function (e) { return e = e || {}, this._private.renderer.png(e) }, jpg: function (e) { var t = this._private.renderer; return (e = e || {}).bg = e.bg || "#fff", t.jpg(e) } }; r.jpeg = r.jpg, e.exports = r }, function (e, t, n) { "use strict"; var r = n(1), i = n(0), o = { layout: function (e) { if (null != e) if (null != e.name) { var t = e.name, n = this.extension("layout", t); if (null != n) { var o = void 0; o = i.string(e.eles) ? this.$(e.eles) : null != e.eles ? e.eles : this.$(); var a = new n(r.extend({}, e, { cy: this, eles: o })); return a } r.error("Can not apply layout: No such layout `" + t + "` found; did you include its JS file?") } else r.error("A `name` must be specified to make a layout"); else r.error("Layout options must be specified to make a layout") } }; o.createLayout = o.makeLayout = o.layout, e.exports = o }, function (e, t, n) { "use strict"; var r = { notify: function (e) { var t = this._private; if (t.batchingNotify) { var n = t.batchNotifyEles, r = t.batchNotifyTypes; return e.eles && n.merge(e.eles), void (r.ids[e.type] || (r.push(e.type), r.ids[e.type] = !0)) } if (t.notificationsEnabled) { var i = this.renderer(); !this.isDestroyed() && i && i.notify(e) } }, notifications: function (e) { var t = this._private; if (void 0 === e) return t.notificationsEnabled; t.notificationsEnabled = !!e }, noNotifications: function (e) { this.notifications(!1), e(), this.notifications(!0) }, batching: function () { return this._private.batchCount > 0 }, startBatch: function () { var e = this._private; return null == e.batchCount && (e.batchCount = 0), 0 === e.batchCount && (e.batchingStyle = e.batchingNotify = !0, e.batchStyleEles = this.collection(), e.batchNotifyEles = this.collection(), e.batchNotifyTypes = [], e.batchNotifyTypes.ids = {}), e.batchCount++, this }, endBatch: function () { var e = this._private; return e.batchCount--, 0 === e.batchCount && (e.batchingStyle = !1, e.batchStyleEles.updateStyle(), e.batchingNotify = !1, this.notify({ type: e.batchNotifyTypes, eles: e.batchNotifyEles })), this }, batch: function (e) { return this.startBatch(), e(), this.endBatch(), this }, batchData: function (e) { var t = this; return this.batch((function () { for (var n = Object.keys(e), r = 0; r < n.length; r++) { var i = n[r], o = e[i]; t.getElementById(i).data(o) } })) } }; e.exports = r }, function (e, t, n) { "use strict"; var r = n(1), i = { renderTo: function (e, t, n, r) { return this._private.renderer.renderTo(e, t, n, r), this }, renderer: function () { return this._private.renderer }, forceRender: function () { return this.notify({ type: "draw" }), this }, resize: function () { return this.invalidateSize(), this.notify({ type: "resize" }), this.emit("resize"), this }, initRenderer: function (e) { var t = this.extension("renderer", e.name); null != t ? (this._private.renderer = new t(r.extend({}, e, { cy: this })), this.notify({ type: "init" })) : r.error("Can not initialise: No such renderer `%s` found; did you include its JS file?", e.name) }, destroyRenderer: function () { this.notify({ type: "destroy" }); var e = this.container(); if (e) for (e._cyreg = null; e.childNodes.length > 0;)e.removeChild(e.childNodes[0]); this._private.renderer = null }, onRender: function (e) { return this.on("render", e) }, offRender: function (e) { return this.off("render", e) } }; i.invalidateDimensions = i.resize, e.exports = i }, function (e, t, n) { "use strict"; var r = n(0), i = n(7), o = { collection: function (e, t) { return r.string(e) ? this.$(e) : r.elementOrCollection(e) ? e.collection() : r.array(e) ? new i(this, e, t) : new i(this) }, nodes: function (e) { var t = this.$((function (e) { return e.isNode() })); return e ? t.filter(e) : t }, edges: function (e) { var t = this.$((function (e) { return e.isEdge() })); return e ? t.filter(e) : t }, $: function (e) { var t = this._private.elements; return e ? t.filter(e) : t.spawnSelf() }, mutableElements: function () { return this._private.elements } }; o.elements = o.filter = o.$, e.exports = o }, function (e, t, n) { "use strict"; var r = n(0), i = n(18), o = { style: function (e) { return e && this.setStyle(e).update(), this._private.style }, setStyle: function (e) { var t = this._private; return r.stylesheet(e) ? t.style = e.generateStyle(this) : r.array(e) ? t.style = i.fromJson(this, e) : r.string(e) ? t.style = i.fromString(this, e) : t.style = i(this), t.style } }; e.exports = o }, function (e, t, n) { "use strict"; var r = n(1), i = n(0), o = n(5), a = { apply: function (e) { var t = this._private, n = t.cy.collection(); t.newStyle && (t.contextStyles = {}, t.propDiffs = {}, this.cleanElements(e, !0)); for (var r = 0; r < e.length; r++) { var i = e[r], o = this.getContextMeta(i); if (!o.empty) { n.merge(i); var a = this.getContextStyle(o), s = this.applyContextStyle(o, a, i); t.newStyle || this.updateTransitions(i, s.diffProps), this.updateStyleHints(i) } } return t.newStyle = !1, n }, getPropertiesDiff: function (e, t) { var n = this._private.propDiffs = this._private.propDiffs || {}, r = e + "-" + t, i = n[r]; if (i) return i; for (var o = [], a = {}, s = 0; s < this.length; s++) { var l = this[s], c = "t" === e[s] != ("t" === t[s]), u = l.mappedProperties.length > 0; if (c || u) { var d = void 0; c && u || c ? d = l.properties : u && (d = l.mappedProperties); for (var p = 0; p < d.length; p++) { for (var f = d[p], h = f.name, g = !1, m = s + 1; m < this.length; m++) { var v = this[m]; if ("t" === t[m] && (g = null != v.properties[f.name])) break } a[h] || g || (a[h] = !0, o.push(h)) } } } return n[r] = o, o }, getContextMeta: function (e) { var t, n = "", r = e._private.styleCxtKey || ""; this._private.newStyle && (r = ""); for (var i = 0; i < this.length; i++) { var o = this[i]; n += o.selector && o.selector.matches(e) ? "t" : "f" } return t = this.getPropertiesDiff(r, n), e._private.styleCxtKey = n, { key: n, diffPropNames: t, empty: 0 === t.length } }, getContextStyle: function (e) { var t = e.key, n = this._private.contextStyles = this._private.contextStyles || {}; if (n[t]) return n[t]; for (var r = { _private: { key: t } }, i = 0; i < this.length; i++) { var o = this[i]; if ("t" === t[i]) for (var a = 0; a < o.properties.length; a++) { var s = o.properties[a]; r[s.name] = s } } return n[t] = r, r }, applyContextStyle: function (e, t, n) { for (var r = e.diffPropNames, i = {}, o = 0; o < r.length; o++) { var a = r[o], s = t[a], l = n.pstyle(a); if (!s) { if (!l) continue; s = l.bypass ? { name: a, deleteBypassed: !0 } : { name: a, delete: !0 } } if (l !== s) { var c = i[a] = { prev: l }; this.applyParsedProperty(n, s), c.next = n.pstyle(a), c.next && c.next.bypass && (c.next = c.next.bypassed) } } return { diffProps: i } }, updateStyleHints: function (e) { var t = e._private; if (!e.removed()) { var n = !1; if ("nodes" === t.group) for (var r = 1; r <= this.pieBackgroundN; r++)if (e.pstyle("pie-" + r + "-background-size").value > 0) { n = !0; break } t.hasPie = n; var i = e.pstyle("text-transform").strValue, o = e.pstyle("label").strValue, a = e.pstyle("source-label").strValue, s = e.pstyle("target-label").strValue, l = e.pstyle("font-style").strValue, c = e.pstyle("font-size").pfValue + "px", u = e.pstyle("font-family").strValue, d = e.pstyle("font-weight").strValue, p = l + "$" + c + "$" + u + "$" + d + "$" + i + "$" + e.pstyle("text-valign").strValue + "$" + e.pstyle("text-valign").strValue + "$" + e.pstyle("text-outline-width").pfValue + "$" + e.pstyle("text-wrap").strValue + "$" + e.pstyle("text-max-width").pfValue; t.labelStyleKey = p, t.sourceLabelKey = p + "$" + a, t.targetLabelKey = p + "$" + s, t.labelKey = p + "$" + o, t.fontKey = l + "$" + d + "$" + c + "$" + u, t.styleKey = Date.now() } }, applyParsedProperty: function (e, t) { var n = this, o = t, a = e._private.style, s = void 0, l = n.types, c = n.properties[o.name].type, u = o.bypass, d = a[o.name], p = d && d.bypass, f = e._private, h = function () { n.checkZOrderTrigger(e, o.name, d ? d.value : null, o.value) }; if ("curve-style" === t.name && "haystack" === t.value && e.isEdge() && (e.isLoop() || e.source().isParent() || e.target().isParent()) && (o = t = this.parse(t.name, "bezier", u)), o.delete) return a[o.name] = void 0, h(), !0; if (o.deleteBypassed) return d ? !!d.bypass && (d.bypassed = void 0, h(), !0) : (h(), !0); if (o.deleteBypass) return d ? !!d.bypass && (a[o.name] = d.bypassed, h(), !0) : (h(), !0); var g = function () { r.error("Do not assign mappings to elements without corresponding data (e.g. ele `" + e.id() + "` for property `" + o.name + "` with data field `" + o.field + "`); try a `[" + o.field + "]` selector to limit scope to elements with `" + o.field + "` defined") }; switch (o.mapped) { case l.mapData: for (var m = o.field.split("."), v = f.data, b = 0; b < m.length && v; b++)v = v[m[b]]; var y = void 0; if ((y = i.number(v) ? (v - o.fieldMin) / (o.fieldMax - o.fieldMin) : 0) < 0 ? y = 0 : y > 1 && (y = 1), c.color) { var x = o.valueMin[0], w = o.valueMax[0], k = o.valueMin[1], A = o.valueMax[1], E = o.valueMin[2], S = o.valueMax[2], $ = null == o.valueMin[3] ? 1 : o.valueMin[3], C = null == o.valueMax[3] ? 1 : o.valueMax[3], _ = [Math.round(x + (w - x) * y), Math.round(k + (A - k) * y), Math.round(E + (S - E) * y), Math.round($ + (C - $) * y)]; s = { bypass: o.bypass, name: o.name, value: _, strValue: "rgb(" + _[0] + ", " + _[1] + ", " + _[2] + ")" } } else { if (!c.number) return !1; var O = o.valueMin + (o.valueMax - o.valueMin) * y; s = this.parse(o.name, O, o.bypass, "mapping") } s || (s = this.parse(o.name, d.strValue, o.bypass, "mapping")), s || g(), s.mapping = o, o = s; break; case l.data: var T = o.field.split("."), j = f.data; if (j) for (var P = 0; P < T.length; P++)j = j[T[P]]; if (!(s = this.parse(o.name, j, o.bypass, "mapping"))) { var D = d ? d.strValue : ""; s = this.parse(o.name, D, o.bypass, "mapping") } s || g(), s.mapping = o, o = s; break; case l.fn: var R = (0, o.value)(e); (s = this.parse(o.name, R, o.bypass, "mapping")).mapping = o, o = s; break; case void 0: break; default: return !1 }return u ? (o.bypassed = p ? d.bypassed : d, a[o.name] = o) : p ? d.bypassed = o : a[o.name] = o, h(), !0 }, cleanElements: function (e, t) { for (var n = this.properties, r = 0; r < e.length; r++) { var i = e[r]; if (t) for (var o = i._private.style, a = 0; a < n.length; a++) { var s = n[a], l = o[s.name]; l && (l.bypass ? l.bypassed = null : o[s.name] = null) } else i._private.style = {} } }, update: function () { this._private.cy.mutableElements().updateStyle() }, updateMappers: function (e) { for (var t = this._private.cy.collection(), n = 0; n < e.length; n++) { for (var r = e[n], i = r._private.style, o = !1, a = 0; a < this.properties.length; a++) { var s = i[this.properties[a].name]; if (s && s.mapping) { var l = s.mapping; this.applyParsedProperty(r, l), o = !0 } } o && (this.updateStyleHints(r), t.merge(r)) } return t }, updateTransitions: function (e, t, n) { var r = this, a = e._private, s = e.pstyle("transition-property").value, l = e.pstyle("transition-duration").pfValue, c = e.pstyle("transition-delay").pfValue; if (s.length > 0 && l > 0) { for (var u = {}, d = !1, p = 0; p < s.length; p++) { var f = s[p], h = e.pstyle(f), g = t[f]; if (g) { var m = g.prev, v = null != g.next ? g.next : h, b = !1, y = void 0; m && (i.number(m.pfValue) && i.number(v.pfValue) ? (b = v.pfValue - m.pfValue, y = m.pfValue + 1e-6 * b) : i.number(m.value) && i.number(v.value) ? (b = v.value - m.value, y = m.value + 1e-6 * b) : i.array(m.value) && i.array(v.value) && (b = m.value[0] !== v.value[0] || m.value[1] !== v.value[1] || m.value[2] !== v.value[2], y = m.strValue), b && (u[f] = v.strValue, this.applyBypass(e, f, y), d = !0)) } } if (!d) return; a.transitioning = !0, new o((function (t) { c > 0 ? e.delayAnimation(c).play().promise().then(t) : t() })).then((function () { return e.animation({ style: u, duration: l, easing: e.pstyle("transition-timing-function").value, queue: !1 }).play().promise() })).then((function () { r.removeBypasses(e, s), e.emitAndNotify("style"), a.transitioning = !1 })) } else a.transitioning && (this.removeBypasses(e, s), e.emitAndNotify("style"), a.transitioning = !1) }, checkZOrderTrigger: function (e, t, n, r) { var i = this.properties[t]; null == i.triggersZOrder || null != n && !i.triggersZOrder(n, r) || this._private.cy.notify({ type: "zorder", eles: e }) } }; e.exports = a }, function (e, t, n) { "use strict"; var r = n(0), i = n(1), o = { applyBypass: function (e, t, n, o) { var a = []; if ("*" === t || "**" === t) { if (void 0 !== n) for (var s = 0; s < this.properties.length; s++) { var l = this.properties[s].name, c = this.parse(l, n, !0); c && a.push(c) } } else if (r.string(t)) { var u = this.parse(t, n, !0); u && a.push(u) } else { if (!r.plainObject(t)) return !1; var d = t; o = n; for (var p = Object.keys(d), f = 0; f < p.length; f++) { var h = p[f], g = (this.properties[h], d[h]); if (void 0 === g && (g = d[i.dash2camel(h)]), void 0 !== g) { var m = this.parse(h, g, !0); m && a.push(m) } } } if (0 === a.length) return !1; for (var v = !1, b = 0; b < e.length; b++) { for (var y = e[b], x = {}, w = void 0, k = 0; k < a.length; k++) { var A = a[k]; if (o) { var E = y.pstyle(A.name); w = x[A.name] = { prev: E } } v = this.applyParsedProperty(y, A) || v, o && (w.next = y.pstyle(A.name)) } v && this.updateStyleHints(y), o && this.updateTransitions(y, x, !0) } return v }, overrideBypass: function (e, t, n) { t = i.camel2dash(t); for (var r = 0; r < e.length; r++) { var o = e[r], a = o._private.style[t], s = this.properties[t].type, l = s.color, c = s.mutiple; if (a && a.bypass) { var u = null != a.pfValue ? a.pfValue : a.value; a.value = n, null != a.pfValue && (a.pfValue = n), a.strValue = l ? "rgb(" + n.join(",") + ")" : c ? n.join(" ") : "" + n, this.checkZOrderTrigger(o, t, u, n) } else this.applyBypass(o, t, n) } }, removeAllBypasses: function (e, t) { return this.removeBypasses(e, this.propertyNames, t) }, removeBypasses: function (e, t, n) { for (var r = 0; r < e.length; r++) { for (var i = e[r], o = {}, a = 0; a < t.length; a++) { var s = t[a], l = this.properties[s], c = i.pstyle(l.name); if (c && c.bypass) { var u = this.parse(s, "", !0), d = o[l.name] = { prev: c }; this.applyParsedProperty(i, u), d.next = i.pstyle(l.name) } } this.updateStyleHints(i), n && this.updateTransitions(i, o, !0) } } }; e.exports = o }, function (e, t, n) { "use strict"; var r = n(3), i = { getEmSizeInPixels: function () { var e = this.containerCss("font-size"); return null != e ? parseFloat(e) : 1 }, containerCss: function (e) { var t = this._private.cy.container(); if (r && t && r.getComputedStyle) return r.getComputedStyle(t).getPropertyValue(e) } }; e.exports = i }, function (e, t, n) { "use strict"; var r = n(1), i = n(0), o = { getRenderedStyle: function (e, t) { return t ? this.getStylePropertyValue(e, t, !0) : this.getRawStyle(e, !0) }, getRawStyle: function (e, t) { if (e = e[0]) { for (var n = {}, i = 0; i < this.properties.length; i++) { var o = this.properties[i], a = this.getStylePropertyValue(e, o.name, t); null != a && (n[o.name] = a, n[r.dash2camel(o.name)] = a) } return n } }, getIndexedStyle: function (e, t, n, r) { var i = e.pstyle(t)[n][r]; return null != i ? i : e.cy().style().getDefaultProperty(t)[n][0] }, getStylePropertyValue: function (e, t, n) { if (e = e[0]) { var r = this.properties[t]; r.alias && (r = r.pointsTo); var i = r.type, o = e.pstyle(r.name), a = e.cy().zoom(); if (o) { var s = o.units ? i.implicitUnits || "px" : null; return s ? [].concat(o.pfValue).map((function (e) { return e * (n ? a : 1) + s })).join(" ") : o.strValue } } }, getAnimationStartStyle: function (e, t) { for (var n = {}, r = 0; r < t.length; r++) { var o = t[r].name, a = e.pstyle(o); void 0 !== a && (a = i.plainObject(a) ? this.parse(o, a.strValue) : this.parse(o, a)), a && (n[o] = a) } return n }, getPropsList: function (e) { var t = [], n = e, i = this.properties; if (n) for (var o = Object.keys(n), a = 0; a < o.length; a++) { var s = o[a], l = n[s], c = i[s] || i[r.camel2dash(s)], u = this.parse(c.name, l); u && t.push(u) } return t } }; e.exports = o }, function (e, t, n) { "use strict"; var r = { appendFromJson: function (e) { for (var t = 0; t < e.length; t++) { var n = e[t], r = n.selector, i = n.style || n.css, o = Object.keys(i); this.selector(r); for (var a = 0; a < o.length; a++) { var s = o[a], l = i[s]; this.css(s, l) } } return this }, fromJson: function (e) { return this.resetToDefault(), this.appendFromJson(e), this }, json: function () { for (var e = [], t = this.defaultLength; t < this.length; t++) { for (var n = this[t], r = n.selector, i = n.properties, o = {}, a = 0; a < i.length; a++) { var s = i[a]; o[s.name] = s.strValue } e.push({ selector: r ? r.toString() : "core", style: o }) } return e } }; e.exports = r }, function (e, t, n) { "use strict"; var r = n(1), i = n(6), o = { appendFromString: function (e) { var t = "" + e, n = void 0, o = void 0, a = void 0; function s() { t = t.length > n.length ? t.substr(n.length) : "" } function l() { o = o.length > a.length ? o.substr(a.length) : "" } for (t = t.replace(/[/][*](\s|.)+?[*][/]/g, ""); !t.match(/^\s*$/);) { var c = t.match(/^\s*((?:.|\s)+?)\s*\{((?:.|\s)+?)\}/); if (!c) { r.error("Halting stylesheet parsing: String stylesheet contains more to parse but no selector and block found in: " + t); break } n = c[0]; var u = c[1]; if ("core" !== u && new i(u)._private.invalid) r.error("Skipping parsing of block: Invalid selector found in string stylesheet: " + u), s(); else { var d = c[2], p = !1; o = d; for (var f = []; !o.match(/^\s*$/);) { var h = o.match(/^\s*(.+?)\s*:\s*(.+?)\s*;/); if (!h) { r.error("Skipping parsing of block: Invalid formatting of style property and value definitions found in:" + d), p = !0; break } a = h[0]; var g = h[1], m = h[2]; this.properties[g] ? this.parse(g, m) ? (f.push({ name: g, val: m }), l()) : (r.error("Skipping property: Invalid property definition in: " + a), l()) : (r.error("Skipping property: Invalid property name in: " + a), l()) } if (p) { s(); break } this.selector(u); for (var v = 0; v < f.length; v++) { var b = f[v]; this.css(b.name, b.val) } s() } } return this }, fromString: function (e) { return this.resetToDefault(), this.appendFromString(e), this } }; e.exports = o }, function (e, t, n) { "use strict"; var r = n(1), i = n(0), o = {}; !function () { var e = r.regex.number, t = r.regex.rgbaNoBackRefs, n = r.regex.hslaNoBackRefs, a = r.regex.hex3, s = r.regex.hex6, l = function (e) { return "^" + e + "\\s*\\(\\s*([\\w\\.]+)\\s*\\)$" }, c = function (r) { var i = e + "|\\w+|" + t + "|" + n + "|" + a + "|" + s; return "^" + r + "\\s*\\(([\\w\\.]+)\\s*\\,\\s*(" + e + ")\\s*\\,\\s*(" + e + ")\\s*,\\s*(" + i + ")\\s*\\,\\s*(" + i + ")\\)$" }, u = ["^url\\s*\\(\\s*['\"]?(.+?)['\"]?\\s*\\)$", "^(none)$", "^(.+)$"]; o.types = { time: { number: !0, min: 0, units: "s|ms", implicitUnits: "ms" }, percent: { number: !0, min: 0, max: 100, units: "%", implicitUnits: "%" }, zeroOneNumber: { number: !0, min: 0, max: 1, unitless: !0 }, zeroOneNumbers: { number: !0, min: 0, max: 1, unitless: !0, multiple: !0 }, nOneOneNumber: { number: !0, min: -1, max: 1, unitless: !0 }, nonNegativeInt: { number: !0, min: 0, integer: !0, unitless: !0 }, position: { enums: ["parent", "origin"] }, nodeSize: { number: !0, min: 0, enums: ["label"] }, number: { number: !0, unitless: !0 }, numbers: { number: !0, unitless: !0, multiple: !0 }, positiveNumber: { number: !0, unitless: !0, min: 0, strictMin: !0 }, size: { number: !0, min: 0 }, bidirectionalSize: { number: !0 }, bidirectionalSizes: { number: !0, multiple: !0 }, sizeMaybePercent: { number: !0, min: 0, allowPercent: !0 }, paddingRelativeTo: { enums: ["width", "height", "average", "min", "max"] }, bgWH: { number: !0, min: 0, allowPercent: !0, enums: ["auto"], multiple: !0 }, bgPos: { number: !0, allowPercent: !0, multiple: !0 }, bgRelativeTo: { enums: ["inner", "include-padding"], multiple: !0 }, bgRepeat: { enums: ["repeat", "repeat-x", "repeat-y", "no-repeat"], multiple: !0 }, bgFit: { enums: ["none", "contain", "cover"], multiple: !0 }, bgCrossOrigin: { enums: ["anonymous", "use-credentials"], multiple: !0 }, bgClip: { enums: ["none", "node"] }, color: { color: !0 }, bool: { enums: ["yes", "no"] }, lineStyle: { enums: ["solid", "dotted", "dashed"] }, borderStyle: { enums: ["solid", "dotted", "dashed", "double"] }, curveStyle: { enums: ["bezier", "unbundled-bezier", "haystack", "segments"] }, fontFamily: { regex: '^([\\w- \\"]+(?:\\s*,\\s*[\\w- \\"]+)*)$' }, fontletiant: { enums: ["small-caps", "normal"] }, fontStyle: { enums: ["italic", "normal", "oblique"] }, fontWeight: { enums: ["normal", "bold", "bolder", "lighter", "100", "200", "300", "400", "500", "600", "800", "900", 100, 200, 300, 400, 500, 600, 700, 800, 900] }, textDecoration: { enums: ["none", "underline", "overline", "line-through"] }, textTransform: { enums: ["none", "uppercase", "lowercase"] }, textWrap: { enums: ["none", "wrap", "ellipsis"] }, textBackgroundShape: { enums: ["rectangle", "roundrectangle"] }, nodeShape: { enums: ["rectangle", "roundrectangle", "cutrectangle", "bottomroundrectangle", "barrel", "ellipse", "triangle", "square", "pentagon", "hexagon", "concavehexagon", "heptagon", "octagon", "tag", "star", "diamond", "vee", "rhomboid", "polygon"] }, compoundIncludeLabels: { enums: ["include", "exclude"] }, arrowShape: { enums: ["tee", "triangle", "triangle-tee", "triangle-cross", "triangle-backcurve", "half-triangle-overshot", "vee", "square", "circle", "diamond", "none"] }, arrowFill: { enums: ["filled", "hollow"] }, display: { enums: ["element", "none"] }, visibility: { enums: ["hidden", "visible"] }, zCompoundDepth: { enums: ["bottom", "orphan", "auto", "top"] }, zIndexCompare: { enums: ["auto", "manual"] }, valign: { enums: ["top", "center", "bottom"] }, halign: { enums: ["left", "center", "right"] }, text: { string: !0 }, data: { mapping: !0, regex: l("data") }, layoutData: { mapping: !0, regex: l("layoutData") }, scratch: { mapping: !0, regex: l("scratch") }, mapData: { mapping: !0, regex: c("mapData") }, mapLayoutData: { mapping: !0, regex: c("mapLayoutData") }, mapScratch: { mapping: !0, regex: c("mapScratch") }, fn: { mapping: !0, fn: !0 }, url: { regexes: u, singleRegexMatchValue: !0 }, urls: { regexes: u, singleRegexMatchValue: !0, multiple: !0 }, propList: { propList: !0 }, angle: { number: !0, units: "deg|rad", implicitUnits: "rad" }, textRotation: { number: !0, units: "deg|rad", implicitUnits: "rad", enums: ["none", "autorotate"] }, polygonPointList: { number: !0, multiple: !0, evenMultiple: !0, min: -1, max: 1, unitless: !0 }, edgeDistances: { enums: ["intersection", "node-position"] }, edgeEndpoint: { number: !0, multiple: !0, units: "%|px|em|deg|rad", implicitUnits: "px", enums: ["inside-to-node", "outside-to-node", "outside-to-line"], singleEnum: !0, validate: function (e, t) { switch (e.length) { case 2: return "deg" !== t[0] && "rad" !== t[0] && "deg" !== t[1] && "rad" !== t[1]; case 1: return i.string(e[0]) || "deg" === t[0] || "rad" === t[0]; default: return !1 } } }, easing: { regexes: ["^(spring)\\s*\\(\\s*(" + e + ")\\s*,\\s*(" + e + ")\\s*\\)$", "^(cubic-bezier)\\s*\\(\\s*(" + e + ")\\s*,\\s*(" + e + ")\\s*,\\s*(" + e + ")\\s*,\\s*(" + e + ")\\s*\\)$"], enums: ["linear", "ease", "ease-in", "ease-out", "ease-in-out", "ease-in-sine", "ease-out-sine", "ease-in-out-sine", "ease-in-quad", "ease-out-quad", "ease-in-out-quad", "ease-in-cubic", "ease-out-cubic", "ease-in-out-cubic", "ease-in-quart", "ease-out-quart", "ease-in-out-quart", "ease-in-quint", "ease-out-quint", "ease-in-out-quint", "ease-in-expo", "ease-out-expo", "ease-in-out-expo", "ease-in-circ", "ease-out-circ", "ease-in-out-circ"] } }; var d = { zeroNonZero: function (e, t) { return 0 === e && 0 !== t || 0 !== e && 0 === t }, anyDiff: function (e, t) { return e !== t } }, p = o.types, f = o.properties = [{ name: "label", type: p.text }, { name: "text-rotation", type: p.textRotation }, { name: "text-margin-x", type: p.bidirectionalSize }, { name: "text-margin-y", type: p.bidirectionalSize }, { name: "source-label", type: p.text }, { name: "source-text-rotation", type: p.textRotation }, { name: "source-text-margin-x", type: p.bidirectionalSize }, { name: "source-text-margin-y", type: p.bidirectionalSize }, { name: "source-text-offset", type: p.size }, { name: "target-label", type: p.text }, { name: "target-text-rotation", type: p.textRotation }, { name: "target-text-margin-x", type: p.bidirectionalSize }, { name: "target-text-margin-y", type: p.bidirectionalSize }, { name: "target-text-offset", type: p.size }, { name: "text-valign", type: p.valign }, { name: "text-halign", type: p.halign }, { name: "color", type: p.color }, { name: "text-outline-color", type: p.color }, { name: "text-outline-width", type: p.size }, { name: "text-outline-opacity", type: p.zeroOneNumber }, { name: "text-opacity", type: p.zeroOneNumber }, { name: "text-background-color", type: p.color }, { name: "text-background-opacity", type: p.zeroOneNumber }, { name: "text-background-padding", type: p.size }, { name: "text-border-opacity", type: p.zeroOneNumber }, { name: "text-border-color", type: p.color }, { name: "text-border-width", type: p.size }, { name: "text-border-style", type: p.borderStyle }, { name: "text-background-shape", type: p.textBackgroundShape }, { name: "text-transform", type: p.textTransform }, { name: "text-wrap", type: p.textWrap }, { name: "text-max-width", type: p.size }, { name: "text-events", type: p.bool }, { name: "font-family", type: p.fontFamily }, { name: "font-style", type: p.fontStyle }, { name: "font-weight", type: p.fontWeight }, { name: "font-size", type: p.size }, { name: "min-zoomed-font-size", type: p.size }, { name: "events", type: p.bool }, { name: "display", type: p.display, triggersZOrder: d.anyDiff }, { name: "visibility", type: p.visibility, triggersZOrder: d.anyDiff }, { name: "opacity", type: p.zeroOneNumber, triggersZOrder: d.zeroNonZero }, { name: "z-compound-depth", type: p.zCompoundDepth, triggersZOrder: d.anyDiff }, { name: "z-index-compare", type: p.zIndexCompare, triggersZOrder: d.anyDiff }, { name: "z-index", type: p.nonNegativeInt, triggersZOrder: d.anyDiff }, { name: "overlay-padding", type: p.size }, { name: "overlay-color", type: p.color }, { name: "overlay-opacity", type: p.zeroOneNumber }, { name: "transition-property", type: p.propList }, { name: "transition-duration", type: p.time }, { name: "transition-delay", type: p.time }, { name: "transition-timing-function", type: p.easing }, { name: "height", type: p.nodeSize }, { name: "width", type: p.nodeSize }, { name: "shape", type: p.nodeShape }, { name: "shape-polygon-points", type: p.polygonPointList }, { name: "background-color", type: p.color }, { name: "background-opacity", type: p.zeroOneNumber }, { name: "background-blacken", type: p.nOneOneNumber }, { name: "padding", type: p.sizeMaybePercent }, { name: "padding-relative-to", type: p.paddingRelativeTo }, { name: "border-color", type: p.color }, { name: "border-opacity", type: p.zeroOneNumber }, { name: "border-width", type: p.size }, { name: "border-style", type: p.borderStyle }, { name: "background-image", type: p.urls }, { name: "background-image-crossorigin", type: p.bgCrossOrigin }, { name: "background-image-opacity", type: p.zeroOneNumbers }, { name: "background-position-x", type: p.bgPos }, { name: "background-position-y", type: p.bgPos }, { name: "background-width-relative-to", type: p.bgRelativeTo }, { name: "background-height-relative-to", type: p.bgRelativeTo }, { name: "background-repeat", type: p.bgRepeat }, { name: "background-fit", type: p.bgFit }, { name: "background-clip", type: p.bgClip }, { name: "background-width", type: p.bgWH }, { name: "background-height", type: p.bgWH }, { name: "position", type: p.position }, { name: "compound-sizing-wrt-labels", type: p.compoundIncludeLabels }, { name: "min-width", type: p.size }, { name: "min-width-bias-left", type: p.sizeMaybePercent }, { name: "min-width-bias-right", type: p.sizeMaybePercent }, { name: "min-height", type: p.size }, { name: "min-height-bias-top", type: p.sizeMaybePercent }, { name: "min-height-bias-bottom", type: p.sizeMaybePercent }, { name: "line-style", type: p.lineStyle }, { name: "line-color", type: p.color }, { name: "curve-style", type: p.curveStyle }, { name: "haystack-radius", type: p.zeroOneNumber }, { name: "source-endpoint", type: p.edgeEndpoint }, { name: "target-endpoint", type: p.edgeEndpoint }, { name: "control-point-step-size", type: p.size }, { name: "control-point-distances", type: p.bidirectionalSizes }, { name: "control-point-weights", type: p.numbers }, { name: "segment-distances", type: p.bidirectionalSizes }, { name: "segment-weights", type: p.numbers }, { name: "edge-distances", type: p.edgeDistances }, { name: "arrow-scale", type: p.positiveNumber }, { name: "loop-direction", type: p.angle }, { name: "loop-sweep", type: p.angle }, { name: "source-distance-from-node", type: p.size }, { name: "target-distance-from-node", type: p.size }, { name: "ghost", type: p.bool }, { name: "ghost-offset-x", type: p.bidirectionalSize }, { name: "ghost-offset-y", type: p.bidirectionalSize }, { name: "ghost-opacity", type: p.zeroOneNumber }, { name: "selection-box-color", type: p.color }, { name: "selection-box-opacity", type: p.zeroOneNumber }, { name: "selection-box-border-color", type: p.color }, { name: "selection-box-border-width", type: p.size }, { name: "active-bg-color", type: p.color }, { name: "active-bg-opacity", type: p.zeroOneNumber }, { name: "active-bg-size", type: p.size }, { name: "outside-texture-bg-color", type: p.color }, { name: "outside-texture-bg-opacity", type: p.zeroOneNumber }], h = o.aliases = [{ name: "content", pointsTo: "label" }, { name: "control-point-distance", pointsTo: "control-point-distances" }, { name: "control-point-weight", pointsTo: "control-point-weights" }, { name: "edge-text-rotation", pointsTo: "text-rotation" }, { name: "padding-left", pointsTo: "padding" }, { name: "padding-right", pointsTo: "padding" }, { name: "padding-top", pointsTo: "padding" }, { name: "padding-bottom", pointsTo: "padding" }]; o.pieBackgroundN = 16, f.push({ name: "pie-size", type: p.sizeMaybePercent }); for (var g = 1; g <= o.pieBackgroundN; g++)f.push({ name: "pie-" + g + "-background-color", type: p.color }), f.push({ name: "pie-" + g + "-background-size", type: p.percent }), f.push({ name: "pie-" + g + "-background-opacity", type: p.zeroOneNumber }); var m = o.arrowPrefixes = ["source", "mid-source", "target", "mid-target"];[{ name: "arrow-shape", type: p.arrowShape }, { name: "arrow-color", type: p.color }, { name: "arrow-fill", type: p.arrowFill }].forEach((function (e) { m.forEach((function (t) { var n = t + "-" + e.name, r = e.type; f.push({ name: n, type: r }) })) }), {}), o.propertyNames = f.map((function (e) { return e.name })); for (var v = 0; v < f.length; v++) { var b = f[v]; f[b.name] = b } for (var y = 0; y < h.length; y++) { var x = h[y], w = f[x.pointsTo], k = { name: x.name, alias: !0, pointsTo: w }; f.push(k), f[x.name] = k } }(), o.getDefaultProperty = function (e) { return this.getDefaultProperties()[e] }, o.getDefaultProperties = r.memoize((function () { for (var e = r.extend({ events: "yes", "text-events": "no", "text-valign": "top", "text-halign": "center", color: "#000", "text-outline-color": "#000", "text-outline-width": 0, "text-outline-opacity": 1, "text-opacity": 1, "text-decoration": "none", "text-transform": "none", "text-wrap": "none", "text-max-width": 9999, "text-background-color": "#000", "text-background-opacity": 0, "text-background-shape": "rectangle", "text-background-padding": 0, "text-border-opacity": 0, "text-border-width": 0, "text-border-style": "solid", "text-border-color": "#000", "font-family": "Helvetica Neue, Helvetica, sans-serif", "font-style": "normal", "font-weight": "normal", "font-size": 16, "min-zoomed-font-size": 0, "text-rotation": "none", "source-text-rotation": "none", "target-text-rotation": "none", visibility: "visible", display: "element", opacity: 1, "z-compound-depth": "auto", "z-index-compare": "auto", "z-index": 0, label: "", "text-margin-x": 0, "text-margin-y": 0, "source-label": "", "source-text-offset": 0, "source-text-margin-x": 0, "source-text-margin-y": 0, "target-label": "", "target-text-offset": 0, "target-text-margin-x": 0, "target-text-margin-y": 0, "overlay-opacity": 0, "overlay-color": "#000", "overlay-padding": 10, "transition-property": "none", "transition-duration": 0, "transition-delay": 0, "transition-timing-function": "linear", "background-blacken": 0, "background-color": "#999", "background-opacity": 1, "background-image": "none", "background-image-crossorigin": "anonymous", "background-image-opacity": 1, "background-position-x": "50%", "background-position-y": "50%", "background-width-relative-to": "include-padding", "background-height-relative-to": "include-padding", "background-repeat": "no-repeat", "background-fit": "none", "background-clip": "node", "background-width": "auto", "background-height": "auto", "border-color": "#000", "border-opacity": 1, "border-width": 0, "border-style": "solid", height: 30, width: 30, shape: "ellipse", "shape-polygon-points": "-1, -1, 1, -1, 1, 1, -1, 1", ghost: "no", "ghost-offset-y": 0, "ghost-offset-x": 0, "ghost-opacity": 0, padding: 0, "padding-relative-to": "width", position: "origin", "compound-sizing-wrt-labels": "include", "min-width": 0, "min-width-bias-left": 0, "min-width-bias-right": 0, "min-height": 0, "min-height-bias-top": 0, "min-height-bias-bottom": 0 }, { "pie-size": "100%" }, [{ name: "pie-{{i}}-background-color", value: "black" }, { name: "pie-{{i}}-background-size", value: "0%" }, { name: "pie-{{i}}-background-opacity", value: 1 }].reduce((function (e, t) { for (var n = 1; n <= o.pieBackgroundN; n++) { var r = t.name.replace("{{i}}", n), i = t.value; e[r] = i } return e }), {}), { "line-style": "solid", "line-color": "#999", "control-point-step-size": 40, "control-point-weights": .5, "segment-weights": .5, "segment-distances": 20, "edge-distances": "intersection", "curve-style": "bezier", "haystack-radius": 0, "arrow-scale": 1, "loop-direction": "-45deg", "loop-sweep": "-90deg", "source-distance-from-node": 0, "target-distance-from-node": 0, "source-endpoint": "outside-to-node", "target-endpoint": "outside-to-node" }, [{ name: "arrow-shape", value: "none" }, { name: "arrow-color", value: "#999" }, { name: "arrow-fill", value: "filled" }].reduce((function (e, t) { return o.arrowPrefixes.forEach((function (n) { var r = n + "-" + t.name, i = t.value; e[r] = i })), e }), {})), t = {}, n = 0; n < this.properties.length; n++) { var i = this.properties[n]; if (!i.pointsTo) { var a = i.name, s = e[a], l = this.parse(a, s); t[a] = l } } return t })), o.addDefaultStylesheet = function () { this.selector("$node > node").css({ shape: "rectangle", padding: 10, "background-color": "#eee", "border-color": "#ccc", "border-width": 1 }).selector("edge").css({ width: 3, "curve-style": "haystack" }).selector(":parent <-> node").css({ "curve-style": "bezier", "source-endpoint": "outside-to-line", "target-endpoint": "outside-to-line" }).selector(":selected").css({ "background-color": "#0169D9", "line-color": "#0169D9", "source-arrow-color": "#0169D9", "target-arrow-color": "#0169D9", "mid-source-arrow-color": "#0169D9", "mid-target-arrow-color": "#0169D9" }).selector("node:parent:selected").css({ "background-color": "#CCE1F9", "border-color": "#aec8e5" }).selector(":active").css({ "overlay-color": "black", "overlay-padding": 10, "overlay-opacity": .25 }).selector("core").css({ "selection-box-color": "#ddd", "selection-box-opacity": .65, "selection-box-border-color": "#aaa", "selection-box-border-width": 1, "active-bg-color": "black", "active-bg-opacity": .15, "active-bg-size": 30, "outside-texture-bg-color": "#000", "outside-texture-bg-opacity": .125 }), this.defaultLength = this.length }, e.exports = o }, function (e, t, n) { "use strict"; var r = n(1), i = n(0), o = n(2), a = { parse: function (e, t, n, o) { if (i.fn(t)) return this.parseImplWarn(e, t, n, o); var a = [e, t, n, "mapping" === o || !0 === o || !1 === o || null == o ? "dontcare" : o].join("$"), s = this.propCache = this.propCache || {}, l = void 0; return (l = s[a]) || (l = s[a] = this.parseImplWarn(e, t, n, o)), (n || "mapping" === o) && (l = r.copy(l)) && (l.value = r.copy(l.value)), l }, parseImplWarn: function (e, t, n, i) { var o = this.parseImpl(e, t, n, i); return o || null == t || r.error("The style property `%s: %s` is invalid", e, t), o }, parseImpl: function (e, t, n, a) { e = r.camel2dash(e); var s = this.properties[e], l = t, c = this.types; if (!s) return null; if (void 0 === t) return null; s.alias && (s = s.pointsTo, e = s.name); var u = i.string(t); u && (t = t.trim()); var d = s.type; if (!d) return null; if (n && ("" === t || null === t)) return { name: e, value: t, bypass: !0, deleteBypass: !0 }; if (i.fn(t)) return { name: e, value: t, strValue: "fn", mapped: c.fn, bypass: n }; var p = void 0, f = void 0; if (!u || a); else { if (p = new RegExp(c.data.regex).exec(t)) { if (n) return !1; var h = c.data; return { name: e, value: p, strValue: "" + t, mapped: h, field: p[1], bypass: n } } if (f = new RegExp(c.mapData.regex).exec(t)) { if (n) return !1; if (d.multiple) return !1; var g = c.mapData; if (!d.color && !d.number) return !1; var m = this.parse(e, f[4]); if (!m || m.mapped) return !1; var v = this.parse(e, f[5]); if (!v || v.mapped) return !1; if (m.value === v.value) return !1; if (d.color) { var b = m.value, y = v.value; if (!(b[0] !== y[0] || b[1] !== y[1] || b[2] !== y[2] || b[3] !== y[3] && (null != b[3] && 1 !== b[3] || null != y[3] && 1 !== y[3]))) return !1 } return { name: e, value: f, strValue: "" + t, mapped: g, field: f[1], fieldMin: parseFloat(f[2]), fieldMax: parseFloat(f[3]), valueMin: m.value, valueMax: v.value, bypass: n } } } if (d.multiple && "multiple" !== a) { var x = void 0; if (x = u ? t.split(/\s+/) : i.array(t) ? t : [t], d.evenMultiple && x.length % 2 != 0) return null; for (var w = [], k = [], A = [], E = !1, S = 0; S < x.length; S++) { var $ = this.parse(e, x[S], n, "multiple"); E = E || i.string($.value), w.push($.value), A.push(null != $.pfValue ? $.pfValue : $.value), k.push($.units) } return d.validate && !d.validate(w, k) ? null : d.singleEnum && E ? 1 === w.length && i.string(w[0]) ? { name: e, value: w[0], strValue: w[0], bypass: n } : null : { name: e, value: w, pfValue: A, strValue: w.map((function (e, t) { return e + (k[t] || "") })).join(" "), bypass: n, units: k } } var C = function () { for (var r = 0; r < d.enums.length; r++)if (d.enums[r] === t) return { name: e, value: t, strValue: "" + t, bypass: n }; return null }; if (d.number) { var _ = void 0, O = "px"; if (d.units && (_ = d.units), d.implicitUnits && (O = d.implicitUnits), !d.unitless) if (u) { var T = "px|em" + (d.allowPercent ? "|\\%" : ""); _ && (T = _); var j = t.match("^(" + r.regex.number + ")(" + T + ")?$"); j && (t = j[1], _ = j[2] || O) } else _ && !d.implicitUnits || (_ = O); if (t = parseFloat(t), isNaN(t) && void 0 === d.enums) return null; if (isNaN(t) && void 0 !== d.enums) return t = l, C(); if (d.integer && !i.integer(t)) return null; if (void 0 !== d.min && (t < d.min || d.strictMin && t === d.min) || void 0 !== d.max && (t > d.max || d.strictMax && t === d.max)) return null; var P = { name: e, value: t, strValue: "" + t + (_ || ""), units: _, bypass: n }; return d.unitless || "px" !== _ && "em" !== _ ? P.pfValue = t : P.pfValue = "px" !== _ && _ ? this.getEmSizeInPixels() * t : t, "ms" !== _ && "s" !== _ || (P.pfValue = "ms" === _ ? t : 1e3 * t), "deg" !== _ && "rad" !== _ || (P.pfValue = "rad" === _ ? t : o.deg2rad(t)), "%" === _ && (P.pfValue = t / 100), P } if (d.propList) { var D = [], R = "" + t; if ("none" === R); else { for (var I = R.split(","), N = 0; N < I.length; N++) { var M = I[N].trim(); this.properties[M] && D.push(M) } if (0 === D.length) return null } return { name: e, value: D, strValue: 0 === D.length ? "none" : D.join(", "), bypass: n } } if (d.color) { var z = r.color2tuple(t); return z ? { name: e, value: z, pfValue: z, strValue: "" + t, bypass: n } : null } if (d.regex || d.regexes) { if (d.enums) { var L = C(); if (L) return L } for (var B = d.regexes ? d.regexes : [d.regex], F = 0; F < B.length; F++) { var q = new RegExp(B[F]).exec(t); if (q) return { name: e, value: d.singleRegexMatchValue ? q[1] : q, strValue: "" + t, bypass: n } } return null } return d.string ? { name: e, value: "" + t, strValue: "" + t, bypass: n } : d.enums ? C() : null } }; e.exports = a }, function (e, t, n) { "use strict"; var r = n(0), i = n(3), o = n(2), a = { autolock: function (e) { return void 0 === e ? this._private.autolock : (this._private.autolock = !!e, this) }, autoungrabify: function (e) { return void 0 === e ? this._private.autoungrabify : (this._private.autoungrabify = !!e, this) }, autounselectify: function (e) { return void 0 === e ? this._private.autounselectify : (this._private.autounselectify = !!e, this) }, panningEnabled: function (e) { return void 0 === e ? this._private.panningEnabled : (this._private.panningEnabled = !!e, this) }, userPanningEnabled: function (e) { return void 0 === e ? this._private.userPanningEnabled : (this._private.userPanningEnabled = !!e, this) }, zoomingEnabled: function (e) { return void 0 === e ? this._private.zoomingEnabled : (this._private.zoomingEnabled = !!e, this) }, userZoomingEnabled: function (e) { return void 0 === e ? this._private.userZoomingEnabled : (this._private.userZoomingEnabled = !!e, this) }, boxSelectionEnabled: function (e) { return void 0 === e ? this._private.boxSelectionEnabled : (this._private.boxSelectionEnabled = !!e, this) }, pan: function () { var e = arguments, t = this._private.pan, n = void 0, i = void 0, o = void 0, a = void 0, s = void 0; switch (e.length) { case 0: return t; case 1: if (r.string(e[0])) return t[n = e[0]]; if (r.plainObject(e[0])) { if (!this._private.panningEnabled) return this; a = (o = e[0]).x, s = o.y, r.number(a) && (t.x = a), r.number(s) && (t.y = s), this.emit("pan viewport") } break; case 2: if (!this._private.panningEnabled) return this; n = e[0], i = e[1], "x" !== n && "y" !== n || !r.number(i) || (t[n] = i), this.emit("pan viewport") }return this.notify({ type: "viewport" }), this }, panBy: function (e, t) { var n = arguments, i = this._private.pan, o = void 0, a = void 0, s = void 0, l = void 0, c = void 0; if (!this._private.panningEnabled) return this; switch (n.length) { case 1: r.plainObject(e) && (l = (s = n[0]).x, c = s.y, r.number(l) && (i.x += l), r.number(c) && (i.y += c), this.emit("pan viewport")); break; case 2: a = t, "x" !== (o = e) && "y" !== o || !r.number(a) || (i[o] += a), this.emit("pan viewport") }return this.notify({ type: "viewport" }), this }, fit: function (e, t) { var n = this.getFitViewport(e, t); if (n) { var r = this._private; r.zoom = n.zoom, r.pan = n.pan, this.emit("pan zoom viewport"), this.notify({ type: "viewport" }) } return this }, getFitViewport: function (e, t) { if (r.number(e) && void 0 === t && (t = e, e = void 0), this._private.panningEnabled && this._private.zoomingEnabled) { var n = void 0; if (r.string(e)) { var i = e; e = this.$(i) } else if (r.boundingBox(e)) { var o = e; (n = { x1: o.x1, y1: o.y1, x2: o.x2, y2: o.y2 }).w = n.x2 - n.x1, n.h = n.y2 - n.y1 } else r.elementOrCollection(e) || (e = this.mutableElements()); if (!r.elementOrCollection(e) || !e.empty()) { n = n || e.boundingBox(); var a = this.width(), s = this.height(), l = void 0; if (t = r.number(t) ? t : 0, !isNaN(a) && !isNaN(s) && a > 0 && s > 0 && !isNaN(n.w) && !isNaN(n.h) && n.w > 0 && n.h > 0) return { zoom: l = (l = (l = Math.min((a - 2 * t) / n.w, (s - 2 * t) / n.h)) > this._private.maxZoom ? this._private.maxZoom : l) < this._private.minZoom ? this._private.minZoom : l, pan: { x: (a - l * (n.x1 + n.x2)) / 2, y: (s - l * (n.y1 + n.y2)) / 2 } } } } }, minZoom: function (e) { return void 0 === e ? this._private.minZoom : (r.number(e) && (this._private.minZoom = e), this) }, maxZoom: function (e) { return void 0 === e ? this._private.maxZoom : (r.number(e) && (this._private.maxZoom = e), this) }, getZoomedViewport: function (e) { var t = this._private, n = t.pan, i = t.zoom, a = void 0, s = void 0, l = !1; if (t.zoomingEnabled || (l = !0), r.number(e) ? s = e : r.plainObject(e) && (s = e.level, null != e.position ? a = o.modelToRenderedPosition(e.position, i, n) : null != e.renderedPosition && (a = e.renderedPosition), null == a || t.panningEnabled || (l = !0)), s = (s = s > t.maxZoom ? t.maxZoom : s) < t.minZoom ? t.minZoom : s, l || !r.number(s) || s === i || null != a && (!r.number(a.x) || !r.number(a.y))) return null; if (null != a) { var c = n, u = i, d = s; return { zoomed: !0, panned: !0, zoom: d, pan: { x: -d / u * (a.x - c.x) + a.x, y: -d / u * (a.y - c.y) + a.y } } } return { zoomed: !0, panned: !1, zoom: s, pan: n } }, zoom: function (e) { if (void 0 === e) return this._private.zoom; var t = this.getZoomedViewport(e), n = this._private; return null != t && t.zoomed ? (n.zoom = t.zoom, t.panned && (n.pan.x = t.pan.x, n.pan.y = t.pan.y), this.emit("zoom" + (t.panned ? " pan" : "") + " viewport"), this.notify({ type: "viewport" }), this) : this }, viewport: function (e) { var t = this._private, n = !0, i = !0, o = [], a = !1, s = !1; if (!e) return this; if (r.number(e.zoom) || (n = !1), r.plainObject(e.pan) || (i = !1), !n && !i) return this; if (n) { var l = e.zoom; l < t.minZoom || l > t.maxZoom || !t.zoomingEnabled ? a = !0 : (t.zoom = l, o.push("zoom")) } if (i && (!a || !e.cancelOnFailedZoom) && t.panningEnabled) { var c = e.pan; r.number(c.x) && (t.pan.x = c.x, s = !1), r.number(c.y) && (t.pan.y = c.y, s = !1), s || o.push("pan") } return o.length > 0 && (o.push("viewport"), this.emit(o.join(" ")), this.notify({ type: "viewport" })), this }, center: function (e) { var t = this.getCenterPan(e); return t && (this._private.pan = t, this.emit("pan viewport"), this.notify({ type: "viewport" })), this }, getCenterPan: function (e, t) { if (this._private.panningEnabled) { if (r.string(e)) { var n = e; e = this.mutableElements().filter(n) } else r.elementOrCollection(e) || (e = this.mutableElements()); if (0 !== e.length) { var i = e.boundingBox(), o = this.width(), a = this.height(); return { x: (o - (t = void 0 === t ? this._private.zoom : t) * (i.x1 + i.x2)) / 2, y: (a - t * (i.y1 + i.y2)) / 2 } } } }, reset: function () { return this._private.panningEnabled && this._private.zoomingEnabled ? (this.viewport({ pan: { x: 0, y: 0 }, zoom: 1 }), this) : this }, invalidateSize: function () { this._private.sizeCache = null }, size: function () { var e, t, n = this._private, r = n.container; return n.sizeCache = n.sizeCache || (r ? (e = i.getComputedStyle(r), t = function (t) { return parseFloat(e.getPropertyValue(t)) }, { width: r.clientWidth - t("padding-left") - t("padding-right"), height: r.clientHeight - t("padding-top") - t("padding-bottom") }) : { width: 1, height: 1 }) }, width: function () { return this.size().width }, height: function () { return this.size().height }, extent: function () { var e = this._private.pan, t = this._private.zoom, n = this.renderedExtent(), r = { x1: (n.x1 - e.x) / t, x2: (n.x2 - e.x) / t, y1: (n.y1 - e.y) / t, y2: (n.y2 - e.y) / t }; return r.w = r.x2 - r.x1, r.h = r.y2 - r.y1, r }, renderedExtent: function () { var e = this.width(), t = this.height(); return { x1: 0, y1: 0, x2: e, y2: t, w: e, h: t } } }; a.centre = a.center, a.autolockNodes = a.autolock, a.autoungrabifyNodes = a.autoungrabify, e.exports = a }, function (e, t, n) { "use strict"; var r = n(1), i = n(4), o = n(7), a = n(12), s = n(95), l = n(0), c = n(11), u = {}, d = {}; function p(e, t, n) { var s = n, d = function (n) { r.error("Can not register `" + t + "` for `" + e + "` since `" + n + "` already exists in the prototype and can not be overridden") }; if ("core" === e) { if (a.prototype[t]) return d(t); a.prototype[t] = n } else if ("collection" === e) { if (o.prototype[t]) return d(t); o.prototype[t] = n } else if ("layout" === e) { for (var p = function (e) { this.options = e, n.call(this, e), l.plainObject(this._private) || (this._private = {}), this._private.cy = e.cy, this._private.listeners = [], this.createEmitter() }, h = p.prototype = Object.create(n.prototype), g = [], m = 0; m < g.length; m++) { var v = g[m]; h[v] = h[v] || function () { return this } } h.start && !h.run ? h.run = function () { return this.start(), this } : !h.start && h.run && (h.start = function () { return this.run(), this }); var b = n.prototype.stop; h.stop = function () { var e = this.options; if (e && e.animate) { var t = this.animations; if (t) for (var n = 0; n < t.length; n++)t[n].stop() } return b ? b.call(this) : this.emit("layoutstop"), this }, h.destroy || (h.destroy = function () { return this }), h.cy = function () { return this._private.cy }; var y = function (e) { return e._private.cy }; r.assign(h, { createEmitter: function () { return this._private.emitter = new c({ eventFields: function (e) { return { layout: e, cy: y(e), target: e } }, bubble: function () { return !0 }, parent: function (e) { return y(e) }, context: this }), this }, emitter: function () { return this._private.emitter }, on: function (e, t) { return this.emitter().on(e, t), this }, one: function (e, t) { return this.emitter().one(e, t), this }, once: function (e, t) { return this.emitter().one(e, t), this }, removeListener: function (e, t) { return this.emitter().removeListener(e, t), this }, emit: function (e, t) { return this.emitter().emit(e, t), this } }), i.eventAliasesOn(h), s = p } else if ("renderer" === e && "null" !== t && "base" !== t) { var x = f("renderer", "base"), w = x.prototype, k = n, A = n.prototype, E = function () { x.apply(this, arguments), k.apply(this, arguments) }, S = E.prototype; for (var $ in w) { var C = w[$]; if (null != A[$]) return d($); S[$] = C } for (var _ in A) S[_] = A[_]; w.clientFunctions.forEach((function (e) { S[e] = S[e] || function () { r.error("Renderer does not implement `renderer." + e + "()` on its prototype") } })), s = E } return r.setMap({ map: u, keys: [e, t], value: s }) } function f(e, t) { return r.getMap({ map: u, keys: [e, t] }) } function h(e, t, n, i, o) { return r.setMap({ map: d, keys: [e, t, n, i], value: o }) } function g(e, t, n, i) { return r.getMap({ map: d, keys: [e, t, n, i] }) } var m = function () { return 2 === arguments.length ? f.apply(null, arguments) : 3 === arguments.length ? p.apply(null, arguments) : 4 === arguments.length ? g.apply(null, arguments) : 5 === arguments.length ? h.apply(null, arguments) : void r.error("Invalid extension access syntax") }; a.prototype.extension = m, s.forEach((function (e) { e.extensions.forEach((function (t) { p(e.type, t.name, t.impl) })) })), e.exports = m }, function (e, t, n) { "use strict"; e.exports = [{ type: "layout", extensions: n(96) }, { type: "renderer", extensions: n(105) }] }, function (e, t, n) { "use strict"; e.exports = [{ name: "breadthfirst", impl: n(97) }, { name: "circle", impl: n(98) }, { name: "concentric", impl: n(99) }, { name: "cose", impl: n(100) }, { name: "grid", impl: n(101) }, { name: "null", impl: n(102) }, { name: "preset", impl: n(103) }, { name: "random", impl: n(104) }] }, function (e, t, n) { "use strict"; var r = n(1), i = n(2), o = n(0), a = { fit: !0, directed: !1, padding: 30, circle: !1, spacingFactor: 1.75, boundingBox: void 0, avoidOverlap: !0, nodeDimensionsIncludeLabels: !1, roots: void 0, maximalAdjustments: 0, animate: !1, animationDuration: 500, animationEasing: void 0, animateFilter: function (e, t) { return !0 }, ready: void 0, stop: void 0, transform: function (e, t) { return t } }; function s(e) { this.options = r.extend({}, a, e) } s.prototype.run = function () { var e = this.options, t = e, n = e.cy, r = t.eles, a = r.nodes().not(":parent"), s = r, l = i.makeBoundingBox(t.boundingBox ? t.boundingBox : { x1: 0, y1: 0, w: n.width(), h: n.height() }), c = void 0; if (o.elementOrCollection(t.roots)) c = t.roots; else if (o.array(t.roots)) { for (var u = [], d = 0; d < t.roots.length; d++) { var p = t.roots[d], f = n.getElementById(p); u.push(f) } c = n.collection(u) } else if (o.string(t.roots)) c = n.$(t.roots); else if (t.directed) c = a.roots(); else { for (var h = [], g = a, m = function () { var e = n.collection(); r.bfs({ roots: g[0], visit: function (t, n, r, i, o) { e = e.add(t) }, directed: !1 }), g = g.not(e), h.push(e) }; g.length > 0;)m(); c = n.collection(); for (var v = function (e) { var t = h[e], n = t.maxDegree(!1), r = t.filter((function (e) { return e.degree(!1) === n })); c = c.add(r) }, b = 0; b < h.length; b++)v(b) } var y = [], x = {}, w = {}, k = {}, A = {}, E = {}; s.bfs({ roots: c, directed: t.directed, visit: function (e, t, n, r, i) { var o = e[0], a = o.id(); if (y[i] || (y[i] = []), y[i].push(o), x[a] = !0, w[a] = i, k[a] = n, A[a] = t, n) { var s = n.id(); (E[s] = E[s] || []).push(e) } } }); for (var S = [], $ = 0; $ < a.length; $++) { var C = a[$]; x[C.id()] || S.push(C) } for (var _ = 3 * S.length, O = 0; 0 !== S.length && O < _;) { for (var T = S.shift(), j = T.neighborhood().nodes(), P = !1, D = 0; D < j.length; D++) { var R = w[j[D].id()]; if (void 0 !== R) { y[R].push(T), P = !0; break } } P || S.push(T), O++ } for (; 0 !== S.length;) { var I = S.shift(); 0 === y.length && y.push([]), y[0].push(I) } var N = function () { for (var e = 0; e < y.length; e++)for (var t = y[e], n = 0; n < t.length; n++) { var r = t[n]; null != r ? r._private.scratch.breadthfirst = { depth: e, index: n } : (t.splice(n, 1), n--) } }; N(); for (var M = function (e) { for (var t = e.connectedEdges((function (t) { return t.data("target") === e.id() })), n = e._private.scratch.breadthfirst, r = 0, i = void 0, o = 0; o < t.length; o++) { var a = t[o].source()[0], s = a._private.scratch.breadthfirst; n.depth <= s.depth && r < s.depth && (r = s.depth, i = a) } return i }, z = 0; z < t.maximalAdjustments; z++) { for (var L = y.length, B = [], F = 0; F < L; F++)for (var q = y[F], V = q.length, U = 0; U < V; U++) { var H = q[U], G = H._private.scratch.breadthfirst, W = M(H); W && (G.intEle = W, B.push(H)) } for (var Y = 0; Y < B.length; Y++) { var X = B[Y], Z = X._private.scratch.breadthfirst, Q = Z.intEle._private.scratch.breadthfirst; y[Z.depth][Z.index] = null; for (var J = Q.depth + 1; J > y.length - 1;)y.push([]); y[J].push(X), Z.depth = J, Z.index = y[J].length - 1 } N() } var K = 0; if (t.avoidOverlap) for (var ee = 0; ee < a.length; ee++) { var te = a[ee].layoutDimensions(t), ne = te.w, re = te.h; K = Math.max(K, ne, re) } for (var ie = {}, oe = function (e) { if (ie[e.id()]) return ie[e.id()]; for (var t = e._private.scratch.breadthfirst.depth, n = e.neighborhood().nodes().not(":parent").intersection(a), r = 0, i = 0, o = 0; o < n.length; o++) { var s = n[o]._private.scratch.breadthfirst, l = s.index, c = s.depth, u = y[c].length; (t > c || 0 === t) && (r += l / u, i++) } return r /= i = Math.max(1, i), 0 === i && (r = void 0), ie[e.id()] = r, r }, ae = function (e, t) { return oe(e) - oe(t) }, se = 0; se < 3; se++) { for (var le = 0; le < y.length; le++)y[le] = y[le].sort(ae); N() } for (var ce = 0, ue = 0; ue < y.length; ue++)ce = Math.max(y[ue].length, ce); for (var de = l.x1 + l.w / 2, pe = l.x1 + l.h / 2, fe = function (e, n) { var r = e._private.scratch.breadthfirst, i = r.depth, o = r.index, a = y[i].length, s = Math.max(l.w / (a + 1), K), c = Math.max(l.h / (y.length + 1), K), u = Math.min(l.w / 2 / y.length, l.h / 2 / y.length); if (u = Math.max(u, K), t.circle) { if (t.circle) { var d = u * i + u - (y.length > 0 && y[0].length <= 3 ? u / 2 : 0), p = 2 * Math.PI / y[i].length * o; return 0 === i && 1 === y[0].length && (d = 1), { x: de + d * Math.cos(p), y: pe + d * Math.sin(p) } } return { x: de + (o + 1 - (a + 1) / 2) * s, y: (i + 1) * c } } var f = { x: de + (o + 1 - (a + 1) / 2) * s, y: (i + 1) * c }; return f }, he = {}, ge = y.length - 1; ge >= 0; ge--)for (var me = y[ge], ve = 0; ve < me.length; ve++) { var be = me[ve]; he[be.id()] = fe(be, y.length) } return a.layoutPositions(this, t, (function (e) { return he[e.id()] })), this }, e.exports = s }, function (e, t, n) { "use strict"; var r = n(1), i = n(2), o = n(0), a = { fit: !0, padding: 30, boundingBox: void 0, avoidOverlap: !0, nodeDimensionsIncludeLabels: !1, spacingFactor: void 0, radius: void 0, startAngle: 1.5 * Math.PI, sweep: void 0, clockwise: !0, sort: void 0, animate: !1, animationDuration: 500, animationEasing: void 0, animateFilter: function (e, t) { return !0 }, ready: void 0, stop: void 0, transform: function (e, t) { return t } }; function s(e) { this.options = r.extend({}, a, e) } s.prototype.run = function () { var e = this.options, t = e, n = e.cy, r = t.eles, a = void 0 !== t.counterclockwise ? !t.counterclockwise : t.clockwise, s = r.nodes().not(":parent"); t.sort && (s = s.sort(t.sort)); for (var l = i.makeBoundingBox(t.boundingBox ? t.boundingBox : { x1: 0, y1: 0, w: n.width(), h: n.height() }), c = l.x1 + l.w / 2, u = l.y1 + l.h / 2, d = (void 0 === t.sweep ? 2 * Math.PI - 2 * Math.PI / s.length : t.sweep) / Math.max(1, s.length - 1), p = void 0, f = 0, h = 0; h < s.length; h++) { var g = s[h].layoutDimensions(t), m = g.w, v = g.h; f = Math.max(f, m, v) } if (p = o.number(t.radius) ? t.radius : s.length <= 1 ? 0 : Math.min(l.h, l.w) / 2 - f, s.length > 1 && t.avoidOverlap) { f *= 1.75; var b = Math.cos(d) - Math.cos(0), y = Math.sin(d) - Math.sin(0), x = Math.sqrt(f * f / (b * b + y * y)); p = Math.max(x, p) } return s.layoutPositions(this, t, (function (e, n) { var r = t.startAngle + n * d * (a ? 1 : -1), i = p * Math.cos(r), o = p * Math.sin(r); return { x: c + i, y: u + o } })), this }, e.exports = s }, function (e, t, n) { "use strict"; var r = n(1), i = n(2), o = { fit: !0, padding: 30, startAngle: 1.5 * Math.PI, sweep: void 0, clockwise: !0, equidistant: !1, minNodeSpacing: 10, boundingBox: void 0, avoidOverlap: !0, nodeDimensionsIncludeLabels: !1, height: void 0, width: void 0, spacingFactor: void 0, concentric: function (e) { return e.degree() }, levelWidth: function (e) { return e.maxDegree() / 4 }, animate: !1, animationDuration: 500, animationEasing: void 0, animateFilter: function (e, t) { return !0 }, ready: void 0, stop: void 0, transform: function (e, t) { return t } }; function a(e) { this.options = r.extend({}, o, e) } a.prototype.run = function () { for (var e = this.options, t = e, n = void 0 !== t.counterclockwise ? !t.counterclockwise : t.clockwise, r = e.cy, o = t.eles.nodes().not(":parent"), a = i.makeBoundingBox(t.boundingBox ? t.boundingBox : { x1: 0, y1: 0, w: r.width(), h: r.height() }), s = a.x1 + a.w / 2, l = a.y1 + a.h / 2, c = [], u = (t.startAngle, 0), d = 0; d < o.length; d++) { var p, f = o[d]; p = t.concentric(f), c.push({ value: p, node: f }), f._private.scratch.concentric = p } o.updateStyle(); for (var h = 0; h < o.length; h++) { var g = o[h].layoutDimensions(t); u = Math.max(u, g.w, g.h) } c.sort((function (e, t) { return t.value - e.value })); for (var m = t.levelWidth(o), v = [[]], b = v[0], y = 0; y < c.length; y++) { var x = c[y]; b.length > 0 && Math.abs(b[0].value - x.value) >= m && (b = [], v.push(b)), b.push(x) } var w = u + t.minNodeSpacing; if (!t.avoidOverlap) { var k = v.length > 0 && v[0].length > 1, A = (Math.min(a.w, a.h) / 2 - w) / (v.length + k ? 1 : 0); w = Math.min(w, A) } for (var E = 0, S = 0; S < v.length; S++) { var $ = v[S], C = void 0 === t.sweep ? 2 * Math.PI - 2 * Math.PI / $.length : t.sweep, _ = $.dTheta = C / Math.max(1, $.length - 1); if ($.length > 1 && t.avoidOverlap) { var O = Math.cos(_) - Math.cos(0), T = Math.sin(_) - Math.sin(0), j = Math.sqrt(w * w / (O * O + T * T)); E = Math.max(j, E) } $.r = E, E += w } if (t.equidistant) { for (var P = 0, D = 0, R = 0; R < v.length; R++) { var I = v[R].r - D; P = Math.max(P, I) } D = 0; for (var N = 0; N < v.length; N++) { var M = v[N]; 0 === N && (D = M.r), M.r = D, D += P } } for (var z = {}, L = 0; L < v.length; L++)for (var B = v[L], F = B.dTheta, q = B.r, V = 0; V < B.length; V++) { var U = B[V], H = t.startAngle + (n ? 1 : -1) * F * V, G = { x: s + q * Math.cos(H), y: l + q * Math.sin(H) }; z[U.node.id()] = G } return o.layoutPositions(this, t, (function (e) { var t = e.id(); return z[t] })), this }, e.exports = a }, function (e, t, n) { "use strict"; var r, i = n(1), o = n(2), a = n(0), s = n(5), l = { ready: function () { }, stop: function () { }, animate: !0, animationEasing: void 0, animationDuration: void 0, animateFilter: function (e, t) { return !0 }, animationThreshold: 250, refresh: 20, fit: !0, padding: 30, boundingBox: void 0, nodeDimensionsIncludeLabels: !1, randomize: !1, componentSpacing: 40, nodeRepulsion: function (e) { return 2048 }, nodeOverlap: 4, idealEdgeLength: function (e) { return 32 }, edgeElasticity: function (e) { return 32 }, nestingFactor: 1.2, gravity: 1, numIter: 1e3, initialTemp: 1e3, coolingFactor: .99, minTemp: 1, weaver: !1 }; function c(e) { this.options = i.extend({}, l, e), this.options.layout = this } c.prototype.run = function () { var e = this.options, t = e.cy, n = this, o = this.thread, l = e.weaver ? e.weaver.Thread : null, c = { listeners: [], on: function (e, t) { return this.listeners.push({ event: e, callback: t }), this }, trigger: function (e) { a.string(e) && (e = { type: e }); var t = function (t) { t.callback(e) }; return this.listeners.filter((function (t) { return t.event === e.type })).forEach(t), this }, pass: function (e) { return this.pass = e, this }, run: function (e) { var t = this.pass; return new s((function (n) { n(e(t)) })) }, stop: function () { return this }, stopped: function () { return !0 } }; o && !o.stopped() || (o = this.thread = l ? new l : c), n.stopped = !1, !0 !== e.animate && !1 !== e.animate || n.emit({ type: "layoutstart", layout: n }), r = !0 === e.debug; var d = u(t, n, e); r && f(d), e.randomize && h(d, t); var p = Date.now(), m = !1, v = function (n) { n = n || {}, m && !n.next || !n.force && Date.now() - p < e.animationThreshold || (m = !0, i.requestAnimationFrame((function () { g(d, t, e), !0 === e.fit && t.fit(e.padding), m = !1, n.next && n.next() }))) }; o.on("message", (function (e) { var t = e.message; d.layoutNodes = t, v() })), o.pass({ layoutInfo: d, options: { animate: e.animate, refresh: e.refresh, componentSpacing: e.componentSpacing, nodeOverlap: e.nodeOverlap, nestingFactor: e.nestingFactor, gravity: e.gravity, numIter: e.numIter, initialTemp: e.initialTemp, coolingFactor: e.coolingFactor, minTemp: e.minTemp } }).run((function (e) { var t, n, r = e.layoutInfo, i = e.options, o = function (e, t) { for (var n = 0; n < e.graphSet.length; n++)for (var r = e.graphSet[n], i = r.length, o = 0; o < i; o++)for (var a = e.layoutNodes[e.idToIndex[r[o]]], l = o + 1; l < i; l++) { var c = e.layoutNodes[e.idToIndex[r[l]]]; s(a, c, e, t) } }, a = function (e) { return -e + 2 * e * Math.random() }, s = function (e, t, n, r) { if (e.cmptId === t.cmptId || n.isCompound) { var i = t.positionX - e.positionX, o = t.positionY - e.positionY; 0 === i && 0 === o && (i = a(1), o = a(1)); var s = l(e, t, i, o); if (s > 0) var c = (p = r.nodeOverlap * s) * i / (b = Math.sqrt(i * i + o * o)), d = p * o / b; else { var p, f = u(e, i, o), h = u(t, -1 * i, -1 * o), g = h.x - f.x, m = h.y - f.y, v = g * g + m * m, b = Math.sqrt(v); c = (p = (e.nodeRepulsion + t.nodeRepulsion) / v) * g / b, d = p * m / b } e.isLocked || (e.offsetX -= c, e.offsetY -= d), t.isLocked || (t.offsetX += c, t.offsetY += d) } }, l = function (e, t, n, r) { if (n > 0) var i = e.maxX - t.minX; else i = t.maxX - e.minX; if (r > 0) var o = e.maxY - t.minY; else o = t.maxY - e.minY; return i >= 0 && o >= 0 ? Math.sqrt(i * i + o * o) : 0 }, u = function (e, t, n) { var r = e.positionX, i = e.positionY, o = e.height || 1, a = e.width || 1, s = n / t, l = o / a, c = {}; return 0 === t && 0 < n || 0 === t && 0 > n ? (c.x = r, c.y = i + o / 2, c) : 0 < t && -1 * l <= s && s <= l ? (c.x = r + a / 2, c.y = i + a * n / 2 / t, c) : 0 > t && -1 * l <= s && s <= l ? (c.x = r - a / 2, c.y = i - a * n / 2 / t, c) : 0 < n && (s <= -1 * l || s >= l) ? (c.x = r + o * t / 2 / n, c.y = i + o / 2, c) : 0 > n && (s <= -1 * l || s >= l) ? (c.x = r - o * t / 2 / n, c.y = i - o / 2, c) : c }, d = function (e, t) { for (var n = 0; n < e.edgeSize; n++) { var r = e.layoutEdges[n], i = e.idToIndex[r.sourceId], o = e.layoutNodes[i], a = e.idToIndex[r.targetId], s = e.layoutNodes[a], l = s.positionX - o.positionX, c = s.positionY - o.positionY; if (0 !== l || 0 !== c) { var d = u(o, l, c), p = u(s, -1 * l, -1 * c), f = p.x - d.x, h = p.y - d.y, g = Math.sqrt(f * f + h * h), m = Math.pow(r.idealLength - g, 2) / r.elasticity; if (0 !== g) var v = m * f / g, b = m * h / g; else v = 0, b = 0; o.isLocked || (o.offsetX += v, o.offsetY += b), s.isLocked || (s.offsetX -= v, s.offsetY -= b) } } }, p = function (e, t) { for (var n = 0; n < e.graphSet.length; n++) { var r = e.graphSet[n], i = r.length; if (0 === n) var o = e.clientHeight / 2, a = e.clientWidth / 2; else { var s = e.layoutNodes[e.idToIndex[r[0]]], l = e.layoutNodes[e.idToIndex[s.parentId]]; o = l.positionX, a = l.positionY } for (var c = 0; c < i; c++) { var u = e.layoutNodes[e.idToIndex[r[c]]]; if (!u.isLocked) { var d = o - u.positionX, p = a - u.positionY, f = Math.sqrt(d * d + p * p); if (f > 1) { var h = t.gravity * d / f, g = t.gravity * p / f; u.offsetX += h, u.offsetY += g } } } } }, f = function (e, t) { var n = [], r = 0, i = -1; for (n.push.apply(n, e.graphSet[0]), i += e.graphSet[0].length; r <= i;) { var o = n[r++], a = e.idToIndex[o], s = e.layoutNodes[a], l = s.children; if (0 < l.length && !s.isLocked) { for (var c = s.offsetX, u = s.offsetY, d = 0; d < l.length; d++) { var p = e.layoutNodes[e.idToIndex[l[d]]]; p.offsetX += c, p.offsetY += u, n[++i] = l[d] } s.offsetX = 0, s.offsetY = 0 } } }, h = function (e, t) { for (var n = 0; n < e.nodeSize; n++)0 < (i = e.layoutNodes[n]).children.length && (i.maxX = void 0, i.minX = void 0, i.maxY = void 0, i.minY = void 0); for (n = 0; n < e.nodeSize; n++)if (!(0 < (i = e.layoutNodes[n]).children.length || i.isLocked)) { var r = g(i.offsetX, i.offsetY, e.temperature); i.positionX += r.x, i.positionY += r.y, i.offsetX = 0, i.offsetY = 0, i.minX = i.positionX - i.width, i.maxX = i.positionX + i.width, i.minY = i.positionY - i.height, i.maxY = i.positionY + i.height, m(i, e) } for (n = 0; n < e.nodeSize; n++) { var i; 0 < (i = e.layoutNodes[n]).children.length && !i.isLocked && (i.positionX = (i.maxX + i.minX) / 2, i.positionY = (i.maxY + i.minY) / 2, i.width = i.maxX - i.minX, i.height = i.maxY - i.minY) } }, g = function (e, t, n) { var r = Math.sqrt(e * e + t * t); if (r > n) var i = { x: n * e / r, y: n * t / r }; else i = { x: e, y: t }; return i }, m = function e(t, n) { var r = t.parentId; if (null != r) { var i = n.layoutNodes[n.idToIndex[r]], o = !1; return (null == i.maxX || t.maxX + i.padRight > i.maxX) && (i.maxX = t.maxX + i.padRight, o = !0), (null == i.minX || t.minX - i.padLeft < i.minX) && (i.minX = t.minX - i.padLeft, o = !0), (null == i.maxY || t.maxY + i.padBottom > i.maxY) && (i.maxY = t.maxY + i.padBottom, o = !0), (null == i.minY || t.minY - i.padTop < i.minY) && (i.minY = t.minY - i.padTop, o = !0), o ? e(i, n) : void 0 } }, v = function (e) { return function (e, t, n) { o(e, t), d(e, t), p(e, t), f(e, t), h(e, t) }(r, i), r.temperature = r.temperature * i.coolingFactor, !(r.temperature < i.minTemp) }, b = 0; do { for (var y = 0; y < i.refresh && b < i.numIter;) { var x; if (!(x = v())) break; y++, b++ } !0 === i.animate && (t = r.layoutNodes, n = void 0, n = { type: "message", message: t }, c.trigger(n)) } while (x && b + 1 < i.numIter); return function (e, t) { for (var n = r.layoutNodes, i = [], o = 0; o < n.length; o++) { var a = n[o], s = a.cmptId; (i[s] = i[s] || []).push(a) } var l = 0; for (o = 0; o < i.length; o++)if (m = i[o]) { m.x1 = 1 / 0, m.x2 = -1 / 0, m.y1 = 1 / 0, m.y2 = -1 / 0; for (var c = 0; c < m.length; c++) { var u = m[c]; m.x1 = Math.min(m.x1, u.positionX - u.width / 2), m.x2 = Math.max(m.x2, u.positionX + u.width / 2), m.y1 = Math.min(m.y1, u.positionY - u.height / 2), m.y2 = Math.max(m.y2, u.positionY + u.height / 2) } m.w = m.x2 - m.x1, m.h = m.y2 - m.y1, l += m.w * m.h } i.sort((function (e, t) { return t.w * t.h - e.w * e.h })); var d = 0, p = 0, f = 0, h = 0, g = Math.sqrt(l) * r.clientWidth / r.clientHeight; for (o = 0; o < i.length; o++) { var m; if (m = i[o]) { for (c = 0; c < m.length; c++)(u = m[c]).isLocked || (u.positionX += d, u.positionY += p); d += m.w + t.componentSpacing, f += m.w + t.componentSpacing, h = Math.max(h, m.h), f > g && (p += h + t.componentSpacing, d = 0, f = 0, h = 0) } } }(0, i), r })).then((function (e) { d.layoutNodes = e.layoutNodes, o.stop(), b() })); var b = function () { !0 === e.animate || !1 === e.animate ? v({ force: !0, next: function () { n.one("layoutstop", e.stop), n.emit({ type: "layoutstop", layout: n }) } }) : e.eles.nodes().layoutPositions(n, e, (function (e) { var t = d.layoutNodes[d.idToIndex[e.data("id")]]; return { x: t.positionX, y: t.positionY } })) }; return this }, c.prototype.stop = function () { return this.stopped = !0, this.thread && this.thread.stop(), this.emit("layoutstop"), this }, c.prototype.destroy = function () { return this.thread && this.thread.stop(), this }; var u = function (e, t, n) { for (var r = n.eles.edges(), i = n.eles.nodes(), s = { isCompound: e.hasCompoundNodes(), layoutNodes: [], idToIndex: {}, nodeSize: i.size(), graphSet: [], indexToGraph: [], layoutEdges: [], edgeSize: r.size(), temperature: n.initialTemp, clientWidth: e.width(), clientHeight: e.width(), boundingBox: o.makeBoundingBox(n.boundingBox ? n.boundingBox : { x1: 0, y1: 0, w: e.width(), h: e.height() }) }, l = n.eles.components(), c = {}, u = 0; u < l.length; u++)for (var p = l[u], f = 0; f < p.length; f++)c[p[f].id()] = u; for (u = 0; u < s.nodeSize; u++) { var h = (y = i[u]).layoutDimensions(n); (I = {}).isLocked = y.locked(), I.id = y.data("id"), I.parentId = y.data("parent"), I.cmptId = c[y.id()], I.children = [], I.positionX = y.position("x"), I.positionY = y.position("y"), I.offsetX = 0, I.offsetY = 0, I.height = h.w, I.width = h.h, I.maxX = I.positionX + I.width / 2, I.minX = I.positionX - I.width / 2, I.maxY = I.positionY + I.height / 2, I.minY = I.positionY - I.height / 2, I.padLeft = parseFloat(y.style("padding")), I.padRight = parseFloat(y.style("padding")), I.padTop = parseFloat(y.style("padding")), I.padBottom = parseFloat(y.style("padding")), I.nodeRepulsion = a.fn(n.nodeRepulsion) ? n.nodeRepulsion(y) : n.nodeRepulsion, s.layoutNodes.push(I), s.idToIndex[I.id] = u } var g = [], m = 0, v = -1, b = []; for (u = 0; u < s.nodeSize; u++) { var y, x = (y = s.layoutNodes[u]).parentId; null != x ? s.layoutNodes[s.idToIndex[x]].children.push(y.id) : (g[++v] = y.id, b.push(y.id)) } for (s.graphSet.push(b); m <= v;) { var w = g[m++], k = s.idToIndex[w], A = s.layoutNodes[k].children; if (A.length > 0) for (s.graphSet.push(A), u = 0; u < A.length; u++)g[++v] = A[u] } for (u = 0; u < s.graphSet.length; u++) { var E = s.graphSet[u]; for (f = 0; f < E.length; f++) { var S = s.idToIndex[E[f]]; s.indexToGraph[S] = u } } for (u = 0; u < s.edgeSize; u++) { var $ = r[u], C = {}; C.id = $.data("id"), C.sourceId = $.data("source"), C.targetId = $.data("target"); var _ = a.fn(n.idealEdgeLength) ? n.idealEdgeLength($) : n.idealEdgeLength, O = a.fn(n.edgeElasticity) ? n.edgeElasticity($) : n.edgeElasticity, T = s.idToIndex[C.sourceId], j = s.idToIndex[C.targetId]; if (s.indexToGraph[T] != s.indexToGraph[j]) { for (var P = d(C.sourceId, C.targetId, s), D = s.graphSet[P], R = 0, I = s.layoutNodes[T]; -1 === D.indexOf(I.id);)I = s.layoutNodes[s.idToIndex[I.parentId]], R++; for (I = s.layoutNodes[j]; -1 === D.indexOf(I.id);)I = s.layoutNodes[s.idToIndex[I.parentId]], R++; _ *= R * n.nestingFactor } C.idealLength = _, C.elasticity = O, s.layoutEdges.push(C) } return s }, d = function (e, t, n) { var r = p(e, t, 0, n); return 2 > r.count ? 0 : r.graph }, p = function e(t, n, r, i) { var o = i.graphSet[r]; if (-1 < o.indexOf(t) && -1 < o.indexOf(n)) return { count: 2, graph: r }; for (var a = 0, s = 0; s < o.length; s++) { var l = o[s], c = i.idToIndex[l], u = i.layoutNodes[c].children; if (0 !== u.length) { var d = e(t, n, i.indexToGraph[i.idToIndex[u[0]]], i); if (0 !== d.count) { if (1 !== d.count) return d; if (2 == ++a) break } } } return { count: a, graph: r } }, f = function (e) { if (r) { console.debug("layoutNodes:"); for (var t = 0; t < e.nodeSize; t++) { var n = e.layoutNodes[t], i = "\nindex: " + t + "\nId: " + n.id + "\nChildren: " + n.children.toString() + "\nparentId: " + n.parentId + "\npositionX: " + n.positionX + "\npositionY: " + n.positionY + "\nOffsetX: " + n.offsetX + "\nOffsetY: " + n.offsetY + "\npadLeft: " + n.padLeft + "\npadRight: " + n.padRight + "\npadTop: " + n.padTop + "\npadBottom: " + n.padBottom; console.debug(i) } for (var t in console.debug("idToIndex"), e.idToIndex) console.debug("Id: " + t + "\nIndex: " + e.idToIndex[t]); console.debug("Graph Set"); var o = e.graphSet; for (t = 0; t < o.length; t++)console.debug("Set : " + t + ": " + o[t].toString()); for (i = "IndexToGraph", t = 0; t < e.indexToGraph.length; t++)i += "\nIndex : " + t + " Graph: " + e.indexToGraph[t]; for (console.debug(i), i = "Layout Edges", t = 0; t < e.layoutEdges.length; t++) { var a = e.layoutEdges[t]; i += "\nEdge Index: " + t + " ID: " + a.id + " SouceID: " + a.sourceId + " TargetId: " + a.targetId + " Ideal Length: " + a.idealLength } console.debug(i), i = "nodeSize: " + e.nodeSize, i += "\nedgeSize: " + e.edgeSize, i += "\ntemperature: " + e.temperature, console.debug(i) } }, h = function (e, t) { for (var n = e.clientWidth, r = e.clientHeight, i = 0; i < e.nodeSize; i++) { var o = e.layoutNodes[i]; 0 !== o.children.length || o.isLocked || (o.positionX = Math.random() * n, o.positionY = Math.random() * r) } }, g = function (e, t, n) { var r = n.layout, i = n.eles.nodes(), o = e.boundingBox, a = { x1: 1 / 0, x2: -1 / 0, y1: 1 / 0, y2: -1 / 0 }; n.boundingBox && (i.forEach((function (t) { var n = e.layoutNodes[e.idToIndex[t.data("id")]]; a.x1 = Math.min(a.x1, n.positionX), a.x2 = Math.max(a.x2, n.positionX), a.y1 = Math.min(a.y1, n.positionY), a.y2 = Math.max(a.y2, n.positionY) })), a.w = a.x2 - a.x1, a.h = a.y2 - a.y1), i.positions((function (t, r) { var i = e.layoutNodes[e.idToIndex[t.data("id")]]; if (n.boundingBox) { var s = (i.positionX - a.x1) / a.w, l = (i.positionY - a.y1) / a.h; return { x: o.x1 + s * o.w, y: o.y1 + l * o.h } } return { x: i.positionX, y: i.positionY } })), !0 !== e.ready && (e.ready = !0, r.one("layoutready", n.ready), r.emit({ type: "layoutready", layout: this })) }; e.exports = c }, function (e, t, n) { "use strict"; var r = n(1), i = n(2), o = { fit: !0, padding: 30, boundingBox: void 0, avoidOverlap: !0, avoidOverlapPadding: 10, nodeDimensionsIncludeLabels: !1, spacingFactor: void 0, condense: !1, rows: void 0, cols: void 0, position: function (e) { }, sort: void 0, animate: !1, animationDuration: 500, animationEasing: void 0, animateFilter: function (e, t) { return !0 }, ready: void 0, stop: void 0, transform: function (e, t) { return t } }; function a(e) { this.options = r.extend({}, o, e) } a.prototype.run = function () { var e = this.options, t = e, n = e.cy, r = t.eles.nodes().not(":parent"); t.sort && (r = r.sort(t.sort)); var o = i.makeBoundingBox(t.boundingBox ? t.boundingBox : { x1: 0, y1: 0, w: n.width(), h: n.height() }); if (0 === o.h || 0 === o.w) r.layoutPositions(this, t, (function (e) { return { x: o.x1, y: o.y1 } })); else { var a = r.size(), s = Math.sqrt(a * o.h / o.w), l = Math.round(s), c = Math.round(o.w / o.h * s), u = function (e) { if (null == e) return Math.min(l, c); Math.min(l, c) == l ? l = e : c = e }, d = function (e) { if (null == e) return Math.max(l, c); Math.max(l, c) == l ? l = e : c = e }, p = t.rows, f = null != t.cols ? t.cols : t.columns; if (null != p && null != f) l = p, c = f; else if (null != p && null == f) l = p, c = Math.ceil(a / l); else if (null == p && null != f) c = f, l = Math.ceil(a / c); else if (c * l > a) { var h = u(), g = d(); (h - 1) * g >= a ? u(h - 1) : (g - 1) * h >= a && d(g - 1) } else for (; c * l < a;) { var m = u(), v = d(); (v + 1) * m >= a ? d(v + 1) : u(m + 1) } var b = o.w / c, y = o.h / l; if (t.condense && (b = 0, y = 0), t.avoidOverlap) for (var x = 0; x < r.length; x++) { var w = r[x], k = w._private.position; null != k.x && null != k.y || (k.x = 0, k.y = 0); var A = w.layoutDimensions(t), E = t.avoidOverlapPadding, S = A.w + E, $ = A.h + E; b = Math.max(b, S), y = Math.max(y, $) } for (var C = {}, _ = function (e, t) { return !!C["c-" + e + "-" + t] }, O = function (e, t) { C["c-" + e + "-" + t] = !0 }, T = 0, j = 0, P = function () { ++j >= c && (j = 0, T++) }, D = {}, R = 0; R < r.length; R++) { var I = r[R], N = t.position(I); if (N && (void 0 !== N.row || void 0 !== N.col)) { var M = { row: N.row, col: N.col }; if (void 0 === M.col) for (M.col = 0; _(M.row, M.col);)M.col++; else if (void 0 === M.row) for (M.row = 0; _(M.row, M.col);)M.row++; D[I.id()] = M, O(M.row, M.col) } } r.layoutPositions(this, t, (function (e, t) { var n = void 0, r = void 0; if (e.locked() || e.isParent()) return !1; var i = D[e.id()]; if (i) n = i.col * b + b / 2 + o.x1, r = i.row * y + y / 2 + o.y1; else { for (; _(T, j);)P(); n = j * b + b / 2 + o.x1, r = T * y + y / 2 + o.y1, O(T, j), P() } return { x: n, y: r } })) } return this }, e.exports = a }, function (e, t, n) { "use strict"; var r = n(1), i = { ready: function () { }, stop: function () { } }; function o(e) { this.options = r.extend({}, i, e) } o.prototype.run = function () { var e = this.options, t = e.eles; return e.cy, this.emit("layoutstart"), t.nodes().positions((function () { return { x: 0, y: 0 } })), this.one("layoutready", e.ready), this.emit("layoutready"), this.one("layoutstop", e.stop), this.emit("layoutstop"), this }, o.prototype.stop = function () { return this }, e.exports = o }, function (e, t, n) { "use strict"; var r = n(1), i = n(0), o = { positions: void 0, zoom: void 0, pan: void 0, fit: !0, padding: 30, animate: !1, animationDuration: 500, animationEasing: void 0, animateFilter: function (e, t) { return !0 }, ready: void 0, stop: void 0, transform: function (e, t) { return t } }; function a(e) { this.options = r.extend({}, o, e) } a.prototype.run = function () { var e = this.options, t = e.eles.nodes(), n = i.fn(e.positions); return t.layoutPositions(this, e, (function (t, r) { var i = function (t) { if (null == e.positions) return null; if (n) return e.positions(t); var r = e.positions[t._private.data.id]; return null == r ? null : r }(t); return !t.locked() && null != i && i })), this }, e.exports = a }, function (e, t, n) { "use strict"; var r = n(1), i = n(2), o = { fit: !0, padding: 30, boundingBox: void 0, animate: !1, animationDuration: 500, animationEasing: void 0, animateFilter: function (e, t) { return !0 }, ready: void 0, stop: void 0, transform: function (e, t) { return t } }; function a(e) { this.options = r.extend({}, o, e) } a.prototype.run = function () { var e = this.options, t = e.cy, n = e.eles.nodes().not(":parent"), r = i.makeBoundingBox(e.boundingBox ? e.boundingBox : { x1: 0, y1: 0, w: t.width(), h: t.height() }); return n.layoutPositions(this, e, (function (e, t) { return { x: r.x1 + Math.round(Math.random() * r.w), y: r.y1 + Math.round(Math.random() * r.h) } })), this }, e.exports = a }, function (e, t, n) { "use strict"; e.exports = [{ name: "null", impl: n(106) }, { name: "base", impl: n(107) }, { name: "canvas", impl: n(123) }] }, function (e, t, n) { "use strict"; function r(e) { this.options = e, this.notifications = 0 } var i = function () { }; r.prototype = { recalculateRenderedStyle: i, notify: function () { this.notifications++ }, init: i }, e.exports = r }, function (e, t, n) { "use strict"; var r = n(0), i = n(1), o = n(3), a = function (e) { this.init(e) }, s = a.prototype; s.clientFunctions = ["redrawHint", "render", "renderTo", "matchCanvasSize", "nodeShapeImpl", "arrowShapeImpl"], s.init = function (e) { var t = this; t.options = e, t.cy = e.cy; var n = t.container = e.cy.container(); if (o) { var r = o.document, a = r.head, s = "__________cytoscape_container", l = null != r.getElementById("__________cytoscape_stylesheet"); if (n.className.indexOf(s) < 0 && (n.className = (n.className || "") + " " + s), !l) { var c = r.createElement("style"); c.id = "__________cytoscape_stylesheet", c.innerHTML = "." + s + " { position: relative; }", a.insertBefore(c, a.children[0]) } "static" === o.getComputedStyle(n).getPropertyValue("position") && i.error("A Cytoscape container has style position:static and so can not use UI extensions properly") } t.selection = [void 0, void 0, void 0, void 0, 0], t.bezierProjPcts = [.05, .225, .4, .5, .6, .775, .95], t.hoverData = { down: null, last: null, downTime: null, triggerMode: null, dragging: !1, initialPan: [null, null], capture: !1 }, t.dragData = { possibleDragElements: [] }, t.touchData = { start: null, capture: !1, startPosition: [null, null, null, null, null, null], singleTouchStartTime: null, singleTouchMoved: !0, now: [null, null, null, null, null, null], earlier: [null, null, null, null, null, null] }, t.redraws = 0, t.showFps = e.showFps, t.debug = e.debug, t.hideEdgesOnViewport = e.hideEdgesOnViewport, t.hideLabelsOnViewport = e.hideLabelsOnViewport, t.textureOnViewport = e.textureOnViewport, t.wheelSensitivity = e.wheelSensitivity, t.motionBlurEnabled = e.motionBlur, t.forcedPixelRatio = e.pixelRatio, t.motionBlur = e.motionBlur, t.motionBlurOpacity = e.motionBlurOpacity, t.motionBlurTransparency = 1 - t.motionBlurOpacity, t.motionBlurPxRatio = 1, t.mbPxRBlurry = 1, t.minMbLowQualFrames = 4, t.fullQualityMb = !1, t.clearedForMotionBlur = [], t.desktopTapThreshold = e.desktopTapThreshold, t.desktopTapThreshold2 = e.desktopTapThreshold * e.desktopTapThreshold, t.touchTapThreshold = e.touchTapThreshold, t.touchTapThreshold2 = e.touchTapThreshold * e.touchTapThreshold, t.tapholdDuration = 500, t.bindings = [], t.beforeRenderCallbacks = [], t.beforeRenderPriorities = { animations: 400, eleCalcs: 300, eleTxrDeq: 200, lyrTxrDeq: 100 }, t.registerNodeShapes(), t.registerArrowShapes(), t.registerCalculationListeners() }, s.notify = function (e) { var t, n = this; if (!this.destroyed) { t = r.array(e.type) ? e.type : [e.type]; for (var i = {}, o = 0; o < t.length; o++)i[t[o]] = !0; i.init ? n.load() : i.destroy ? n.destroy() : ((i.add || i.remove || i.load || i.zorder) && n.invalidateCachedZSortedEles(), i.viewport && n.redrawHint("select", !0), (i.load || i.resize) && (n.invalidateContainerClientCoordsCache(), n.matchCanvasSize(n.container)), n.redrawHint("eles", !0), n.redrawHint("drag", !0), this.startRenderLoop(), this.redraw()) } }, s.destroy = function () { var e = this; e.destroyed = !0, e.cy.stopAnimationLoop(); for (var t = 0; t < e.bindings.length; t++) { var n = e.bindings[t], r = n.target; (r.off || r.removeEventListener).apply(r, n.args) } if (e.bindings = [], e.beforeRenderCallbacks = [], e.onUpdateEleCalcsFns = [], e.removeObserver && e.removeObserver.disconnect(), e.styleObserver && e.styleObserver.disconnect(), e.labelCalcDiv) try { document.body.removeChild(e.labelCalcDiv) } catch (e) { } }, [n(108), n(109), n(119), n(120), n(121), n(122)].forEach((function (e) { i.extend(s, e) })), e.exports = a }, function (e, t, n) { "use strict"; var r = n(2), i = n(0), o = n(1), a = { arrowShapeWidth: .3, registerArrowShapes: function () { var e = this.arrowShapes = {}, t = this, n = function (e, t, n, r, i, o, a) { var s = i.x - n / 2 - a, l = i.x + n / 2 + a, c = i.y - n / 2 - a, u = i.y + n / 2 + a; return s <= e && e <= l && c <= t && t <= u }, a = function (e, t, n, r, i) { var o = e * Math.cos(r) - t * Math.sin(r), a = (e * Math.sin(r) + t * Math.cos(r)) * n; return { x: o * n + i.x, y: a + i.y } }, s = function (e, t, n, r) { for (var i = [], o = 0; o < e.length; o += 2) { var s = e[o], l = e[o + 1]; i.push(a(s, l, t, n, r)) } return i }, l = function (e) { for (var t = [], n = 0; n < e.length; n++) { var r = e[n]; t.push(r.x, r.y) } return t }, c = function (e) { return e.pstyle("width").pfValue * e.pstyle("arrow-scale").pfValue * 2 }, u = function (a, u) { i.string(u) && (u = e[u]), e[a] = o.extend({ name: a, points: [-.15, -.3, .15, -.3, .15, .3, -.15, .3], collide: function (e, t, n, i, o, a) { var c = l(s(this.points, n + 2 * a, i, o)); return r.pointInsidePolygonPoints(e, t, c) }, roughCollide: n, draw: function (e, n, r, i) { var o = s(this.points, n, r, i); t.arrowShapeImpl("polygon")(e, o) }, spacing: function (e) { return 0 }, gap: c }, u) }; u("none", { collide: o.falsify, roughCollide: o.falsify, draw: o.noop, spacing: o.zeroify, gap: o.zeroify }), u("triangle", { points: [-.15, -.3, 0, 0, .15, -.3] }), u("arrow", "triangle"), u("triangle-backcurve", { points: e.triangle.points, controlPoint: [0, -.15], roughCollide: n, draw: function (e, n, r, i, o) { var l = s(this.points, n, r, i), c = this.controlPoint, u = a(c[0], c[1], n, r, i); t.arrowShapeImpl(this.name)(e, l, u) }, gap: function (e) { return .8 * c(e) } }), u("triangle-tee", { points: [-.15, -.3, 0, 0, .15, -.3, -.15, -.3], pointsTee: [-.15, -.4, -.15, -.5, .15, -.5, .15, -.4], collide: function (e, t, n, i, o, a, c) { var u = l(s(this.points, n + 2 * c, i, o)), d = l(s(this.pointsTee, n + 2 * c, i, o)); return r.pointInsidePolygonPoints(e, t, u) || r.pointInsidePolygonPoints(e, t, d) }, draw: function (e, n, r, i, o) { var a = s(this.points, n, r, i), l = s(this.pointsTee, n, r, i); t.arrowShapeImpl(this.name)(e, a, l) } }), u("triangle-cross", { points: [-.15, -.3, 0, 0, .15, -.3, -.15, -.3], baseCrossLinePts: [-.15, -.4, -.15, -.4, .15, -.4, .15, -.4], crossLinePts: function (e, t) { var n = this.baseCrossLinePts.slice(), r = t / e; return n[3] = n[3] - r, n[5] = n[5] - r, n }, collide: function (e, t, n, i, o, a, c) { var u = l(s(this.points, n + 2 * c, i, o)), d = l(s(this.crossLinePts(n, a), n + 2 * c, i, o)); return r.pointInsidePolygonPoints(e, t, u) || r.pointInsidePolygonPoints(e, t, d) }, draw: function (e, n, r, i, o) { var a = s(this.points, n, r, i), l = s(this.crossLinePts(n, o), n, r, i); t.arrowShapeImpl(this.name)(e, a, l) } }), u("vee", { points: [-.15, -.3, 0, 0, .15, -.3, 0, -.15], gap: function (e) { return .525 * c(e) } }), u("circle", { radius: .15, collide: function (e, t, n, r, i, o, a) { var s = i; return Math.pow(s.x - e, 2) + Math.pow(s.y - t, 2) <= Math.pow((n + 2 * a) * this.radius, 2) }, draw: function (e, n, r, i, o) { t.arrowShapeImpl(this.name)(e, i.x, i.y, this.radius * n) }, spacing: function (e) { return t.getArrowWidth(e.pstyle("width").pfValue, e.pstyle("arrow-scale").value) * this.radius } }), u("tee", { points: [-.15, 0, -.15, -.1, .15, -.1, .15, 0], spacing: function (e) { return 1 }, gap: function (e) { return 1 } }), u("square", { points: [-.15, 0, .15, 0, .15, -.3, -.15, -.3] }), u("diamond", { points: [-.15, -.15, 0, -.3, .15, -.15, 0, 0], gap: function (e) { return e.pstyle("width").pfValue * e.pstyle("arrow-scale").value } }) } }; e.exports = a }, function (e, t, n) { "use strict"; var r = n(1), i = {};[n(110), n(111), n(112), n(113), n(114), n(115), n(116), n(117), n(118)].forEach((function (e) { r.extend(i, e) })), e.exports = i }, function (e, t, n) { "use strict"; var r = n(3), i = n(2), o = n(1), a = (r = n(3), { projectIntoViewport: function (e, t) { var n = this.cy, r = this.findContainerClientCoords(), i = r[0], o = r[1], a = r[4], s = n.pan(), l = n.zoom(); return [((e - i) / a - s.x) / l, ((t - o) / a - s.y) / l] }, findContainerClientCoords: function () { if (this.containerBB) return this.containerBB; var e = this.container, t = e.getBoundingClientRect(), n = r.getComputedStyle(e), i = function (e) { return parseFloat(n.getPropertyValue(e)) }, o = i("padding-left"), a = i("padding-right"), s = i("padding-top"), l = i("padding-bottom"), c = i("border-left-width"), u = i("border-right-width"), d = i("border-top-width"), p = (i("border-bottom-width"), e.clientWidth), f = e.clientHeight, h = o + a, g = s + l, m = c + u, v = t.width / (p + m), b = p - h, y = f - g, x = (t.width, t.height, t.left + o + c), w = t.top + s + d; return this.containerBB = [x, w, b, y, v] }, invalidateContainerClientCoordsCache: function () { this.containerBB = null }, findNearestElement: function (e, t, n, r) { return this.findNearestElements(e, t, n, r)[0] }, findNearestElements: function (e, t, n, r) { var a, s, l = this, c = this, u = c.getCachedZSortedEles(), d = [], p = c.cy.zoom(), f = c.cy.hasCompoundNodes(), h = (r ? 24 : 8) / p, g = (r ? 8 : 2) / p, m = (r ? 8 : 2) / p, v = 1 / 0; function b(e, t) { if (e.isNode()) { if (s) return; s = e, d.push(e) } if (e.isEdge() && (null == t || t < v)) if (a) { if (a.pstyle("z-index").value === e.pstyle("z-index").value) for (var n = 0; n < d.length; n++)if (d[n].isEdge()) { d[n] = e, a = e, v = null != t ? t : v; break } } else d.push(e), a = e, v = null != t ? t : v } function y(n) { var r = n.outerWidth() + 2 * g, i = n.outerHeight() + 2 * g, o = r / 2, a = i / 2, s = n.position(); if (s.x - o <= e && e <= s.x + o && s.y - a <= t && t <= s.y + a && c.nodeShapes[l.getNodeShape(n)].checkPoint(e, t, 0, r, i, s.x, s.y)) return b(n, 0), !0 } function x(n) { var r, o = n._private, a = o.rscratch, s = n.pstyle("width").pfValue, u = n.pstyle("arrow-scale").value, p = s / 2 + h, g = p * p, m = 2 * p, v = o.source, x = o.target; if ("segments" === a.edgeType || "straight" === a.edgeType || "haystack" === a.edgeType) { for (var w = a.allpts, k = 0; k + 3 < w.length; k += 2)if (i.inLineVicinity(e, t, w[k], w[k + 1], w[k + 2], w[k + 3], m) && g > (r = i.sqdistToFiniteLine(e, t, w[k], w[k + 1], w[k + 2], w[k + 3]))) return b(n, r), !0 } else if ("bezier" === a.edgeType || "multibezier" === a.edgeType || "self" === a.edgeType || "compound" === a.edgeType) for (w = a.allpts, k = 0; k + 5 < a.allpts.length; k += 4)if (i.inBezierVicinity(e, t, w[k], w[k + 1], w[k + 2], w[k + 3], w[k + 4], w[k + 5], m) && g > (r = i.sqdistToQuadraticBezier(e, t, w[k], w[k + 1], w[k + 2], w[k + 3], w[k + 4], w[k + 5]))) return b(n, r), !0; v = v || o.source, x = x || o.target; var A = l.getArrowWidth(s, u), E = [{ name: "source", x: a.arrowStartX, y: a.arrowStartY, angle: a.srcArrowAngle }, { name: "target", x: a.arrowEndX, y: a.arrowEndY, angle: a.tgtArrowAngle }, { name: "mid-source", x: a.midX, y: a.midY, angle: a.midsrcArrowAngle }, { name: "mid-target", x: a.midX, y: a.midY, angle: a.midtgtArrowAngle }]; for (k = 0; k < E.length; k++) { var S = E[k], $ = c.arrowShapes[n.pstyle(S.name + "-arrow-shape").value], C = n.pstyle("width").pfValue; if ($.roughCollide(e, t, A, S.angle, { x: S.x, y: S.y }, C, h) && $.collide(e, t, A, S.angle, { x: S.x, y: S.y }, C, h)) return b(n), !0 } f && d.length > 0 && (y(v), y(x)) } function w(e, t, n) { return o.getPrefixedProperty(e, t, n) } function k(n, r) { var o, a = n._private, s = m; o = r ? r + "-" : ""; var l = n.pstyle(o + "label").value; if ("yes" === n.pstyle("text-events").strValue && l) { var c = a.rstyle, u = n.pstyle("text-border-width").pfValue, d = n.pstyle("text-background-padding").pfValue, p = w(c, "labelWidth", r) + u + 2 * s + 2 * d, f = w(c, "labelHeight", r) + u + 2 * s + 2 * d, h = w(c, "labelX", r), g = w(c, "labelY", r), v = w(a.rscratch, "labelAngle", r), y = h - p / 2, x = h + p / 2, k = g - f / 2, A = g + f / 2; if (v) { var E = Math.cos(v), S = Math.sin(v), $ = function (e, t) { return { x: (e -= h) * E - (t -= g) * S + h, y: e * S + t * E + g } }, C = $(y, k), _ = $(y, A), O = $(x, k), T = $(x, A), j = [C.x, C.y, O.x, O.y, T.x, T.y, _.x, _.y]; if (i.pointInsidePolygonPoints(e, t, j)) return b(n), !0 } else { var P = { w: p, h: f, x1: y, x2: x, y1: k, y2: A }; if (i.inBoundingBox(P, e, t)) return b(n), !0 } } } n && (u = u.interactive); for (var A = u.length - 1; A >= 0; A--) { var E = u[A]; E.isNode() ? y(E) || k(E) : x(E) || k(E) || k(E, "source") || k(E, "target") } return d }, getAllInBox: function (e, t, n, r) { var o = this.getCachedZSortedEles().interactive, a = [], s = Math.min(e, n), l = Math.max(e, n), c = Math.min(t, r), u = Math.max(t, r); e = s, n = l, t = c, r = u; for (var d = i.makeBoundingBox({ x1: e, y1: t, x2: n, y2: r }), p = 0; p < o.length; p++) { var f = o[p]; if (f.isNode()) { var h = f, g = h.boundingBox({ includeNodes: !0, includeEdges: !1, includeLabels: !1 }); i.boundingBoxesIntersect(d, g) && !i.boundingBoxInBoundingBox(g, d) && a.push(h) } else { var m = f, v = m._private, b = v.rscratch; if (null != b.startX && null != b.startY && !i.inBoundingBox(d, b.startX, b.startY)) continue; if (null != b.endX && null != b.endY && !i.inBoundingBox(d, b.endX, b.endY)) continue; if ("bezier" === b.edgeType || "multibezier" === b.edgeType || "self" === b.edgeType || "compound" === b.edgeType || "segments" === b.edgeType || "haystack" === b.edgeType) { for (var y = v.rstyle.bezierPts || v.rstyle.linePts || v.rstyle.haystackPts, x = !0, w = 0; w < y.length; w++)if (!i.pointInBoundingBox(d, y[w])) { x = !1; break } x && a.push(m) } else "haystack" !== b.edgeType && "straight" !== b.edgeType || a.push(m) } } return a } }); e.exports = a }, function (e, t, n) { "use strict"; var r = n(2), i = { calculateArrowAngles: function (e) { var t, n, i, o, a, s, l = e._private.rscratch, c = "haystack" === l.edgeType, u = "bezier" === l.edgeType, d = "multibezier" === l.edgeType, p = "segments" === l.edgeType, f = "compound" === l.edgeType, h = "self" === l.edgeType; if (c ? (i = l.haystackPts[0], o = l.haystackPts[1], a = l.haystackPts[2], s = l.haystackPts[3]) : (i = l.arrowStartX, o = l.arrowStartY, a = l.arrowEndX, s = l.arrowEndY), m = l.midX, v = l.midY, p) t = i - l.segpts[0], n = o - l.segpts[1]; else if (d || f || h || u) { var g = l.allpts; t = i - r.qbezierAt(g[0], g[2], g[4], .1), n = o - r.qbezierAt(g[1], g[3], g[5], .1) } else t = i - m, n = o - v; l.srcArrowAngle = r.getAngleFromDisp(t, n); var m = l.midX, v = l.midY; if (c && (m = (i + a) / 2, v = (o + s) / 2), t = a - i, n = s - o, p) if ((g = l.allpts).length / 2 % 2 == 0) { var b = (y = g.length / 2) - 2; t = g[y] - g[b], n = g[y + 1] - g[b + 1] } else { b = (y = g.length / 2 - 1) - 2; var y, x = y + 2; t = g[y] - g[b], n = g[y + 1] - g[b + 1] } else if (d || f || h) { var w, k, A, E, g = l.allpts; if (l.ctrlpts.length / 2 % 2 == 0) { var S = 2 + ($ = 2 + (C = g.length / 2 - 1)); w = r.qbezierAt(g[C], g[$], g[S], 0), k = r.qbezierAt(g[C + 1], g[$ + 1], g[S + 1], 0), A = r.qbezierAt(g[C], g[$], g[S], 1e-4), E = r.qbezierAt(g[C + 1], g[$ + 1], g[S + 1], 1e-4) } else { var $, C = ($ = g.length / 2 - 1) - 2; S = $ + 2, w = r.qbezierAt(g[C], g[$], g[S], .4999), k = r.qbezierAt(g[C + 1], g[$ + 1], g[S + 1], .4999), A = r.qbezierAt(g[C], g[$], g[S], .5), E = r.qbezierAt(g[C + 1], g[$ + 1], g[S + 1], .5) } t = A - w, n = E - k } if (l.midtgtArrowAngle = r.getAngleFromDisp(t, n), l.midDispX = t, l.midDispY = n, t *= -1, n *= -1, p && ((g = l.allpts).length / 2 % 2 == 0 || (t = -(g[x = 2 + (y = g.length / 2 - 1)] - g[y]), n = -(g[x + 1] - g[y + 1]))), l.midsrcArrowAngle = r.getAngleFromDisp(t, n), p) t = a - l.segpts[l.segpts.length - 2], n = s - l.segpts[l.segpts.length - 1]; else if (d || f || h || u) { var _ = (g = l.allpts).length; t = a - r.qbezierAt(g[_ - 6], g[_ - 4], g[_ - 2], .9), n = s - r.qbezierAt(g[_ - 5], g[_ - 3], g[_ - 1], .9) } else t = a - m, n = s - v; l.tgtArrowAngle = r.getAngleFromDisp(t, n) } }; i.getArrowWidth = i.getArrowHeight = function (e, t) { var n = this.arrowWidthCache = this.arrowWidthCache || {}, r = n[e + ", " + t]; return r || (r = Math.max(Math.pow(13.37 * e, .9), 29) * t, n[e + ", " + t] = r, r) }, e.exports = i }, function (e, t, n) { "use strict"; var r = n(2), i = n(0), o = {}; function a(e) { var t = []; if (null != e) { for (var n = 0; n < e.length; n += 2) { var r = e[n], i = e[n + 1]; t.push({ x: r, y: i }) } return t } } o.findEdgeControlPoints = function (e) { if (e && 0 !== e.length) { for (var t, n, o, a, s, l = this, c = l.cy.hasCompoundNodes(), u = {}, d = [], p = [], f = 0; f < e.length; f++) { var h = (Ft = (Bt = e[f])._private).data, g = "unbundled-bezier" === (D = Bt.pstyle("curve-style").value) || "segments" === D, m = "unbundled-bezier" === D || "bezier" === D; if ("none" !== Bt.pstyle("display").value) if ("haystack" !== D) { var v = h.source, b = h.target; t = v > b ? b + "$-$" + v : v + "$-$" + b, g && (t = "unbundled$-$" + h.id); var y = u[t]; null == y && (y = u[t] = [], d.push(t)), y.push(Bt), g && (y.hasUnbundled = !0), m && (y.hasBezier = !0) } else p.push(Bt) } for (var x = 0; x < d.length; x++) { var w = u[t = d[x]]; if (w.sort((function (e, t) { return e.poolIndex() - t.poolIndex() })), Ht = w[0]._private.source, Gt = w[0]._private.target, !w.hasUnbundled && Ht.id() > Gt.id()) { var k = Ht; Ht = Gt, Gt = k } Wt = Ht.position(), Yt = Gt.position(), Xt = Ht.outerWidth(), Qt = Ht.outerHeight(), Zt = Gt.outerWidth(), Jt = Gt.outerHeight(), n = l.nodeShapes[this.getNodeShape(Ht)], o = l.nodeShapes[this.getNodeShape(Gt)], s = !1; var A = { north: 0, west: 0, south: 0, east: 0, northwest: 0, southwest: 0, northeast: 0, southeast: 0 }, E = Wt.x, S = Wt.y, $ = Xt, C = Qt, _ = Yt.x, O = Yt.y, T = Zt, j = Jt, P = w.length; for (f = 0; f < w.length; f++) { var D, R, I = (Vt = (Bt = w[f])._private.rscratch).lastEdgeIndex, N = f, M = Vt.lastNumEdges, z = (g = "unbundled-bezier" === (D = Bt.pstyle("curve-style").value) || "segments" === D, Ht.id() !== Bt.source().id()), L = Bt.pstyle("control-point-distances"), B = Bt.pstyle("loop-direction").pfValue, F = Bt.pstyle("loop-sweep").pfValue, q = Bt.pstyle("control-point-weights"), V = L && q ? Math.min(L.value.length, q.value.length) : 1, U = Bt.pstyle("control-point-step-size").pfValue, H = L ? L.pfValue[0] : void 0, G = q.value[0], W = Bt.pstyle("edge-distances").value, Y = Bt.pstyle("segment-weights"), X = Bt.pstyle("segment-distances"), Z = Math.min(Y.pfValue.length, X.pfValue.length), Q = Bt.pstyle("source-endpoint").value, J = Bt.pstyle("target-endpoint").value, K = Bt.pstyle("source-arrow-shape").value, ee = Bt.pstyle("target-arrow-shape").value, te = Bt.pstyle("arrow-scale").value, ne = Bt.pstyle("width").pfValue, re = Vt.lastSrcCtlPtX, ie = Vt.lastSrcCtlPtY, oe = Vt.lastSrcCtlPtW, ae = Vt.lastSrcCtlPtH, se = Vt.lastTgtCtlPtX, le = Vt.lastTgtCtlPtY, ce = Vt.lastTgtCtlPtW, ue = Vt.lastTgtCtlPtH, de = Vt.lastCurveStyle, pe = D, fe = Vt.lastCtrlptDists, he = L ? L.strValue : null, ge = Vt.lastCtrlptWs, me = q.strValue, ve = Vt.lastSegmentWs, be = Y.strValue, ye = Vt.lastSegmentDs, xe = X.strValue, we = Vt.lastStepSize, ke = U, Ae = Vt.lastLoopDir, Ee = B, Se = Vt.lastLoopSwp, $e = F, Ce = Vt.lastEdgeDistances, _e = W, Oe = Vt.lastSrcEndpt, Te = Q, je = Vt.lastTgtEndpt, Pe = J, De = Vt.lastSrcArr, Re = K, Ie = Vt.lastTgtArr, Ne = ee, Me = Vt.lastLineW, ze = ne, Le = Vt.lastArrScl, Be = te; if (Vt.badBezier = !!s, re === E && ie === S && oe === $ && ae === C && se === _ && le === O && ce === T && ue === j && de === pe && fe === he && ge === me && ve === be && ye === xe && we === ke && Ae === Ee && Se === $e && Ce === _e && Oe === Te && je === Pe && De === Re && Ie === Ne && Me === ze && Le === Be && (I === N && M === P || g) ? R = !0 : (R = !1, Vt.lastSrcCtlPtX = E, Vt.lastSrcCtlPtY = S, Vt.lastSrcCtlPtW = $, Vt.lastSrcCtlPtH = C, Vt.lastTgtCtlPtX = _, Vt.lastTgtCtlPtY = O, Vt.lastTgtCtlPtW = T, Vt.lastTgtCtlPtH = j, Vt.lastEdgeIndex = N, Vt.lastNumEdges = P, Vt.lastCurveStyle = pe, Vt.lastCtrlptDists = he, Vt.lastCtrlptWs = me, Vt.lastSegmentDs = xe, Vt.lastSegmentWs = be, Vt.lastStepSize = ke, Vt.lastLoopDir = Ee, Vt.lastLoopSwp = $e, Vt.lastEdgeDistances = _e, Vt.lastSrcEndpt = Te, Vt.lastTgtEndpt = Pe, Vt.lastSrcArr = Re, Vt.lastTgtArr = Ne, Vt.lastLineW = ze, Vt.lastArrScl = Be), !R) { if (!w.calculatedIntersection && Ht !== Gt && (w.hasBezier || w.hasUnbundled)) { w.calculatedIntersection = !0; var Fe = n.intersectLine(Wt.x, Wt.y, Xt, Qt, Yt.x, Yt.y, 0); w.srcIntn = Fe; var qe = o.intersectLine(Yt.x, Yt.y, Zt, Jt, Wt.x, Wt.y, 0); w.tgtIntn = qe; var Ve = { x1: Fe[0], x2: qe[0], y1: Fe[1], y2: qe[1] }, Ue = { x1: Wt.x, x2: Yt.x, y1: Wt.y, y2: Yt.y }, He = qe[1] - Fe[1], Ge = qe[0] - Fe[0], We = Math.sqrt(Ge * Ge + He * He), Ye = { x: Ge, y: He }, Xe = { x: Ye.x / We, y: Ye.y / We }; a = { x: -Xe.y, y: Xe.x }, o.checkPoint(Fe[0], Fe[1], 0, Zt, Jt, Yt.x, Yt.y) && n.checkPoint(qe[0], qe[1], 0, Xt, Qt, Wt.x, Wt.y) && (a = {}, s = !0) } if (z ? (Vt.srcIntn = w.tgtIntn, Vt.tgtIntn = w.srcIntn) : (Vt.srcIntn = w.srcIntn, Vt.tgtIntn = w.tgtIntn), Ht === Gt) { Vt.edgeType = "self"; var Ze = f, Qe = U; g && (Ze = 0, Qe = H); var Je = B - Math.PI / 2, Ke = Je - F / 2, et = Je + F / 2, tt = String(B + "_" + F); Ze = void 0 === A[tt] ? A[tt] = 0 : ++A[tt], Vt.ctrlpts = [Wt.x + 1.4 * Math.cos(Ke) * Qe * (Ze / 3 + 1), Wt.y + 1.4 * Math.sin(Ke) * Qe * (Ze / 3 + 1), Wt.x + 1.4 * Math.cos(et) * Qe * (Ze / 3 + 1), Wt.y + 1.4 * Math.sin(et) * Qe * (Ze / 3 + 1)] } else if (c && (Ht.isParent() || Ht.isChild() || Gt.isParent() || Gt.isChild()) && (Ht.parents().anySame(Gt) || Gt.parents().anySame(Ht))) { Vt.edgeType = "compound", Vt.badBezier = !1, Ze = f, Qe = U, g && (Ze = 0, Qe = H); var nt = { x: Wt.x - Xt / 2, y: Wt.y - Qt / 2 }, rt = { x: Yt.x - Zt / 2, y: Yt.y - Jt / 2 }, it = { x: Math.min(nt.x, rt.x), y: Math.min(nt.y, rt.y) }, ot = Math.max(.5, Math.log(.01 * Xt)), at = Math.max(.5, Math.log(.01 * Zt)); Vt.ctrlpts = [it.x, it.y - (1 + Math.pow(50, 1.12) / 100) * Qe * (Ze / 3 + 1) * ot, it.x - (1 + Math.pow(50, 1.12) / 100) * Qe * (Ze / 3 + 1) * at, it.y] } else if ("segments" === D) { Vt.edgeType = "segments", Vt.segpts = []; for (var st = 0; st < Z; st++) { var lt = Y.pfValue[st], ct = X.pfValue[st], ut = 1 - lt, dt = lt, pt = { x: (bt = "node-position" === W ? Ue : Ve).x1 * ut + bt.x2 * dt, y: bt.y1 * ut + bt.y2 * dt }; Vt.segpts.push(pt.x + a.x * ct, pt.y + a.y * ct) } } else if (w.length % 2 != 1 || f !== Math.floor(w.length / 2) || g) { var ft = g; Vt.edgeType = ft ? "multibezier" : "bezier", Vt.ctrlpts = []; for (var ht = 0; ht < V; ht++) { var gt, mt = (.5 - w.length / 2 + f) * U, vt = r.signum(mt); ft && (H = L ? L.pfValue[ht] : U, G = q.value[ht]); var bt, yt = void 0 !== (gt = g ? H : void 0 !== H ? vt * H : void 0) ? gt : mt; ut = 1 - G, dt = G, z && (k = ut, ut = dt, dt = k), pt = { x: (bt = "node-position" === W ? Ue : Ve).x1 * ut + bt.x2 * dt, y: bt.y1 * ut + bt.y2 * dt }, Vt.ctrlpts.push(pt.x + a.x * yt, pt.y + a.y * yt) } } else Vt.edgeType = "straight"; this.findEndpoints(Bt); var xt = !i.number(Vt.startX) || !i.number(Vt.startY), wt = !i.number(Vt.arrowStartX) || !i.number(Vt.arrowStartY), kt = !i.number(Vt.endX) || !i.number(Vt.endY), At = !i.number(Vt.arrowEndX) || !i.number(Vt.arrowEndY), Et = this.getArrowWidth(Bt.pstyle("width").pfValue, Bt.pstyle("arrow-scale").value) * this.arrowShapeWidth * 3; if ("bezier" === Vt.edgeType) { var St = r.dist({ x: Vt.ctrlpts[0], y: Vt.ctrlpts[1] }, { x: Vt.startX, y: Vt.startY }), $t = St < Et, Ct = r.dist({ x: Vt.ctrlpts[0], y: Vt.ctrlpts[1] }, { x: Vt.endX, y: Vt.endY }), _t = Ct < Et, Ot = !1; if (xt || wt || $t) { Ot = !0; var Tt = { x: Vt.ctrlpts[0] - Wt.x, y: Vt.ctrlpts[1] - Wt.y }, jt = Math.sqrt(Tt.x * Tt.x + Tt.y * Tt.y), Pt = { x: Tt.x / jt, y: Tt.y / jt }, Dt = Math.max(Xt, Qt), Rt = { x: Vt.ctrlpts[0] + 2 * Pt.x * Dt, y: Vt.ctrlpts[1] + 2 * Pt.y * Dt }, It = n.intersectLine(Wt.x, Wt.y, Xt, Qt, Rt.x, Rt.y, 0); $t ? (Vt.ctrlpts[0] = Vt.ctrlpts[0] + Pt.x * (Et - St), Vt.ctrlpts[1] = Vt.ctrlpts[1] + Pt.y * (Et - St)) : (Vt.ctrlpts[0] = It[0] + Pt.x * Et, Vt.ctrlpts[1] = It[1] + Pt.y * Et) } if (kt || At || _t) { Ot = !0, Tt = { x: Vt.ctrlpts[0] - Yt.x, y: Vt.ctrlpts[1] - Yt.y }, jt = Math.sqrt(Tt.x * Tt.x + Tt.y * Tt.y), Pt = { x: Tt.x / jt, y: Tt.y / jt }, Dt = Math.max(Xt, Qt), Rt = { x: Vt.ctrlpts[0] + 2 * Pt.x * Dt, y: Vt.ctrlpts[1] + 2 * Pt.y * Dt }; var Nt = o.intersectLine(Yt.x, Yt.y, Zt, Jt, Rt.x, Rt.y, 0); _t ? (Vt.ctrlpts[0] = Vt.ctrlpts[0] + Pt.x * (Et - Ct), Vt.ctrlpts[1] = Vt.ctrlpts[1] + Pt.y * (Et - Ct)) : (Vt.ctrlpts[0] = Nt[0] + Pt.x * Et, Vt.ctrlpts[1] = Nt[1] + Pt.y * Et) } Ot && this.findEndpoints(Bt) } if ("multibezier" === Vt.edgeType || "bezier" === Vt.edgeType || "self" === Vt.edgeType || "compound" === Vt.edgeType) { for (Vt.allpts = [], Vt.allpts.push(Vt.startX, Vt.startY), ht = 0; ht + 1 < Vt.ctrlpts.length; ht += 2)Vt.allpts.push(Vt.ctrlpts[ht], Vt.ctrlpts[ht + 1]), ht + 3 < Vt.ctrlpts.length && Vt.allpts.push((Vt.ctrlpts[ht] + Vt.ctrlpts[ht + 2]) / 2, (Vt.ctrlpts[ht + 1] + Vt.ctrlpts[ht + 3]) / 2); var Mt; Vt.allpts.push(Vt.endX, Vt.endY), Vt.ctrlpts.length / 2 % 2 == 0 ? (Mt = Vt.allpts.length / 2 - 1, Vt.midX = Vt.allpts[Mt], Vt.midY = Vt.allpts[Mt + 1]) : (Mt = Vt.allpts.length / 2 - 3, Vt.midX = r.qbezierAt(Vt.allpts[Mt], Vt.allpts[Mt + 2], Vt.allpts[Mt + 4], .5), Vt.midY = r.qbezierAt(Vt.allpts[Mt + 1], Vt.allpts[Mt + 3], Vt.allpts[Mt + 5], .5)) } else if ("straight" === Vt.edgeType) Vt.allpts = [Vt.startX, Vt.startY, Vt.endX, Vt.endY], Vt.midX = (Vt.startX + Vt.endX + Vt.arrowStartX + Vt.arrowEndX) / 4, Vt.midY = (Vt.startY + Vt.endY + Vt.arrowStartY + Vt.arrowEndY) / 4; else if ("segments" === Vt.edgeType) if (Vt.allpts = [], Vt.allpts.push(Vt.startX, Vt.startY), Vt.allpts.push.apply(Vt.allpts, Vt.segpts), Vt.allpts.push(Vt.endX, Vt.endY), Vt.segpts.length % 4 == 0) { var zt = Vt.segpts.length / 2, Lt = zt - 2; Vt.midX = (Vt.segpts[Lt] + Vt.segpts[zt]) / 2, Vt.midY = (Vt.segpts[Lt + 1] + Vt.segpts[zt + 1]) / 2 } else Lt = Vt.segpts.length / 2 - 1, Vt.midX = Vt.segpts[Lt], Vt.midY = Vt.segpts[Lt + 1]; this.storeEdgeProjections(Bt), this.calculateArrowAngles(Bt) } this.recalculateEdgeLabelProjections(Bt), this.calculateLabelAngles(Bt) } } for (f = 0; f < p.length; f++) { var Bt, Ft, qt = (Ft = (Bt = p[f])._private).rscratch, Vt = qt; if (!qt.haystack) { var Ut = 2 * Math.random() * Math.PI; qt.source = { x: Math.cos(Ut), y: Math.sin(Ut) }, Ut = 2 * Math.random() * Math.PI, qt.target = { x: Math.cos(Ut), y: Math.sin(Ut) } } var Ht = Ft.source, Gt = Ft.target, Wt = Ht.position(), Yt = Gt.position(), Xt = Ht.width(), Zt = Gt.width(), Qt = Ht.height(), Jt = Gt.height(), Kt = (Dt = Bt.pstyle("haystack-radius").value) / 2; Vt.haystackPts = Vt.allpts = [Vt.source.x * Xt * Kt + Wt.x, Vt.source.y * Qt * Kt + Wt.y, Vt.target.x * Zt * Kt + Yt.x, Vt.target.y * Jt * Kt + Yt.y], Vt.midX = (Vt.allpts[0] + Vt.allpts[2]) / 2, Vt.midY = (Vt.allpts[1] + Vt.allpts[3]) / 2, qt.edgeType = qt.lastCurveStyle = "haystack", qt.haystack = !0, this.storeEdgeProjections(Bt), this.calculateArrowAngles(Bt), this.recalculateEdgeLabelProjections(Bt), this.calculateLabelAngles(Bt) } } }, o.getSegmentPoints = function (e) { var t = e[0]._private.rscratch; if ("segments" === t.edgeType) return a(t.segpts) }, o.getControlPoints = function (e) { var t = e[0]._private.rscratch, n = t.edgeType; if ("bezier" === n || "multibezier" === n || "self" === n || "compound" === n) return a(t.ctrlpts) }, o.getEdgeMidpoint = function (e) { var t = e[0]._private.rscratch; return { x: t.midX, y: t.midY } }, e.exports = o }, function (e, t, n) { "use strict"; var r = n(2), i = n(0), o = { manualEndptToPx: function (e, t) { var n = e.position(), r = e.outerWidth(), i = e.outerHeight(); if (2 === t.value.length) { var o = [t.pfValue[0], t.pfValue[1]]; return "%" === t.units[0] && (o[0] = o[0] * r), "%" === t.units[1] && (o[1] = o[1] * i), o[0] += n.x, o[1] += n.y, o } var a = t.pfValue[0]; a = -Math.PI / 2 + a; var s = 2 * Math.max(r, i), l = [n.x + Math.cos(a) * s, n.y + Math.sin(a) * s]; return this.nodeShapes[this.getNodeShape(e)].intersectLine(n.x, n.y, r, i, l[0], l[1], 0) }, findEndpoints: function (e) { var t = this, n = void 0, o = e.source()[0], a = e.target()[0], s = o.position(), l = a.position(), c = e.pstyle("target-arrow-shape").value, u = e.pstyle("source-arrow-shape").value, d = e.pstyle("target-distance-from-node").pfValue, p = e.pstyle("source-distance-from-node").pfValue, f = e._private.rscratch, h = f.edgeType, g = "self" === h || "compound" === h, m = "bezier" === h || "multibezier" === h || g, v = "bezier" !== h, b = "straight" === h || "segments" === h, y = "segments" === h, x = m || v || b, w = e.pstyle("source-endpoint"), k = g ? "outside-to-node" : w.value, A = e.pstyle("target-endpoint"), E = g ? "outside-to-node" : A.value; f.srcManEndpt = w, f.tgtManEndpt = A; var S = void 0, $ = void 0, C = void 0, _ = void 0; if (m) { var O = [f.ctrlpts[0], f.ctrlpts[1]]; S = v ? [f.ctrlpts[f.ctrlpts.length - 2], f.ctrlpts[f.ctrlpts.length - 1]] : O, $ = O } else if (b) { var T = y ? f.segpts.slice(0, 2) : [l.x, l.y]; S = y ? f.segpts.slice(f.segpts.length - 2) : [s.x, s.y], $ = T } "inside-to-node" === E ? n = [l.x, l.y] : A.units ? n = this.manualEndptToPx(a, A) : "outside-to-line" === E ? n = f.tgtIntn : ("outside-to-node" === E ? C = S : "outside-to-line" === E && (C = [s.x, s.y]), n = t.nodeShapes[this.getNodeShape(a)].intersectLine(l.x, l.y, a.outerWidth(), a.outerHeight(), C[0], C[1], 0)); var j = r.shortenIntersection(n, S, t.arrowShapes[c].spacing(e) + d), P = r.shortenIntersection(n, S, t.arrowShapes[c].gap(e) + d); e.hasClass("horizontal") ? (f.endX = n[0] - d, f.endY = n[1], f.arrowEndX = n[0] - d / 2, f.arrowEndY = n[1]) : e.hasClass("vertical") ? (f.endX = n[0], f.endY = n[1] - d, f.arrowEndX = n[0], f.arrowEndY = n[1] - d / 2) : (f.endX = P[0], f.endY = P[1], f.arrowEndX = j[0], f.arrowEndY = j[1]), "inside-to-node" === k ? n = [s.x, s.y] : w.units ? n = this.manualEndptToPx(o, w) : "outside-to-line" === k ? n = f.srcIntn : ("outside-to-node" === k ? _ = $ : "outside-to-line" === k && (_ = [l.x, l.y]), n = t.nodeShapes[this.getNodeShape(o)].intersectLine(s.x, s.y, o.outerWidth(), o.outerHeight(), _[0], _[1], 0)); var D = r.shortenIntersection(n, $, t.arrowShapes[u].spacing(e) + p), R = r.shortenIntersection(n, $, t.arrowShapes[u].gap(e) + p); f.startX = R[0], f.startY = R[1], f.arrowStartX = D[0], f.arrowStartY = D[1], x && (i.number(f.startX) && i.number(f.startY) && i.number(f.endX) && i.number(f.endY) ? f.badLine = !1 : f.badLine = !0) }, getSourceEndpoint: function (e) { var t = e[0]._private.rscratch; switch (t.edgeType) { case "haystack": return { x: t.haystackPts[0], y: t.haystackPts[1] }; default: return { x: t.arrowStartX, y: t.arrowStartY } } }, getTargetEndpoint: function (e) { var t = e[0]._private.rscratch; switch (t.edgeType) { case "haystack": return { x: t.haystackPts[2], y: t.haystackPts[3] }; default: return { x: t.arrowEndX, y: t.arrowEndY } } } }; e.exports = o }, function (e, t, n) { "use strict"; var r = n(2), i = {}; function o(e, t, n) { for (var i = function (e, t, n, i) { return r.qbezierAt(e, t, n, i) }, o = t._private.rstyle.bezierPts, a = 0; a < e.bezierProjPcts.length; a++) { var s = e.bezierProjPcts[a]; o.push({ x: i(n[0], n[2], n[4], s), y: i(n[1], n[3], n[5], s) }) } } i.storeEdgeProjections = function (e) { var t = e._private, n = t.rscratch, r = n.edgeType; if (t.rstyle.bezierPts = null, t.rstyle.linePts = null, t.rstyle.haystackPts = null, "multibezier" === r || "bezier" === r || "self" === r || "compound" === r) { t.rstyle.bezierPts = []; for (var i = 0; i + 5 < n.allpts.length; i += 4)o(this, e, n.allpts.slice(i, i + 6)) } else if ("segments" === r) { var a = t.rstyle.linePts = []; for (i = 0; i + 1 < n.allpts.length; i += 2)a.push({ x: n.allpts[i], y: n.allpts[i + 1] }) } else if ("haystack" === r) { var s = n.haystackPts; t.rstyle.haystackPts = [{ x: s[0], y: s[1] }, { x: s[2], y: s[3] }] } t.rstyle.arrowWidth = this.getArrowWidth(e.pstyle("width").pfValue, e.pstyle("arrow-scale").value) * this.arrowShapeWidth }, i.recalculateEdgeProjections = function (e) { this.findEdgeControlPoints(e) }, e.exports = i }, function (e, t, n) { "use strict"; var r = n(2), i = n(0), o = n(1), a = { recalculateNodeLabelProjection: function (e) { var t = e.pstyle("label").strValue; if (!i.emptyString(t)) { var n, r, o = e._private, a = e.width(), s = e.height(), l = e.padding(), c = e.position(), u = e.pstyle("text-halign").strValue, d = e.pstyle("text-valign").strValue, p = o.rscratch, f = o.rstyle; switch (u) { case "left": n = c.x - a / 2 - l; break; case "right": n = c.x + a / 2 + l; break; default: n = c.x }switch (d) { case "top": r = c.y - s / 2 - l; break; case "bottom": r = c.y + s / 2 + l; break; default: r = c.y }p.labelX = n, p.labelY = r, f.labelX = n, f.labelY = r, this.applyLabelDimensions(e) } }, recalculateEdgeLabelProjections: function (e) { var t, n = e._private, i = n.rscratch, a = this, s = { mid: e.pstyle("label").strValue, source: e.pstyle("source-label").strValue, target: e.pstyle("target-label").strValue }; if (s.mid || s.source || s.target) { t = { x: i.midX, y: i.midY }; var l = function (e, t, r) { o.setPrefixedProperty(n.rscratch, e, t, r), o.setPrefixedProperty(n.rstyle, e, t, r) }; l("labelX", null, t.x), l("labelY", null, t.y); var c = function (o) { var c, u = "source" === o; if (s[o]) { var d = e.pstyle(o + "-text-offset").pfValue, p = function (e, t) { var n = t.x - e.x, r = t.y - e.y; return Math.atan(r / n) }; switch (i.edgeType) { case "self": case "compound": case "bezier": case "multibezier": for (var f, h = function e() { if (e.cache) return e.cache; for (var t = [], o = 0; o + 5 < i.allpts.length; o += 4) { var s = { x: i.allpts[o], y: i.allpts[o + 1] }, l = { x: i.allpts[o + 2], y: i.allpts[o + 3] }, c = { x: i.allpts[o + 4], y: i.allpts[o + 5] }; t.push({ p0: s, p1: l, p2: c, startDist: 0, length: 0, segments: [] }) } var u = n.rstyle.bezierPts, d = a.bezierProjPcts.length; function p(e, t, n, i, o) { var a = r.dist(t, n), s = e.segments[e.segments.length - 1], l = { p0: t, p1: n, t0: i, t1: o, startDist: s ? s.startDist + s.length : 0, length: a }; e.segments.push(l), e.length += a } for (o = 0; o < t.length; o++) { var f = t[o], h = t[o - 1]; h && (f.startDist = h.startDist + h.length), p(f, f.p0, u[o * d], 0, a.bezierProjPcts[0]); for (var g = 0; g < d - 1; g++)p(f, u[o * d + g], u[o * d + g + 1], a.bezierProjPcts[g], a.bezierProjPcts[g + 1]); p(f, u[o * d + d - 1], f.p2, a.bezierProjPcts[d - 1], 1) } return e.cache = t }(), g = 0, m = 0, v = 0; v < h.length; v++) { for (var b = h[u ? v : h.length - 1 - v], y = 0; y < b.segments.length; y++) { var x = b.segments[u ? y : b.segments.length - 1 - y], w = v === h.length - 1 && y === b.segments.length - 1; if (g = m, (m += x.length) >= d || w) { f = { cp: b, segment: x }; break } } if (f) break } b = f.cp; var k = (d - g) / (x = f.segment).length, A = x.t1 - x.t0, E = u ? x.t0 + A * k : x.t1 - A * k; E = r.bound(0, E, 1), t = r.qbezierPtAt(b.p0, b.p1, b.p2, E), c = function (e, t, n, i) { var o = r.bound(0, i - .001, 1), a = r.bound(0, i + .001, 1), s = r.qbezierPtAt(e, t, n, o), l = r.qbezierPtAt(e, t, n, a); return p(s, l) }(b.p0, b.p1, b.p2, E); break; case "straight": case "segments": case "haystack": var S, $, C, _, O = 0, T = i.allpts.length; for (v = 0; v + 3 < T && (u ? (C = { x: i.allpts[v], y: i.allpts[v + 1] }, _ = { x: i.allpts[v + 2], y: i.allpts[v + 3] }) : (C = { x: i.allpts[T - 2 - v], y: i.allpts[T - 1 - v] }, _ = { x: i.allpts[T - 4 - v], y: i.allpts[T - 3 - v] }), $ = O, !((O += S = r.dist(C, _)) >= d)); v += 2); E = (d - $) / S, E = r.bound(0, E, 1), t = r.lineAt(C, _, E), c = p(C, _) }l("labelX", o, t.x), l("labelY", o, t.y), l("labelAutoAngle", o, c) } }; c("source"), c("target"), this.applyLabelDimensions(e) } }, applyLabelDimensions: function (e) { this.applyPrefixedLabelDimensions(e), e.isEdge() && (this.applyPrefixedLabelDimensions(e, "source"), this.applyPrefixedLabelDimensions(e, "target")) }, applyPrefixedLabelDimensions: function (e, t) { var n = e._private, r = this.getLabelText(e, t), i = this.calculateLabelDimensions(e, r); o.setPrefixedProperty(n.rstyle, "labelWidth", t, i.width), o.setPrefixedProperty(n.rscratch, "labelWidth", t, i.width), o.setPrefixedProperty(n.rstyle, "labelHeight", t, i.height), o.setPrefixedProperty(n.rscratch, "labelHeight", t, i.height) }, getLabelText: function (e, t) { var n = e._private, r = t ? t + "-" : "", i = e.pstyle(r + "label").strValue, a = e.pstyle("text-transform").value, s = function (e, r) { return r ? (o.setPrefixedProperty(n.rscratch, e, t, r), r) : o.getPrefixedProperty(n.rscratch, e, t) }; "none" == a || ("uppercase" == a ? i = i.toUpperCase() : "lowercase" == a && (i = i.toLowerCase())); var l = e.pstyle("text-wrap").value; if ("wrap" === l) { var c = s("labelKey"); if (c && s("labelWrapKey") === c) return s("labelWrapCachedText"); for (var u = i.split("\n"), d = e.pstyle("text-max-width").pfValue, p = [], f = 0; f < u.length; f++) { var h = u[f]; if (this.calculateLabelDimensions(e, h, "line=" + h).width > d) { for (var g = h.split(/\s+/), m = "", v = 0; v < g.length; v++) { var b = g[v], y = 0 === m.length ? b : m + " " + b; this.calculateLabelDimensions(e, y, "testLine=" + y).width <= d ? m += b + " " : (p.push(m), m = b + " ") } m.match(/^\s+$/) || p.push(m) } else p.push(h) } s("labelWrapCachedLines", p), i = s("labelWrapCachedText", p.join("\n")), s("labelWrapKey", c) } else if ("ellipsis" === l) { d = e.pstyle("text-max-width").pfValue; for (var x = "", w = !1, k = 0; k < i.length && !(this.calculateLabelDimensions(e, x + i[k] + "…").width > d); k++)x += i[k], k === i.length - 1 && (w = !0); return w || (x += "…"), x } return i }, calculateLabelDimensions: function (e, t, n) { var r = e._private.labelStyleKey + "$@$" + t; n && (r += "$@$" + n); var i = this.labelDimCache || (this.labelDimCache = {}); if (i[r]) return i[r]; var o = e.pstyle("font-style").strValue, a = 1 * e.pstyle("font-size").pfValue + "px", s = e.pstyle("font-family").strValue, l = e.pstyle("font-weight").strValue, c = this.labelCalcDiv; c || (c = this.labelCalcDiv = document.createElement("div"), document.body.appendChild(c)); var u = c.style; return u.fontFamily = s, u.fontStyle = o, u.fontSize = a, u.fontWeight = l, u.position = "absolute", u.left = "-9999px", u.top = "-9999px", u.zIndex = "-1", u.visibility = "hidden", u.pointerEvents = "none", u.padding = "0", u.lineHeight = "1", "wrap" === e.pstyle("text-wrap").value ? u.whiteSpace = "pre" : u.whiteSpace = "normal", c.textContent = t, i[r] = { width: Math.ceil(c.clientWidth / 1), height: Math.ceil(c.clientHeight / 1) }, i[r] }, calculateLabelAngles: function (e) { var t = e._private.rscratch, n = e.isEdge(), r = e.pstyle("text-rotation"), i = r.strValue; "none" === i ? t.labelAngle = t.sourceLabelAngle = t.targetLabelAngle = 0 : n && "autorotate" === i ? (t.labelAngle = Math.atan(t.midDispY / t.midDispX), t.sourceLabelAngle = t.sourceLabelAutoAngle, t.targetLabelAngle = t.targetLabelAutoAngle) : t.labelAngle = t.sourceLabelAngle = t.targetLabelAngle = "autorotate" === i ? 0 : r.pfValue } }; e.exports = a }, function (e, t, n) { "use strict"; var r = { getNodeShape: function (e) { var t = e.pstyle("shape").value; if (e.isParent()) return "rectangle" === t || "roundrectangle" === t || "cutrectangle" === t || "barrel" === t ? t : "rectangle"; if ("polygon" === t) { var n = e.pstyle("shape-polygon-points").value; return this.nodeShapes.makePolygon(n).name } return t } }; e.exports = r }, function (e, t, n) { "use strict"; var r = { registerCalculationListeners: function () { var e = this.cy, t = e.collection(), n = this, r = function (e, n) { var r = !(arguments.length > 2 && void 0 !== arguments[2]) || arguments[2]; t.merge(e); for (var i = 0; i < e.length; i++) { var o = e[i], a = o._private, s = a.rstyle; r && (s.clean = !1, a.bbCache = null); var l = s.dirtyEvents = s.dirtyEvents || { length: 0 }; l[n.type] || (l[n.type] = !0, l.length++) } }; n.binder(e).on("position.* style.* free.* bounds.*", "node", (function (e) { var t = e.target; r(t, e), r(t.connectedEdges(), e) })).on("add.*", "node", (function (e) { var t = e.target; r(t, e) })).on("background.*", "node", (function (e) { var t = e.target; r(t, e, !1) })).on("add.* style.*", "edge", (function (e) { var t = e.target; r(t, e), r(t.parallelEdges(), e) })).on("remove.*", "edge", (function (e) { for (var t = e.target.parallelEdges(), n = 0; n < t.length; n++) { var i = t[n]; i.removed() || r(i, e) } })).on("dirty.*", "node", (function (e) { var t = e.target; r(t, e) })), n.beforeRender((function (r) { if (r) { var i = n.onUpdateEleCalcsFns; if (i) for (var o = 0; o < i.length; o++)(0, i[o])(r, t); for (n.recalculateRenderedStyle(t, !1), o = 0; o < t.length; o++)t[o]._private.rstyle.dirtyEvents = null; t = e.collection() } }), n.beforeRenderPriorities.eleCalcs) }, onUpdateEleCalcs: function (e) { (this.onUpdateEleCalcsFns = this.onUpdateEleCalcsFns || []).push(e) }, recalculateRenderedStyle: function (e, t) { var n = [], r = []; if (!this.destroyed) { void 0 === t && (t = !0); for (var i = 0; i < e.length; i++) { var o = (l = (s = e[i])._private).rstyle; t && o.clean || s.removed() || "none" !== s.pstyle("display").value && ("nodes" === l.group ? r.push(s) : n.push(s), o.clean = !0) } for (i = 0; i < r.length; i++) { o = (l = (s = r[i])._private).rstyle; var a = s.position(); this.recalculateNodeLabelProjection(s), o.nodeX = a.x, o.nodeY = a.y, o.nodeW = s.pstyle("width").pfValue, o.nodeH = s.pstyle("height").pfValue } for (this.recalculateEdgeProjections(n), i = 0; i < n.length; i++) { o = (l = (s = n[i])._private).rstyle; var s, l, c = l.rscratch; this.recalculateEdgeLabelProjections(s), o.srcX = c.arrowStartX, o.srcY = c.arrowStartY, o.tgtX = c.arrowEndX, o.tgtY = c.arrowEndY, o.midX = c.midX, o.midY = c.midY, o.labelAngle = c.labelAngle, o.sourceLabelAngle = c.sourceLabelAngle, o.targetLabelAngle = c.targetLabelAngle } } } }; e.exports = r }, function (e, t, n) { "use strict"; var r = n(17), i = { updateCachedGrabbedEles: function () { var e = this.cachedZSortedEles; if (e) { e.drag = [], e.nondrag = []; for (var t = [], n = 0; n < e.length; n++) { var r = (i = e[n])._private.rscratch; i.grabbed() && !i.isParent() ? t.push(i) : r.inDragLayer ? e.drag.push(i) : e.nondrag.push(i) } for (n = 0; n < t.length; n++) { var i = t[n]; e.drag.push(i) } } }, invalidateCachedZSortedEles: function () { this.cachedZSortedEles = null }, getCachedZSortedEles: function (e) { if (e || !this.cachedZSortedEles) { var t = this.cy.mutableElements().toArray(); t.sort(r), t.interactive = t.filter((function (e) { return e.interactive() })), this.cachedZSortedEles = t, this.updateCachedGrabbedEles() } else t = this.cachedZSortedEles; return t } }; e.exports = i }, function (e, t, n) { "use strict"; var r = { getCachedImage: function (e, t, n) { var r = this.imageCache = this.imageCache || {}, i = r[e]; if (i) return i.image.complete || i.image.addEventListener("load", n), i.image; var o = (i = r[e] = r[e] || {}).image = new Image; return o.addEventListener("load", n), o.addEventListener("error", (function () { o.error = !0 })), "data:" === e.substring(0, "data:".length).toLowerCase() || (o.crossOrigin = t), o.src = e, o } }; e.exports = r }, function (e, t, n) { "use strict"; var r = n(0), i = n(1), o = n(2), a = (n(16), { registerBinding: function (e, t, n, r) { var i = Array.prototype.slice.apply(arguments, [1]), o = this.binder(e); return o.on.apply(o, i) }, binder: function (e) { var t = this, n = e === window || e === document || e === document.body || r.domElement(e); if (null == t.supportsPassiveEvents) { var i = !1; try { var o = Object.defineProperty({}, "passive", { get: function () { i = !0 } }); window.addEventListener("test", null, o) } catch (e) { } t.supportsPassiveEvents = i } var a = function (r, i, o) { var a = Array.prototype.slice.call(arguments); return n && t.supportsPassiveEvents && (a[2] = { capture: null != o && o, passive: !1, once: !1 }), t.bindings.push({ target: e, args: a }), (e.addEventListener || e.on).apply(e, a), this }; return { on: a, addEventListener: a, addListener: a, bind: a } }, nodeIsDraggable: function (e) { return e && e.isNode() && !e.locked() && e.grabbable() }, nodeIsGrabbable: function (e) { return this.nodeIsDraggable(e) && e.interactive() } }); a.load = function () { var e = this, t = function (t, n, r, o) { null == t && (t = e.cy); for (var a = 0; a < n.length; a++) { var s = n[a]; t.emit(i.extend({ originalEvent: r, type: s }, o)) } }, n = function (e) { return e.shiftKey || e.metaKey || e.ctrlKey }, a = function (t, n) { var r = !0; if (e.cy.hasCompoundNodes() && t && t.isEdge()) { for (var i = 0; n && i < n.length; i++)if ((t = n[i]).isNode() && t.isParent()) { r = !1; break } } else r = !0; return r }, s = function (e) { e[0]._private.rscratch.inDragLayer = !0 }, l = function (e) { e[0]._private.rscratch.isGrabTarget = !0 }, c = function (t, n) { var r = function (t) { var n; if (t.addToList && e.cy.hasCompoundNodes()) { if (!t.addToList.hasId) { t.addToList.hasId = {}; for (var r = 0; r < t.addToList.length; r++) { var i = t.addToList[r]; t.addToList.hasId[i.id()] = !0 } } n = t.addToList.hasId } return n || {} }(n); r[t.id()] || (n.addToList.push(t), r[t.id()] = !0, function (e) { e[0]._private.grabbed = !0 }(t)) }, u = function (t, n) { n = n || {}; var r = t.cy().hasCompoundNodes(); n.inDragLayer && (t.forEach(s), t.neighborhood().stdFilter((function (e) { return !r || e.isEdge() })).forEach(s)), n.addToList && t.forEach((function (e) { c(e, n) })), function (e, t) { if (e.cy().hasCompoundNodes() && (null != t.inDragLayer || null != t.addToList)) { var n = e.descendants(); t.inDragLayer && (n.forEach(s), n.connectedEdges().forEach(s)), t.addToList && n.forEach((function (e) { c(e, t) })) } }(t, n), f(t, { inDragLayer: n.inDragLayer }), e.updateCachedGrabbedEles() }, d = u, p = function (t) { t && (t.hasId = {}, e.getCachedZSortedEles().forEach((function (e) { !function (e) { e[0]._private.grabbed = !1 }(e), function (e) { e[0]._private.rscratch.inDragLayer = !1 }(e), function (e) { e[0]._private.rscratch.isGrabTarget = !1 }(e) })), e.updateCachedGrabbedEles()) }, f = function (e, t) { if ((null != t.inDragLayer || null != t.addToList) && e.cy().hasCompoundNodes()) { var n = e.ancestors().orphans(); if (!n.same(e)) { var r = n.descendants().spawnSelf().merge(n).unmerge(e).unmerge(e.descendants()), i = r.connectedEdges(); t.inDragLayer && (i.forEach(s), r.forEach(s)), t.addToList && r.forEach((function (e) { c(e, t) })) } } }, h = "undefined" != typeof MutationObserver; h ? (e.removeObserver = new MutationObserver((function (t) { for (var n = 0; n < t.length; n++) { var r = t[n].removedNodes; if (r) for (var i = 0; i < r.length; i++)if (r[i] === e.container) { e.destroy(); break } } })), e.container.parentNode && e.removeObserver.observe(e.container.parentNode, { childList: !0 })) : e.registerBinding(e.container, "DOMNodeRemoved", (function (t) { e.destroy() })); var g = i.debounce((function () { e.cy.resize() }), 100); h && (e.styleObserver = new MutationObserver(g), e.styleObserver.observe(e.container, { attributes: !0 })), e.registerBinding(window, "resize", g); var m = function () { e.invalidateContainerClientCoordsCache() }; !function (e, t) { for (; null != e;)t(e), e = e.parentNode }(e.container, (function (t) { e.registerBinding(t, "transitionend", m), e.registerBinding(t, "animationend", m), e.registerBinding(t, "scroll", m) })), e.registerBinding(e.container, "contextmenu", (function (e) { e.preventDefault() })); var v, b, y, x, w, k, A, E, S, $, C, _, O, T = function (t) { for (var n = e.findContainerClientCoords(), r = n[0], i = n[1], o = n[2], a = n[3], s = t.touches ? t.touches : [t], l = !1, c = 0; c < s.length; c++) { var u = s[c]; if (r <= u.clientX && u.clientX <= r + o && i <= u.clientY && u.clientY <= i + a) { l = !0; break } } if (!l) return !1; for (var d = e.container, p = t.target.parentNode, f = !1; p;) { if (p === d) { f = !0; break } p = p.parentNode } return !!f }; e.registerBinding(e.container, "mousedown", (function (n) { if (T(n)) { n.preventDefault(), e.hoverData.capture = !0, e.hoverData.which = n.which; var r = e.cy, i = [n.clientX, n.clientY], o = e.projectIntoViewport(i[0], i[1]), a = e.selection, s = e.findNearestElements(o[0], o[1], !0, !1), c = s[0], p = e.dragData.possibleDragElements; if (e.hoverData.mdownPos = o, e.hoverData.mdownGPos = i, 3 == n.which) { e.hoverData.cxtStarted = !0; var f = { originalEvent: n, type: "cxttapstart", position: { x: o[0], y: o[1] } }; c ? (c.activate(), c.emit(f), e.hoverData.down = c) : r.emit(f), e.hoverData.downTime = (new Date).getTime(), e.hoverData.cxtDragged = !1 } else if (1 == n.which) { if (c && c.activate(), null != c && e.nodeIsGrabbable(c)) { var h = function (e) { return { originalEvent: n, type: e, position: { x: o[0], y: o[1] } } }; if (l(c), c.selected()) { p = e.dragData.possibleDragElements = []; var g = r.$((function (t) { return t.isNode() && t.selected() && e.nodeIsGrabbable(t) })); u(g, { addToList: p }), c.emit(h("grabon")), g.forEach((function (e) { e.emit(h("grab")) })) } else p = e.dragData.possibleDragElements = [], d(c, { addToList: p }), c.emit(h("grabon")).emit(h("grab")); e.redrawHint("eles", !0), e.redrawHint("drag", !0) } e.hoverData.down = c, e.hoverData.downs = s, e.hoverData.downTime = (new Date).getTime(), t(c, ["mousedown", "tapstart", "vmousedown"], n, { position: { x: o[0], y: o[1] } }), null == c ? (a[4] = 1, e.data.bgActivePosistion = { x: o[0], y: o[1] }, e.redrawHint("select", !0), e.redraw()) : c.isEdge() && (a[4] = 1), e.hoverData.tapholdCancelled = !1, clearTimeout(e.hoverData.tapholdTimeout), e.hoverData.tapholdTimeout = setTimeout((function () { if (!e.hoverData.tapholdCancelled) { var t = e.hoverData.down; t ? t.emit({ originalEvent: n, type: "taphold", position: { x: o[0], y: o[1] } }) : r.emit({ originalEvent: n, type: "taphold", position: { x: o[0], y: o[1] } }) } }), e.tapholdDuration) } a[0] = a[2] = o[0], a[1] = a[3] = o[1] } }), !1), e.registerBinding(window, "mousemove", (function (i) { if (e.hoverData.capture || T(i)) { var s = !1, l = e.cy, c = l.zoom(), d = [i.clientX, i.clientY], f = e.projectIntoViewport(d[0], d[1]), h = e.hoverData.mdownPos, g = e.hoverData.mdownGPos, m = e.selection, v = null; e.hoverData.draggingEles || e.hoverData.dragging || e.hoverData.selecting || (v = e.findNearestElement(f[0], f[1], !0, !1)); var b, y = e.hoverData.last, x = e.hoverData.down, w = [f[0] - m[2], f[1] - m[3]], k = e.dragData.possibleDragElements; if (g) { var A = d[0] - g[0], E = A * A, S = d[1] - g[1], $ = E + S * S; e.hoverData.isOverThresholdDrag = b = $ >= e.desktopTapThreshold2 } var C = n(i); b && (e.hoverData.tapholdCancelled = !0), s = !0, t(v, ["mousemove", "vmousemove", "tapdrag"], i, { position: { x: f[0], y: f[1] } }); var _ = function () { e.data.bgActivePosistion = void 0, e.hoverData.selecting || l.emit("boxstart"), m[4] = 1, e.hoverData.selecting = !0, e.redrawHint("select", !0), e.redraw() }; if (3 === e.hoverData.which) { if (b) { var O = { originalEvent: i, type: "cxtdrag", position: { x: f[0], y: f[1] } }; x ? x.emit(O) : l.emit(O), e.hoverData.cxtDragged = !0, e.hoverData.cxtOver && v === e.hoverData.cxtOver || (e.hoverData.cxtOver && e.hoverData.cxtOver.emit({ originalEvent: i, type: "cxtdragout", position: { x: f[0], y: f[1] } }), e.hoverData.cxtOver = v, v && v.emit({ originalEvent: i, type: "cxtdragover", position: { x: f[0], y: f[1] } })) } } else if (e.hoverData.dragging) { if (s = !0, l.panningEnabled() && l.userPanningEnabled()) { var j; if (e.hoverData.justStartedPan) { var P = e.hoverData.mdownPos; j = { x: (f[0] - P[0]) * c, y: (f[1] - P[1]) * c }, e.hoverData.justStartedPan = !1 } else j = { x: w[0] * c, y: w[1] * c }; l.panBy(j), e.hoverData.dragged = !0 } f = e.projectIntoViewport(i.clientX, i.clientY) } else if (1 != m[4] || null != x && !x.isEdge()) { if (x && x.isEdge() && x.active() && x.unactivate(), x && x.grabbed() || v == y || (y && t(y, ["mouseout", "tapdragout"], i, { position: { x: f[0], y: f[1] } }), v && t(v, ["mouseover", "tapdragover"], i, { position: { x: f[0], y: f[1] } }), e.hoverData.last = v), x) if (b) { if (l.boxSelectionEnabled() && C) x && x.grabbed() && (p(k), x.emit("free")), _(); else if (x && x.grabbed() && e.nodeIsDraggable(x)) { var D = !e.dragData.didDrag; D && e.redrawHint("eles", !0), e.dragData.didDrag = !0; var R = []; e.hoverData.draggingEles || u(l.collection(k), { inDragLayer: !0 }); for (var I = 0; I < k.length; I++) { var N = k[I]; if (e.nodeIsDraggable(N) && N.grabbed()) { var M = N.position(); if (R.push(N), r.number(w[0]) && r.number(w[1]) && (M.x += w[0], M.y += w[1], D)) { var z = e.hoverData.dragDelta; z && r.number(z[0]) && r.number(z[1]) && (M.x += z[0], M.y += z[1]) } } } e.hoverData.draggingEles = !0; var L = l.collection(R); L.dirtyCompoundBoundsCache(), L.emit("position drag"), e.redrawHint("drag", !0), e.redraw() } } else !function () { var t = e.hoverData.dragDelta = e.hoverData.dragDelta || []; 0 === t.length ? (t.push(w[0]), t.push(w[1])) : (t[0] += w[0], t[1] += w[1]) }(); s = !0 } else b && (e.hoverData.dragging || !l.boxSelectionEnabled() || !C && l.panningEnabled() && l.userPanningEnabled() ? !e.hoverData.selecting && l.panningEnabled() && l.userPanningEnabled() && a(x, e.hoverData.downs) && (e.hoverData.dragging = !0, e.hoverData.justStartedPan = !0, m[4] = 0, e.data.bgActivePosistion = o.array2point(h), e.redrawHint("select", !0), e.redraw()) : _(), x && x.isEdge() && x.active() && x.unactivate()); return m[2] = f[0], m[3] = f[1], s ? (i.stopPropagation && i.stopPropagation(), i.preventDefault && i.preventDefault(), !1) : void 0 } }), !1), e.registerBinding(window, "mouseup", (function (r) { if (e.hoverData.capture) { e.hoverData.capture = !1; var i = e.cy, o = e.projectIntoViewport(r.clientX, r.clientY), a = e.selection, s = e.findNearestElement(o[0], o[1], !0, !1), l = e.dragData.possibleDragElements, c = e.hoverData.down, u = n(r); if (e.data.bgActivePosistion && (e.redrawHint("select", !0), e.redraw()), e.hoverData.tapholdCancelled = !0, e.data.bgActivePosistion = void 0, c && c.unactivate(), 3 === e.hoverData.which) { var d = { originalEvent: r, type: "cxttapend", position: { x: o[0], y: o[1] } }; if (c ? c.emit(d) : i.emit(d), !e.hoverData.cxtDragged) { var f = { originalEvent: r, type: "cxttap", position: { x: o[0], y: o[1] } }; c ? c.emit(f) : i.emit(f) } e.hoverData.cxtDragged = !1, e.hoverData.which = null } else if (1 === e.hoverData.which) { if (null != c || e.dragData.didDrag || e.hoverData.selecting || e.hoverData.dragged || n(r) || (i.$((function (e) { return e.selected() })).unselect(), l.length > 0 && e.redrawHint("eles", !0), e.dragData.possibleDragElements = l = []), t(s, ["mouseup", "tapend", "vmouseup"], r, { position: { x: o[0], y: o[1] } }), e.dragData.didDrag || e.hoverData.dragged || e.hoverData.selecting || e.hoverData.isOverThresholdDrag || t(c, ["click", "tap", "vclick"], r, { position: { x: o[0], y: o[1] } }), s != c || e.dragData.didDrag || e.hoverData.selecting || null != s && s._private.selectable && (e.hoverData.dragging || ("additive" === i.selectionType() || u ? s.selected() ? s.unselect() : s.select() : u || (i.$(":selected").unmerge(s).unselect(), s.select())), e.redrawHint("eles", !0)), e.hoverData.selecting) { var h = i.collection(e.getAllInBox(a[0], a[1], a[2], a[3])); e.redrawHint("select", !0), h.length > 0 && e.redrawHint("eles", !0), i.emit("boxend"); var g = function (e) { return e.selectable() && !e.selected() }; "additive" === i.selectionType() || u || i.$(":selected").unmerge(h).unselect(), h.emit("box").stdFilter(g).select().emit("boxselect"), e.redraw() } if (e.hoverData.dragging && (e.hoverData.dragging = !1, e.redrawHint("select", !0), e.redrawHint("eles", !0), e.redraw()), !a[4]) { e.redrawHint("drag", !0), e.redrawHint("eles", !0); var m = c && c.grabbed(); p(l), m && c.emit("free") } } a[4] = 0, e.hoverData.down = null, e.hoverData.cxtStarted = !1, e.hoverData.draggingEles = !1, e.hoverData.selecting = !1, e.hoverData.isOverThresholdDrag = !1, e.dragData.didDrag = !1, e.hoverData.dragged = !1, e.hoverData.dragDelta = [], e.hoverData.mdownPos = null, e.hoverData.mdownGPos = null } }), !1), e.registerBinding(e.container, "wheel", (function (t) { if (!e.scrollingPage) { var n, r = e.cy, i = e.projectIntoViewport(t.clientX, t.clientY), o = [i[0] * r.zoom() + r.pan().x, i[1] * r.zoom() + r.pan().y]; e.hoverData.draggingEles || e.hoverData.dragging || e.hoverData.cxtStarted || 0 !== e.selection[4] ? t.preventDefault() : r.panningEnabled() && r.userPanningEnabled() && r.zoomingEnabled() && r.userZoomingEnabled() && (t.preventDefault(), e.data.wheelZooming = !0, clearTimeout(e.data.wheelTimeout), e.data.wheelTimeout = setTimeout((function () { e.data.wheelZooming = !1, e.redrawHint("eles", !0), e.redraw() }), 150), n = null != t.deltaY ? t.deltaY / -250 : null != t.wheelDeltaY ? t.wheelDeltaY / 1e3 : t.wheelDelta / 1e3, n *= e.wheelSensitivity, 1 === t.deltaMode && (n *= 33), r.zoom({ level: r.zoom() * Math.pow(10, n), renderedPosition: { x: o[0], y: o[1] } })) } }), !0), e.registerBinding(window, "scroll", (function (t) { e.scrollingPage = !0, clearTimeout(e.scrollingPageTimeout), e.scrollingPageTimeout = setTimeout((function () { e.scrollingPage = !1 }), 250) }), !0), e.registerBinding(e.container, "mouseout", (function (t) { var n = e.projectIntoViewport(t.clientX, t.clientY); e.cy.emit({ originalEvent: t, type: "mouseout", position: { x: n[0], y: n[1] } }) }), !1), e.registerBinding(e.container, "mouseover", (function (t) { var n = e.projectIntoViewport(t.clientX, t.clientY); e.cy.emit({ originalEvent: t, type: "mouseover", position: { x: n[0], y: n[1] } }) }), !1); var j, P, D, R, I = function (e, t, n, r) { return Math.sqrt((n - e) * (n - e) + (r - t) * (r - t)) }, N = function (e, t, n, r) { return (n - e) * (n - e) + (r - t) * (r - t) }; if (e.registerBinding(e.container, "touchstart", j = function (n) { if (T(n)) { e.touchData.capture = !0, e.data.bgActivePosistion = void 0; var r = e.cy, i = e.touchData.now, o = e.touchData.earlier; if (n.touches[0]) { var a = e.projectIntoViewport(n.touches[0].clientX, n.touches[0].clientY); i[0] = a[0], i[1] = a[1] } if (n.touches[1] && (a = e.projectIntoViewport(n.touches[1].clientX, n.touches[1].clientY), i[2] = a[0], i[3] = a[1]), n.touches[2] && (a = e.projectIntoViewport(n.touches[2].clientX, n.touches[2].clientY), i[4] = a[0], i[5] = a[1]), n.touches[1]) { p(e.dragData.touchDragEles); var s = e.findContainerClientCoords(); S = s[0], $ = s[1], C = s[2], _ = s[3], v = n.touches[0].clientX - S, b = n.touches[0].clientY - $, y = n.touches[1].clientX - S, x = n.touches[1].clientY - $, O = 0 <= v && v <= C && 0 <= y && y <= C && 0 <= b && b <= _ && 0 <= x && x <= _; var c = r.pan(), f = r.zoom(); if (w = I(v, b, y, x), k = N(v, b, y, x), E = [((A = [(v + y) / 2, (b + x) / 2])[0] - c.x) / f, (A[1] - c.y) / f], k < 4e4 && !n.touches[2]) { var h = e.findNearestElement(i[0], i[1], !0, !0), g = e.findNearestElement(i[2], i[3], !0, !0); return h && h.isNode() ? (h.activate().emit({ originalEvent: n, type: "cxttapstart", position: { x: i[0], y: i[1] } }), e.touchData.start = h) : g && g.isNode() ? (g.activate().emit({ originalEvent: n, type: "cxttapstart", position: { x: i[0], y: i[1] } }), e.touchData.start = g) : r.emit({ originalEvent: n, type: "cxttapstart", position: { x: i[0], y: i[1] } }), e.touchData.start && (e.touchData.start._private.grabbed = !1), e.touchData.cxt = !0, e.touchData.cxtDragged = !1, e.data.bgActivePosistion = void 0, void e.redraw() } } if (n.touches[2]); else if (n.touches[1]); else if (n.touches[0]) { var m = e.findNearestElements(i[0], i[1], !0, !0), j = m[0]; if (null != j && (j.activate(), e.touchData.start = j, e.touchData.starts = m, e.nodeIsGrabbable(j))) { var P = e.dragData.touchDragEles = [], D = null; e.redrawHint("eles", !0), e.redrawHint("drag", !0), j.selected() ? (D = r.$((function (t) { return t.selected() && e.nodeIsGrabbable(t) })), u(D, { addToList: P })) : d(j, { addToList: P }), l(j); var R = function (e) { return { originalEvent: n, type: e, position: { x: i[0], y: i[1] } } }; j.emit(R("grabon")), D ? D.forEach((function (e) { e.emit(R("grab")) })) : j.emit(R("grab")) } t(j, ["touchstart", "tapstart", "vmousedown"], n, { position: { x: i[0], y: i[1] } }), null == j && (e.data.bgActivePosistion = { x: a[0], y: a[1] }, e.redrawHint("select", !0), e.redraw()), e.touchData.singleTouchMoved = !1, e.touchData.singleTouchStartTime = +new Date, clearTimeout(e.touchData.tapholdTimeout), e.touchData.tapholdTimeout = setTimeout((function () { !1 !== e.touchData.singleTouchMoved || e.pinching || e.touchData.selecting || (t(e.touchData.start, ["taphold"], n, { position: { x: i[0], y: i[1] } }), e.touchData.start || r.$(":selected").unselect()) }), e.tapholdDuration) } if (n.touches.length >= 1) { for (var M = e.touchData.startPosition = [], z = 0; z < i.length; z++)M[z] = o[z] = i[z]; var L = n.touches[0]; e.touchData.startGPosition = [L.clientX, L.clientY] } } }, !1), e.registerBinding(window, "touchmove", P = function (n) { var i = e.touchData.capture; if (i || T(n)) { var s = e.selection, l = e.cy, c = e.touchData.now, d = e.touchData.earlier, f = l.zoom(); if (n.touches[0]) { var h = e.projectIntoViewport(n.touches[0].clientX, n.touches[0].clientY); c[0] = h[0], c[1] = h[1] } n.touches[1] && (h = e.projectIntoViewport(n.touches[1].clientX, n.touches[1].clientY), c[2] = h[0], c[3] = h[1]), n.touches[2] && (h = e.projectIntoViewport(n.touches[2].clientX, n.touches[2].clientY), c[4] = h[0], c[5] = h[1]); var g, m = e.touchData.startGPosition; if (i && n.touches[0] && m) { for (var A = [], C = 0; C < c.length; C++)A[C] = c[C] - d[C]; var _ = n.touches[0].clientX - m[0], j = _ * _, P = n.touches[0].clientY - m[1]; g = j + P * P >= e.touchTapThreshold2 } if (i && e.touchData.cxt) { n.preventDefault(); var D = n.touches[0].clientX - S, R = n.touches[0].clientY - $, M = n.touches[1].clientX - S, z = n.touches[1].clientY - $, L = N(D, R, M, z); if (L / k >= 2.25 || L >= 22500) { e.touchData.cxt = !1, e.data.bgActivePosistion = void 0, e.redrawHint("select", !0); var B = { originalEvent: n, type: "cxttapend", position: { x: c[0], y: c[1] } }; e.touchData.start ? (e.touchData.start.unactivate().emit(B), e.touchData.start = null) : l.emit(B) } } if (i && e.touchData.cxt) { B = { originalEvent: n, type: "cxtdrag", position: { x: c[0], y: c[1] } }, e.data.bgActivePosistion = void 0, e.redrawHint("select", !0), e.touchData.start ? e.touchData.start.emit(B) : l.emit(B), e.touchData.start && (e.touchData.start._private.grabbed = !1), e.touchData.cxtDragged = !0; var F = e.findNearestElement(c[0], c[1], !0, !0); e.touchData.cxtOver && F === e.touchData.cxtOver || (e.touchData.cxtOver && e.touchData.cxtOver.emit({ originalEvent: n, type: "cxtdragout", position: { x: c[0], y: c[1] } }), e.touchData.cxtOver = F, F && F.emit({ originalEvent: n, type: "cxtdragover", position: { x: c[0], y: c[1] } })) } else if (i && n.touches[2] && l.boxSelectionEnabled()) n.preventDefault(), e.data.bgActivePosistion = void 0, this.lastThreeTouch = +new Date, e.touchData.selecting || l.emit("boxstart"), e.touchData.selecting = !0, e.redrawHint("select", !0), s && 0 !== s.length && void 0 !== s[0] ? (s[2] = (c[0] + c[2] + c[4]) / 3, s[3] = (c[1] + c[3] + c[5]) / 3) : (s[0] = (c[0] + c[2] + c[4]) / 3, s[1] = (c[1] + c[3] + c[5]) / 3, s[2] = (c[0] + c[2] + c[4]) / 3 + 1, s[3] = (c[1] + c[3] + c[5]) / 3 + 1), s[4] = 1, e.touchData.selecting = !0, e.redraw(); else if (i && n.touches[1] && l.zoomingEnabled() && l.panningEnabled() && l.userZoomingEnabled() && l.userPanningEnabled()) { if (n.preventDefault(), e.data.bgActivePosistion = void 0, e.redrawHint("select", !0), ee = e.dragData.touchDragEles) { e.redrawHint("drag", !0); for (var q = 0; q < ee.length; q++) { var V = ee[q]._private; V.grabbed = !1, V.rscratch.inDragLayer = !1 } } D = n.touches[0].clientX - S, R = n.touches[0].clientY - $, M = n.touches[1].clientX - S, z = n.touches[1].clientY - $; var U = I(D, R, M, z), H = U / w; if (O) { var G = (D - v + (M - y)) / 2, W = (R - b + (z - x)) / 2, Y = l.zoom(), X = Y * H, Z = l.pan(), Q = E[0] * Y + Z.x, J = E[1] * Y + Z.y, K = { x: -X / Y * (Q - Z.x - G) + Q, y: -X / Y * (J - Z.y - W) + J }; if (e.touchData.start && e.touchData.start.active()) { var ee = e.dragData.touchDragEles; p(ee), e.redrawHint("drag", !0), e.redrawHint("eles", !0), e.touchData.start.unactivate().emit("free") } l.viewport({ zoom: X, pan: K, cancelOnFailedZoom: !0 }), w = U, v = D, b = R, y = M, x = z, e.pinching = !0 } n.touches[0] && (h = e.projectIntoViewport(n.touches[0].clientX, n.touches[0].clientY), c[0] = h[0], c[1] = h[1]), n.touches[1] && (h = e.projectIntoViewport(n.touches[1].clientX, n.touches[1].clientY), c[2] = h[0], c[3] = h[1]), n.touches[2] && (h = e.projectIntoViewport(n.touches[2].clientX, n.touches[2].clientY), c[4] = h[0], c[5] = h[1]) } else if (n.touches[0]) { var te = e.touchData.start, ne = e.touchData.last; if (e.hoverData.draggingEles || e.swipePanning || (F = e.findNearestElement(c[0], c[1], !0, !0)), i && null != te && n.preventDefault(), i && null != te && e.nodeIsDraggable(te)) if (g) { ee = e.dragData.touchDragEles; var re = !e.dragData.didDrag; re && u(l.collection(ee), { inDragLayer: !0 }); for (var ie = 0; ie < ee.length; ie++) { var oe = ee[ie]; if (e.nodeIsDraggable(oe) && oe.grabbed()) { e.dragData.didDrag = !0; var ae = oe.position(); r.number(A[0]) && r.number(A[1]) && (ae.x += A[0], ae.y += A[1]), re && (e.redrawHint("eles", !0), (le = e.touchData.dragDelta) && r.number(le[0]) && r.number(le[1]) && (ae.x += le[0], ae.y += le[1])) } } var se = l.collection(ee); se.dirtyCompoundBoundsCache(), se.emit("position drag"), e.hoverData.draggingEles = !0, e.redrawHint("drag", !0), e.touchData.startPosition[0] == d[0] && e.touchData.startPosition[1] == d[1] && e.redrawHint("eles", !0), e.redraw() } else { var le; 0 === (le = e.touchData.dragDelta = e.touchData.dragDelta || []).length ? (le.push(A[0]), le.push(A[1])) : (le[0] += A[0], le[1] += A[1]) } if (t(te || F, ["touchmove", "tapdrag", "vmousemove"], n, { position: { x: c[0], y: c[1] } }), te && te.grabbed() || F == ne || (ne && ne.emit({ originalEvent: n, type: "tapdragout", position: { x: c[0], y: c[1] } }), F && F.emit({ originalEvent: n, type: "tapdragover", position: { x: c[0], y: c[1] } })), e.touchData.last = F, i) for (q = 0; q < c.length; q++)c[q] && e.touchData.startPosition[q] && g && (e.touchData.singleTouchMoved = !0); i && (null == te || te.isEdge()) && l.panningEnabled() && l.userPanningEnabled() && (a(te, e.touchData.starts) && (n.preventDefault(), e.swipePanning ? l.panBy({ x: A[0] * f, y: A[1] * f }) : g && (e.swipePanning = !0, l.panBy({ x: _ * f, y: P * f }), te && (te.unactivate(), e.data.bgActivePosistion || (e.data.bgActivePosistion = o.array2point(e.touchData.startPosition)), e.redrawHint("select", !0), e.touchData.start = null))), h = e.projectIntoViewport(n.touches[0].clientX, n.touches[0].clientY), c[0] = h[0], c[1] = h[1]) } for (C = 0; C < c.length; C++)d[C] = c[C] } }, !1), e.registerBinding(window, "touchcancel", D = function (t) { var n = e.touchData.start; e.touchData.capture = !1, n && n.unactivate() }), e.registerBinding(window, "touchend", R = function (n) { var r = e.touchData.start; if (e.touchData.capture) { e.touchData.capture = !1, n.preventDefault(); var i = e.selection; e.swipePanning = !1, e.hoverData.draggingEles = !1; var o, a = e.cy, s = a.zoom(), l = e.touchData.now, c = e.touchData.earlier; if (n.touches[0]) { var u = e.projectIntoViewport(n.touches[0].clientX, n.touches[0].clientY); l[0] = u[0], l[1] = u[1] } if (n.touches[1] && (u = e.projectIntoViewport(n.touches[1].clientX, n.touches[1].clientY), l[2] = u[0], l[3] = u[1]), n.touches[2] && (u = e.projectIntoViewport(n.touches[2].clientX, n.touches[2].clientY), l[4] = u[0], l[5] = u[1]), r && r.unactivate(), e.touchData.cxt) { if (o = { originalEvent: n, type: "cxttapend", position: { x: l[0], y: l[1] } }, r ? r.emit(o) : a.emit(o), !e.touchData.cxtDragged) { var d = { originalEvent: n, type: "cxttap", position: { x: l[0], y: l[1] } }; r ? r.emit(d) : a.emit(d) } return e.touchData.start && (e.touchData.start._private.grabbed = !1), e.touchData.cxt = !1, e.touchData.start = null, void e.redraw() } if (!n.touches[2] && a.boxSelectionEnabled() && e.touchData.selecting) { e.touchData.selecting = !1; var f = a.collection(e.getAllInBox(i[0], i[1], i[2], i[3])); i[0] = void 0, i[1] = void 0, i[2] = void 0, i[3] = void 0, i[4] = 0, e.redrawHint("select", !0), a.emit("boxend"), f.emit("box").stdFilter((function (e) { return e.selectable() && !e.selected() })).select().emit("boxselect"), f.nonempty() && e.redrawHint("eles", !0), e.redraw() } if (null != r && r.unactivate(), n.touches[2]) e.data.bgActivePosistion = void 0, e.redrawHint("select", !0); else if (n.touches[1]); else if (n.touches[0]); else if (!n.touches[0]) { e.data.bgActivePosistion = void 0, e.redrawHint("select", !0); var h = e.dragData.touchDragEles; if (null != r) { var g = r._private.grabbed; p(h), e.redrawHint("drag", !0), e.redrawHint("eles", !0), g && r.emit("free"), t(r, ["touchend", "tapend", "vmouseup", "tapdragout"], n, { position: { x: l[0], y: l[1] } }), r.unactivate(), e.touchData.start = null } else { var m = e.findNearestElement(l[0], l[1], !0, !0); t(m, ["touchend", "tapend", "vmouseup", "tapdragout"], n, { position: { x: l[0], y: l[1] } }) } var v = e.touchData.startPosition[0] - l[0], b = v * v, y = e.touchData.startPosition[1] - l[1], x = (b + y * y) * s * s; null != r && !e.dragData.didDrag && r._private.selectable && x < e.touchTapThreshold2 && !e.pinching && ("single" === a.selectionType() ? (a.$(":selected").unmerge(r).unselect(), r.select()) : r.selected() ? r.unselect() : r.select(), e.redrawHint("eles", !0)), e.touchData.singleTouchMoved || t(r, ["tap", "vclick"], n, { position: { x: l[0], y: l[1] } }), e.touchData.singleTouchMoved = !0 } for (var w = 0; w < l.length; w++)c[w] = l[w]; e.dragData.didDrag = !1, 0 === n.touches.length && (e.touchData.dragDelta = [], e.touchData.startPosition = null, e.touchData.startGPosition = null), n.touches.length < 2 && (e.pinching = !1, e.redrawHint("eles", !0), e.redraw()) } }, !1), "undefined" == typeof TouchEvent) { var M = [], z = function (e) { return { clientX: e.clientX, clientY: e.clientY, force: 1, identifier: e.pointerId, pageX: e.pageX, pageY: e.pageY, radiusX: e.width / 2, radiusY: e.height / 2, screenX: e.screenX, screenY: e.screenY, target: e.target } }, L = function (e) { M.push(function (e) { return { event: e, touch: z(e) } }(e)) }, B = function (e) { for (var t = 0; t < M.length; t++)if (M[t].event.pointerId === e.pointerId) return void M.splice(t, 1) }, F = function (e) { e.touches = M.map((function (e) { return e.touch })) }, q = function (e) { return "mouse" === e.pointerType || 4 === e.pointerType }; e.registerBinding(e.container, "pointerdown", (function (e) { q(e) || (e.preventDefault(), L(e), F(e), j(e)) })), e.registerBinding(e.container, "pointerup", (function (e) { q(e) || (B(e), F(e), R(e)) })), e.registerBinding(e.container, "pointercancel", (function (e) { q(e) || (B(e), F(e), D()) })), e.registerBinding(e.container, "pointermove", (function (e) { q(e) || (e.preventDefault(), function (e) { var t = M.filter((function (t) { return t.event.pointerId === e.pointerId }))[0]; t.event = e, t.touch = z(e) }(e), F(e), P(e)) })) } }, e.exports = a }, function (e, t, n) { "use strict"; var r = n(2), i = { generatePolygon: function (e, t) { return this.nodeShapes[e] = { renderer: this, name: e, points: t, draw: function (e, t, n, r, i) { this.renderer.nodeShapeImpl("polygon", e, t, n, r, i, this.points) }, intersectLine: function (e, t, n, i, o, a, s) { return r.polygonIntersectLine(o, a, this.points, e, t, n / 2, i / 2, s) }, checkPoint: function (e, t, n, i, o, a, s) { return r.pointInsidePolygon(e, t, this.points, a, s, i, o, [0, -1], n) } } }, generateEllipse: function () { return this.nodeShapes.ellipse = { renderer: this, name: "ellipse", draw: function (e, t, n, r, i) { this.renderer.nodeShapeImpl(this.name, e, t, n, r, i) }, intersectLine: function (e, t, n, i, o, a, s) { return r.intersectLineEllipse(o, a, e, t, n / 2 + s, i / 2 + s) }, checkPoint: function (e, t, n, i, o, a, s) { return r.checkInEllipse(e, t, i, o, a, s, n) } } }, generateRoundRectangle: function () { return this.nodeShapes.roundrectangle = { renderer: this, name: "roundrectangle", points: r.generateUnitNgonPointsFitToSquare(4, 0), draw: function (e, t, n, r, i) { this.renderer.nodeShapeImpl(this.name, e, t, n, r, i) }, intersectLine: function (e, t, n, i, o, a, s) { return r.roundRectangleIntersectLine(o, a, e, t, n, i, s) }, checkPoint: function (e, t, n, i, o, a, s) { var l = r.getRoundRectangleRadius(i, o), c = 2 * l; return !!(r.pointInsidePolygon(e, t, this.points, a, s, i, o - c, [0, -1], n) || r.pointInsidePolygon(e, t, this.points, a, s, i - c, o, [0, -1], n) || r.checkInEllipse(e, t, c, c, a - i / 2 + l, s - o / 2 + l, n) || r.checkInEllipse(e, t, c, c, a + i / 2 - l, s - o / 2 + l, n) || r.checkInEllipse(e, t, c, c, a + i / 2 - l, s + o / 2 - l, n) || r.checkInEllipse(e, t, c, c, a - i / 2 + l, s + o / 2 - l, n)) } } }, generateCutRectangle: function () { return this.nodeShapes.cutrectangle = { renderer: this, name: "cutrectangle", cornerLength: r.getCutRectangleCornerLength(), points: r.generateUnitNgonPointsFitToSquare(4, 0), draw: function (e, t, n, r, i) { this.renderer.nodeShapeImpl(this.name, e, t, n, r, i) }, generateCutTrianglePts: function (e, t, n, r) { var i = this.cornerLength, o = t / 2, a = e / 2, s = n - a, l = n + a, c = r - o, u = r + o; return { topLeft: [s, c + i, s + i, c, s + i, c + i], topRight: [l - i, c, l, c + i, l - i, c + i], bottomRight: [l, u - i, l - i, u, l - i, u - i], bottomLeft: [s + i, u, s, u - i, s + i, u - i] } }, intersectLine: function (e, t, n, i, o, a, s) { var l = this.generateCutTrianglePts(n + 2 * s, i + 2 * s, e, t), c = [].concat.apply([], [l.topLeft.splice(0, 4), l.topRight.splice(0, 4), l.bottomRight.splice(0, 4), l.bottomLeft.splice(0, 4)]); return r.polygonIntersectLine(o, a, c, e, t) }, checkPoint: function (e, t, n, i, o, a, s) { if (r.pointInsidePolygon(e, t, this.points, a, s, i, o - 2 * this.cornerLength, [0, -1], n)) return !0; if (r.pointInsidePolygon(e, t, this.points, a, s, i - 2 * this.cornerLength, o, [0, -1], n)) return !0; var l = this.generateCutTrianglePts(i, o, a, s); return r.pointInsidePolygonPoints(e, t, l.topLeft) || r.pointInsidePolygonPoints(e, t, l.topRight) || r.pointInsidePolygonPoints(e, t, l.bottomRight) || r.pointInsidePolygonPoints(e, t, l.bottomLeft) } } }, generateBarrel: function () { return this.nodeShapes.barrel = { renderer: this, name: "barrel", points: r.generateUnitNgonPointsFitToSquare(4, 0), draw: function (e, t, n, r, i) { this.renderer.nodeShapeImpl(this.name, e, t, n, r, i) }, intersectLine: function (e, t, n, i, o, a, s) { var l = this.generateBarrelBezierPts(n + 2 * s, i + 2 * s, e, t), c = function (e) { var t = r.qbezierPtAt({ x: e[0], y: e[1] }, { x: e[2], y: e[3] }, { x: e[4], y: e[5] }, .15), n = r.qbezierPtAt({ x: e[0], y: e[1] }, { x: e[2], y: e[3] }, { x: e[4], y: e[5] }, .5), i = r.qbezierPtAt({ x: e[0], y: e[1] }, { x: e[2], y: e[3] }, { x: e[4], y: e[5] }, .85); return [e[0], e[1], t.x, t.y, n.x, n.y, i.x, i.y, e[4], e[5]] }, u = [].concat(c(l.topLeft), c(l.topRight), c(l.bottomRight), c(l.bottomLeft)); return r.polygonIntersectLine(o, a, u, e, t) }, generateBarrelBezierPts: function (e, t, n, i) { var o = t / 2, a = e / 2, s = n - a, l = n + a, c = i - o, u = i + o, d = r.getBarrelCurveConstants(e, t), p = d.heightOffset, f = d.widthOffset, h = d.ctrlPtOffsetPct * e, g = { topLeft: [s, c + p, s + h, c, s + f, c], topRight: [l - f, c, l - h, c, l, c + p], bottomRight: [l, u - p, l - h, u, l - f, u], bottomLeft: [s + f, u, s + h, u, s, u - p] }; return g.topLeft.isTop = !0, g.topRight.isTop = !0, g.bottomLeft.isBottom = !0, g.bottomRight.isBottom = !0, g }, checkPoint: function (e, t, n, i, o, a, s) { var l = r.getBarrelCurveConstants(i, o), c = l.heightOffset, u = l.widthOffset; if (r.pointInsidePolygon(e, t, this.points, a, s, i, o - 2 * c, [0, -1], n)) return !0; if (r.pointInsidePolygon(e, t, this.points, a, s, i - 2 * u, o, [0, -1], n)) return !0; for (var d = this.generateBarrelBezierPts(i, o, a, s), p = function (e, t, n) { var i = n[4], o = n[2], a = n[0], s = n[5], l = n[1], c = Math.min(i, a), u = Math.max(i, a), d = Math.min(s, l), p = Math.max(s, l); if (c <= e && e <= u && d <= t && t <= p) { var f = r.bezierPtsToQuadCoeff(i, o, a), h = r.solveQuadratic(f[0], f[1], f[2], e).filter((function (e) { return 0 <= e && e <= 1 })); if (h.length > 0) return h[0] } return null }, f = Object.keys(d), h = 0; h < f.length; h++) { var g = d[f[h]], m = p(e, t, g); if (null != m) { var v = g[5], b = g[3], y = g[1], x = r.qbezierAt(v, b, y, m); if (g.isTop && x <= t) return !0; if (g.isBottom && t <= x) return !0 } } return !1 } } }, generateBottomRoundrectangle: function () { return this.nodeShapes.bottomroundrectangle = { renderer: this, name: "bottomroundrectangle", points: r.generateUnitNgonPointsFitToSquare(4, 0), draw: function (e, t, n, r, i) { this.renderer.nodeShapeImpl(this.name, e, t, n, r, i) }, intersectLine: function (e, t, n, i, o, a, s) { var l = e - (n / 2 + s), c = t - (i / 2 + s), u = c, d = e + (n / 2 + s), p = r.finiteLinesIntersect(o, a, e, t, l, c, d, u, !1); return p.length > 0 ? p : r.roundRectangleIntersectLine(o, a, e, t, n, i, s) }, checkPoint: function (e, t, n, i, o, a, s) { var l = r.getRoundRectangleRadius(i, o), c = 2 * l; if (r.pointInsidePolygon(e, t, this.points, a, s, i, o - c, [0, -1], n)) return !0; if (r.pointInsidePolygon(e, t, this.points, a, s, i - c, o, [0, -1], n)) return !0; var u = i / 2 + 2 * n, d = o / 2 + 2 * n, p = [a - u, s - d, a - u, s, a + u, s, a + u, s - d]; return !!r.pointInsidePolygonPoints(e, t, p) || !!r.checkInEllipse(e, t, c, c, a + i / 2 - l, s + o / 2 - l, n) || !!r.checkInEllipse(e, t, c, c, a - i / 2 + l, s + o / 2 - l, n) } } }, registerNodeShapes: function () { var e = this.nodeShapes = {}, t = this; this.generateEllipse(), this.generatePolygon("triangle", r.generateUnitNgonPointsFitToSquare(3, 0)), this.generatePolygon("rectangle", r.generateUnitNgonPointsFitToSquare(4, 0)), e.square = e.rectangle, this.generateRoundRectangle(), this.generateCutRectangle(), this.generateBarrel(), this.generateBottomRoundrectangle(), this.generatePolygon("diamond", [0, 1, 1, 0, 0, -1, -1, 0]), this.generatePolygon("pentagon", r.generateUnitNgonPointsFitToSquare(5, 0)), this.generatePolygon("hexagon", r.generateUnitNgonPointsFitToSquare(6, 0)), this.generatePolygon("heptagon", r.generateUnitNgonPointsFitToSquare(7, 0)), this.generatePolygon("octagon", r.generateUnitNgonPointsFitToSquare(8, 0)); var n = new Array(20), i = r.generateUnitNgonPoints(5, 0), o = r.generateUnitNgonPoints(5, Math.PI / 5), a = .5 * (3 - Math.sqrt(5)); a *= 1.57; for (var s = 0; s < o.length / 2; s++)o[2 * s] *= a, o[2 * s + 1] *= a; for (s = 0; s < 5; s++)n[4 * s] = i[2 * s], n[4 * s + 1] = i[2 * s + 1], n[4 * s + 2] = o[2 * s], n[4 * s + 3] = o[2 * s + 1]; n = r.fitPolygonToSquare(n), this.generatePolygon("star", n), this.generatePolygon("vee", [-1, -1, 0, -.333, 1, -1, 0, 1]), this.generatePolygon("rhomboid", [-1, -1, .333, -1, 1, 1, -.333, 1]), this.generatePolygon("concavehexagon", [-1, -.95, -.75, 0, -1, .95, 1, .95, .75, 0, 1, -.95]), this.generatePolygon("tag", [-1, -1, .25, -1, 1, 0, .25, 1, -1, 1]), e.makePolygon = function (e) { var n, r = "polygon-" + e.join("$"); return (n = this[r]) ? n : t.generatePolygon(r, e) } } }; e.exports = i }, function (e, t, n) { "use strict"; var r = n(1), i = { timeToRender: function () { return this.redrawTotalTime / this.redrawCount }, redraw: function (e) { e = e || r.staticEmptyObject(); var t = this; void 0 === t.averageRedrawTime && (t.averageRedrawTime = 0), void 0 === t.lastRedrawTime && (t.lastRedrawTime = 0), void 0 === t.lastDrawTime && (t.lastDrawTime = 0), t.requestedFrame = !0, t.renderOptions = e }, beforeRender: function (e, t) { if (!this.destroyed) { t = t || 0; var n = this.beforeRenderCallbacks; n.push({ fn: e, priority: t }), n.sort((function (e, t) { return t.priority - e.priority })) } } }, o = function (e, t, n) { for (var r = e.beforeRenderCallbacks, i = 0; i < r.length; i++)r[i].fn(t, n) }; i.startRenderLoop = function () { var e = this; e.renderLoopStarted || (e.renderLoopStarted = !0, r.requestAnimationFrame((function t(n) { if (!e.destroyed) { if (e.requestedFrame && !e.skipFrame) { o(e, !0, n); var i = r.performanceNow(); e.render(e.renderOptions); var a = e.lastDrawTime = r.performanceNow(); void 0 === e.averageRedrawTime && (e.averageRedrawTime = a - i), void 0 === e.redrawCount && (e.redrawCount = 0), e.redrawCount++, void 0 === e.redrawTotalTime && (e.redrawTotalTime = 0); var s = a - i; e.redrawTotalTime += s, e.lastRedrawTime = s, e.averageRedrawTime = e.averageRedrawTime / 2 + s / 2, e.requestedFrame = !1 } else o(e, !1, n); e.skipFrame = !1, r.requestAnimationFrame(t) } }))) }, e.exports = i }, function (e, t, n) { "use strict"; var r = n(1), i = n(0), o = n(124), a = n(125), s = c, l = c.prototype; function c(e) { var t = this; t.data = { canvases: new Array(l.CANVAS_LAYERS), contexts: new Array(l.CANVAS_LAYERS), canvasNeedsRedraw: new Array(l.CANVAS_LAYERS), bufferCanvases: new Array(l.BUFFER_COUNT), bufferContexts: new Array(l.CANVAS_LAYERS) }; var n = "-webkit-tap-highlight-color: rgba(0,0,0,0);"; t.data.canvasContainer = document.createElement("div"); var r = t.data.canvasContainer.style; t.data.canvasContainer.setAttribute("style", n), r.position = "relative", r.zIndex = "0", r.overflow = "hidden"; var s = e.cy.container(); s.appendChild(t.data.canvasContainer), (s.getAttribute("style") || "").indexOf(n) < 0 && s.setAttribute("style", (s.getAttribute("style") || "") + n); for (var c = 0; c < l.CANVAS_LAYERS; c++) { var u = t.data.canvases[c] = document.createElement("canvas"); t.data.contexts[c] = u.getContext("2d"), u.setAttribute("style", "-webkit-user-select: none; -moz-user-select: -moz-none; user-select: none; -webkit-tap-highlight-color: rgba(0,0,0,0); outline-style: none;" + (i.ms() ? " -ms-touch-action: none; touch-action: none; " : "")), u.style.position = "absolute", u.setAttribute("data-id", "layer" + c), u.style.zIndex = String(l.CANVAS_LAYERS - c), t.data.canvasContainer.appendChild(u), t.data.canvasNeedsRedraw[c] = !1 } for (t.data.topCanvas = t.data.canvases[0], t.data.canvases[l.NODE].setAttribute("data-id", "layer" + l.NODE + "-node"), t.data.canvases[l.SELECT_BOX].setAttribute("data-id", "layer" + l.SELECT_BOX + "-selectbox"), t.data.canvases[l.DRAG].setAttribute("data-id", "layer" + l.DRAG + "-drag"), c = 0; c < l.BUFFER_COUNT; c++)t.data.bufferCanvases[c] = document.createElement("canvas"), t.data.bufferContexts[c] = t.data.bufferCanvases[c].getContext("2d"), t.data.bufferCanvases[c].style.position = "absolute", t.data.bufferCanvases[c].setAttribute("data-id", "buffer" + c), t.data.bufferCanvases[c].style.zIndex = String(-c - 1), t.data.bufferCanvases[c].style.visibility = "hidden"; t.pathsEnabled = !0, t.data.eleTxrCache = new o(t), t.data.lyrTxrCache = new a(t, t.data.eleTxrCache), t.onUpdateEleCalcs((function (e, n) { for (var r = 0; r < n.length; r++) { var i = n[r], o = i._private.rstyle, a = o.dirtyEvents; if (i.isNode() && a && 1 === a.length && a.position); else if (t.data.eleTxrCache.invalidateElement(i), i.isParent() && a.style) { var s = o.prevParentOpacity, l = i.pstyle("opacity").pfValue; if (o.prevParentOpacity = l, s !== l) for (var c = i.descendants(), u = 0; u < c.length; u++)t.data.eleTxrCache.invalidateElement(c[u]) } } n.length > 0 && t.data.lyrTxrCache.invalidateElements(n) })) } l.CANVAS_LAYERS = 3, l.SELECT_BOX = 0, l.DRAG = 1, l.NODE = 2, l.BUFFER_COUNT = 3, l.TEXTURE_BUFFER = 0, l.MOTIONBLUR_BUFFER_NODE = 1, l.MOTIONBLUR_BUFFER_DRAG = 2, l.redrawHint = function (e, t) { var n = this; switch (e) { case "eles": n.data.canvasNeedsRedraw[l.NODE] = t; break; case "drag": n.data.canvasNeedsRedraw[l.DRAG] = t; break; case "select": n.data.canvasNeedsRedraw[l.SELECT_BOX] = t } }; var u = "undefined" != typeof Path2D; l.path2dEnabled = function (e) { if (void 0 === e) return this.pathsEnabled; this.pathsEnabled = !!e }, l.usePaths = function () { return u && this.pathsEnabled }, [n(126), n(127), n(128), n(129), n(130), n(131), n(132), n(133), n(134), n(135)].forEach((function (e) { r.extend(l, e) })), e.exports = s }, function (e, t, n) { "use strict"; var r = n(2), i = n(1), o = n(9), a = n(19), s = { dequeue: "dequeue", downscale: "downscale", highQuality: "highQuality" }, l = function (e) { this.renderer = e, this.onDequeues = [], this.setupDequeueing() }, c = l.prototype; c.reasons = s, c.getTextureQueue = function (e) { return this.eleImgCaches = this.eleImgCaches || {}, this.eleImgCaches[e] = this.eleImgCaches[e] || [] }, c.getRetiredTextureQueue = function (e) { var t = this.eleImgCaches.retired = this.eleImgCaches.retired || {}; return t[e] = t[e] || [] }, c.getElementQueue = function () { return this.eleCacheQueue = this.eleCacheQueue || new o((function (e, t) { return t.reqs - e.reqs })) }, c.getElementIdToQueue = function () { return this.eleIdToCacheQueue = this.eleIdToCacheQueue || {} }, c.getElement = function (e, t, n, i, o) { var a = this, l = this.renderer, c = e._private.rscratch, u = l.cy.zoom(); if (0 === t.w || 0 === t.h || !e.visible()) return null; if (null == i && (i = Math.ceil(r.log2(u * n))), i < -4) i = -4; else if (u >= 3.99 || i > 2) return null; var d, p = Math.pow(2, i), f = t.h * p, h = t.w * p, g = c.imgCaches = c.imgCaches || {}, m = g[i]; if (m) return m; if (d = f <= 25 ? 25 : f <= 50 ? 50 : 50 * Math.ceil(f / 50), f > 1024 || h > 1024 || e.isEdge() || e.isParent()) return null; var v = a.getTextureQueue(d), b = v[v.length - 2], y = function () { return a.recycleTexture(d, h) || a.addTexture(d, h) }; b || (b = v[v.length - 1]), b || (b = y()), b.width - b.usedWidth < h && (b = y()); for (var x, w = l.eleTextBiggerThanMin(e, p), k = function (e) { return e && e.scaledLabelShown === w }, A = o && o === s.dequeue, E = o && o === s.highQuality, S = o && o === s.downscale, $ = i + 1; $ <= 2; $++)if (T = g[$]) { x = T; break } var C = x && x.level === i + 1 ? x : null, _ = function () { b.context.drawImage(C.texture.canvas, C.x, 0, C.width, C.height, b.usedWidth, 0, h, f) }; if (b.context.setTransform(1, 0, 0, 1, 0, 0), b.context.clearRect(b.usedWidth, 0, h, d), k(C)) _(); else if (k(x)) { if (!E) return a.queueElement(e, x.level - 1), x; for ($ = x.level; $ > i; $--)C = a.getElement(e, t, n, $, s.downscale); _() } else { var O; if (!A && !E && !S) for ($ = i - 1; $ >= -4; $--) { var T; if (T = g[$]) { O = T; break } } if (k(O)) return a.queueElement(e, i), O; b.context.translate(b.usedWidth, 0), b.context.scale(p, p), l.drawElement(b.context, e, t, w), b.context.scale(1 / p, 1 / p), b.context.translate(-b.usedWidth, 0) } return m = g[i] = { ele: e, x: b.usedWidth, texture: b, level: i, scale: p, width: h, height: f, scaledLabelShown: w }, b.usedWidth += Math.ceil(h + 8), b.eleCaches.push(m), a.checkTextureFullness(b), m }, c.invalidateElement = function (e) { var t = e._private.rscratch.imgCaches; if (t) for (var n = -4; n <= 2; n++) { var r = t[n]; if (r) { var o = r.texture; o.invalidatedWidth += r.width, t[n] = null, i.removeFromArray(o.eleCaches, r), this.removeFromQueue(e), this.checkTextureUtility(o) } } }, c.checkTextureUtility = function (e) { e.invalidatedWidth >= .5 * e.width && this.retireTexture(e) }, c.checkTextureFullness = function (e) { var t = this.getTextureQueue(e.height); e.usedWidth / e.width > .8 && e.fullnessChecks >= 10 ? i.removeFromArray(t, e) : e.fullnessChecks++ }, c.retireTexture = function (e) { var t = e.height, n = this.getTextureQueue(t); i.removeFromArray(n, e), e.retired = !0; for (var r = e.eleCaches, o = 0; o < r.length; o++) { var a = r[o], s = a.ele, l = a.level, c = s._private.rscratch.imgCaches; c && (c[l] = null) } i.clearArray(r), this.getRetiredTextureQueue(t).push(e) }, c.addTexture = function (e, t) { var n = {}; return this.getTextureQueue(e).push(n), n.eleCaches = [], n.height = e, n.width = Math.max(1024, t), n.usedWidth = 0, n.invalidatedWidth = 0, n.fullnessChecks = 0, n.canvas = document.createElement("canvas"), n.canvas.width = n.width, n.canvas.height = n.height, n.context = n.canvas.getContext("2d"), n }, c.recycleTexture = function (e, t) { for (var n = this.getTextureQueue(e), r = this.getRetiredTextureQueue(e), o = 0; o < r.length; o++) { var a = r[o]; if (a.width >= t) return a.retired = !1, a.usedWidth = 0, a.invalidatedWidth = 0, a.fullnessChecks = 0, i.clearArray(a.eleCaches), a.context.setTransform(1, 0, 0, 1, 0, 0), a.context.clearRect(0, 0, a.width, a.height), i.removeFromArray(r, a), n.push(a), a } }, c.queueElement = function (e, t) { var n = this.getElementQueue(), r = this.getElementIdToQueue(), i = e.id(), o = r[i]; if (o) o.level = Math.max(o.level, t), o.reqs++, n.updateItem(o); else { var a = { ele: e, level: t, reqs: 1 }; n.push(a), r[i] = a } }, c.dequeue = function (e) { for (var t = this.getElementQueue(), n = this.getElementIdToQueue(), r = [], i = 0; i < 1 && t.size() > 0; i++) { var o = t.pop(), a = o.ele; if (null == a._private.rscratch.imgCaches[o.level]) { n[a.id()] = null, r.push(o); var l = a.boundingBox(); this.getElement(a, l, e, o.level, s.dequeue) } } return r }, c.removeFromQueue = function (e) { var t = this.getElementQueue(), n = this.getElementIdToQueue(), r = n[e.id()]; null != r && (r.reqs = i.MAX_INT, t.updateItem(r), t.pop(), n[e.id()] = null) }, c.onDequeue = function (e) { this.onDequeues.push(e) }, c.offDequeue = function (e) { i.removeFromArray(this.onDequeues, e) }, c.setupDequeueing = a.setupDequeueing({ deqRedrawThreshold: 100, deqCost: .15, deqAvgCost: .1, deqNoDrawCost: .9, deqFastCost: .9, deq: function (e, t, n) { return e.dequeue(t, n) }, onDeqd: function (e, t) { for (var n = 0; n < e.onDequeues.length; n++)(0, e.onDequeues[n])(t) }, shouldRedraw: function (e, t, n, i) { for (var o = 0; o < t.length; o++) { var a = t[o].ele.boundingBox(); if (r.boundingBoxesIntersect(a, i)) return !0 } return !1 }, priority: function (e) { return e.renderer.beforeRenderPriorities.eleTxrDeq } }), e.exports = l }, function (e, t, n) { "use strict"; var r = n(1), i = n(2), o = n(9), a = n(0), s = n(19), l = function (e, t) { var n = this, i = n.renderer = e; n.layersByLevel = {}, n.firstGet = !0, n.lastInvalidationTime = r.performanceNow() - 500, n.skipping = !1, i.beforeRender((function (e, t) { t - n.lastInvalidationTime <= 250 ? n.skipping = !0 : n.skipping = !1 })), n.layersQueue = new o((function (e, t) { return t.reqs - e.reqs })), n.eleTxrCache = t, n.setupEleCacheInvalidation(), n.setupDequeueing() }, c = l.prototype, u = 0, d = Math.pow(2, 53) - 1; function p(e, t) { null != e.imageSmoothingEnabled ? e.imageSmoothingEnabled = t : (e.webkitImageSmoothingEnabled = t, e.mozImageSmoothingEnabled = t, e.msImageSmoothingEnabled = t) } c.makeLayer = function (e, t) { var n = Math.pow(2, t), r = Math.ceil(e.w * n), i = Math.ceil(e.h * n), o = document.createElement("canvas"); o.width = r, o.height = i; var a = { id: u = ++u % d, bb: e, level: t, width: r, height: i, canvas: o, context: o.getContext("2d"), eles: [], elesQueue: [], reqs: 0 }, s = a.context, l = -a.bb.x1, c = -a.bb.y1; return s.scale(n, n), s.translate(l, c), a }, c.getLayers = function (e, t, n) { var o = this, a = o.renderer.cy.zoom(), s = o.firstGet; if (o.firstGet = !1, null == n) if ((n = Math.ceil(i.log2(a * t))) < -4) n = -4; else if (a >= 3.99 || n > 2) return null; o.validateLayersElesOrdering(n, e); var l, c, u = o.layersByLevel, d = Math.pow(2, n), p = u[n] = u[n] || []; if (o.levelIsComplete(n, e)) return p; !function () { var t = function (t) { if (o.validateLayersElesOrdering(t, e), o.levelIsComplete(t, e)) return c = u[t], !0 }, i = function (e) { if (!c) for (var r = n + e; -4 <= r && r <= 2 && !t(r); r += e); }; i(1), i(-1); for (var a = p.length - 1; a >= 0; a--) { var s = p[a]; s.invalid && r.removeFromArray(p, s) } }(); var f = function (t) { var r = (t = t || {}).after; if (function () { if (!l) { l = i.makeBoundingBox(); for (var t = 0; t < e.length; t++)i.updateBoundingBox(l, e[t].boundingBox()) } }(), l.w * d * (l.h * d) > 16e6) return null; var a = o.makeLayer(l, n); if (null != r) { var s = p.indexOf(r) + 1; p.splice(s, 0, a) } else (void 0 === t.insert || t.insert) && p.unshift(a); return a }; if (o.skipping && !s) return null; for (var h = null, g = e.length / 1, m = !s, v = 0; v < e.length; v++) { var b = e[v], y = b._private.rscratch, x = y.imgLayerCaches = y.imgLayerCaches || {}, w = x[n]; if (w) h = w; else { if ((!h || h.eles.length >= g || !i.boundingBoxInBoundingBox(h.bb, b.boundingBox())) && !(h = f({ insert: !0, after: h }))) return null; c || m ? o.queueLayer(h, b) : o.drawEleInLayer(h, b, n, t), h.eles.push(b), x[n] = h } } return c || (m ? null : p) }, c.getEleLevelForLayerLevel = function (e, t) { return e }, c.drawEleInLayer = function (e, t, n, r) { var i = this.renderer, o = e.context, a = t.boundingBox(); if (0 !== a.w && 0 !== a.h && t.visible()) { var s = this.eleTxrCache, l = s.reasons.highQuality; n = this.getEleLevelForLayerLevel(n, r); var c = s.getElement(t, a, null, n, l); c ? (p(o, !1), o.drawImage(c.texture.canvas, c.x, 0, c.width, c.height, a.x1, a.y1, a.w, a.h), p(o, !0)) : i.drawElement(o, t) } }, c.levelIsComplete = function (e, t) { var n = this.layersByLevel[e]; if (!n || 0 === n.length) return !1; for (var r = 0, i = 0; i < n.length; i++) { var o = n[i]; if (o.reqs > 0) return !1; if (o.invalid) return !1; r += o.eles.length } return r === t.length }, c.validateLayersElesOrdering = function (e, t) { var n = this.layersByLevel[e]; if (n) for (var r = 0; r < n.length; r++) { for (var i = n[r], o = -1, a = 0; a < t.length; a++)if (i.eles[0] === t[a]) { o = a; break } if (o < 0) this.invalidateLayer(i); else { var s = o; for (a = 0; a < i.eles.length; a++)if (i.eles[a] !== t[s + a]) { this.invalidateLayer(i); break } } } }, c.updateElementsInLayers = function (e, t) { for (var n = a.element(e[0]), r = 0; r < e.length; r++)for (var i = n ? null : e[r], o = n ? e[r] : e[r].ele, s = o._private.rscratch, l = s.imgLayerCaches = s.imgLayerCaches || {}, c = -4; c <= 2; c++) { var u = l[c]; u && (i && this.getEleLevelForLayerLevel(u.level) !== i.level || t(u, o, i)) } }, c.haveLayers = function () { for (var e = !1, t = -4; t <= 2; t++) { var n = this.layersByLevel[t]; if (n && n.length > 0) { e = !0; break } } return e }, c.invalidateElements = function (e) { var t = this; t.lastInvalidationTime = r.performanceNow(), 0 !== e.length && t.haveLayers() && t.updateElementsInLayers(e, (function (e, n, r) { t.invalidateLayer(e) })) }, c.invalidateLayer = function (e) { if (this.lastInvalidationTime = r.performanceNow(), !e.invalid) { var t = e.level, n = e.eles, i = this.layersByLevel[t]; r.removeFromArray(i, e), e.elesQueue = [], e.invalid = !0, e.replacement && (e.replacement.invalid = !0); for (var o = 0; o < n.length; o++) { var a = n[o]._private.rscratch.imgLayerCaches; a && (a[t] = null) } } }, c.refineElementTextures = function (e) { var t = this; t.updateElementsInLayers(e, (function (e, n, r) { var i = e.replacement; if (i || ((i = e.replacement = t.makeLayer(e.bb, e.level)).replaces = e, i.eles = e.eles), !i.reqs) for (var o = 0; o < i.eles.length; o++)t.queueLayer(i, i.eles[o]) })) }, c.setupEleCacheInvalidation = function () { var e = this, t = [], n = r.debounce((function () { e.refineElementTextures(t), t = [] }), 50); e.eleTxrCache.onDequeue((function (e) { for (var r = 0; r < e.length; r++)t.push(e[r]); n() })) }, c.queueLayer = function (e, t) { var n = this.layersQueue, r = e.elesQueue, i = r.hasId = r.hasId || {}; if (!e.replacement) { if (t) { if (i[t.id()]) return; r.push(t), i[t.id()] = !0 } e.reqs ? (e.reqs++, n.updateItem(e)) : (e.reqs = 1, n.push(e)) } }, c.dequeue = function (e) { for (var t = this.layersQueue, n = [], r = 0; r < 1 && 0 !== t.size();) { var i = t.peek(); if (i.replacement) t.pop(); else if (i.replaces && i !== i.replaces.replacement) t.pop(); else if (i.invalid) t.pop(); else { var o = i.elesQueue.shift(); o && (this.drawEleInLayer(i, o, i.level, e), r++), 0 === n.length && n.push(!0), 0 === i.elesQueue.length && (t.pop(), i.reqs = 0, i.replaces && this.applyLayerReplacement(i), this.requestRedraw()) } } return n }, c.applyLayerReplacement = function (e) { var t = this.layersByLevel[e.level], n = e.replaces, r = t.indexOf(n); if (!(r < 0 || n.invalid)) { t[r] = e; for (var i = 0; i < e.eles.length; i++) { var o = e.eles[i]._private, a = o.imgLayerCaches = o.imgLayerCaches || {}; a && (a[e.level] = e) } this.requestRedraw() } }, c.requestRedraw = r.debounce((function () { var e = this.renderer; e.redrawHint("eles", !0), e.redrawHint("drag", !0), e.redraw() }), 100), c.setupDequeueing = s.setupDequeueing({ deqRedrawThreshold: 50, deqCost: .15, deqAvgCost: .1, deqNoDrawCost: .9, deqFastCost: .9, deq: function (e, t) { return e.dequeue(t) }, onDeqd: r.noop, shouldRedraw: r.trueify, priority: function (e) { return e.renderer.beforeRenderPriorities.lyrTxrDeq } }), e.exports = l }, function (e, t, n) { "use strict"; var r, i = { arrowShapeImpl: function (e) { return (r || (r = { polygon: function (e, t) { for (var n = 0; n < t.length; n++) { var r = t[n]; e.lineTo(r.x, r.y) } }, "triangle-backcurve": function (e, t, n) { for (var r, i = 0; i < t.length; i++) { var o = t[i]; 0 === i && (r = o), e.lineTo(o.x, o.y) } e.quadraticCurveTo(n.x, n.y, r.x, r.y) }, "triangle-tee": function (e, t, n) { e.beginPath && e.beginPath(); for (var r = t, i = 0; i < r.length; i++) { var o = r[i]; e.lineTo(o.x, o.y) } e.closePath && e.closePath(), e.beginPath && e.beginPath(); var a = n, s = n[0]; for (e.moveTo(s.x, s.y), i = 0; i < a.length; i++)o = a[i], e.lineTo(o.x, o.y); e.closePath && e.closePath() }, "triangle-cross": function (e, t, n) { e.beginPath && e.beginPath(); for (var r = t, i = 0; i < r.length; i++) { var o = r[i]; e.lineTo(o.x, o.y) } e.closePath && e.closePath(), e.beginPath && e.beginPath(); var a = n, s = n[0]; for (e.moveTo(s.x, s.y), i = 0; i < a.length; i++)o = a[i], e.lineTo(o.x, o.y); e.closePath && e.closePath() }, circle: function (e, t, n, r) { e.arc(t, n, r, 0, 2 * Math.PI, !1) } }))[e] } }; e.exports = i }, function (e, t, n) { "use strict"; var r = n(2), i = { drawElement: function (e, t, n, r) { t.isNode() ? this.drawNode(e, t, n, r) : this.drawEdge(e, t, n, r) }, drawCachedElement: function (e, t, n, i) { var o = t.boundingBox(); if (0 !== o.w && 0 !== o.h && (!i || r.boundingBoxesIntersect(o, i))) { var a = this.data.eleTxrCache.getElement(t, o, n); null != a ? e.drawImage(a.texture.canvas, a.x, 0, a.width, a.height, o.x1, o.y1, o.w, o.h) : this.drawElement(e, t) } }, drawElements: function (e, t) { for (var n = 0; n < t.length; n++) { var r = t[n]; this.drawElement(e, r) } }, drawCachedElements: function (e, t, n, r) { for (var i = 0; i < t.length; i++) { var o = t[i]; this.drawCachedElement(e, o, n, r) } }, drawCachedNodes: function (e, t, n, r) { for (var i = 0; i < t.length; i++) { var o = t[i]; o.isNode() && this.drawCachedElement(e, o, n, r) } }, drawLayeredElements: function (e, t, n, r) { var i = this.data.lyrTxrCache.getLayers(t, n); if (i) for (var o = 0; o < i.length; o++) { var a = i[o], s = a.bb; 0 !== s.w && 0 !== s.h && e.drawImage(a.canvas, s.x1, s.y1, s.w, s.h) } else this.drawCachedElements(e, t, n, r) }, drawDebugPoints: function (e, t) { for (var n = function (t, n, r) { e.fillStyle = r, e.fillRect(t - 1, n - 1, 3, 3) }, r = 0; r < t.length; r++) { var i = t[r], o = i._private.rscratch; if (i.isNode()) { var a = i.position(); n(a.x, a.y, "magenta") } else { for (var s = o.allpts, l = 0; l + 1 < s.length; l += 2)n(s[l], s[l + 1], "cyan"); n(o.midX, o.midY, "yellow") } } } }; e.exports = i }, function (e, t, n) { "use strict"; var r = { drawEdge: function (e, t, n, r) { var i = this, o = t._private.rscratch, a = i.usePaths(); if (t.visible() && !o.badLine && null != o.allpts && !isNaN(o.allpts[0])) { var s = void 0; n && (s = n, e.translate(-s.x1, -s.y1)); var l = 2 * t.pstyle("overlay-padding").pfValue, c = t.pstyle("overlay-opacity").value, u = t.pstyle("overlay-color").value, d = t.pstyle("line-color").value, p = t.pstyle("opacity").value, f = t.pstyle("line-style").value, h = t.pstyle("width").pfValue, g = function () { var n = arguments.length > 0 && void 0 !== arguments[0] ? arguments[0] : p; e.lineWidth = h, e.lineCap = "butt", i.strokeStyle(e, d[0], d[1], d[2], n), i.drawEdgePath(t, e, o.allpts, f) }, m = function () { var n = arguments.length > 0 && void 0 !== arguments[0] ? arguments[0] : p; i.drawArrowheads(e, t, n) }; if (e.lineJoin = "round", "yes" === t.pstyle("ghost").value) { var v = t.pstyle("ghost-offset-x").pfValue, b = t.pstyle("ghost-offset-y").pfValue, y = t.pstyle("ghost-opacity").value, x = p * y; e.translate(v, b), g(x), m(x), e.translate(-v, -b) } g(), m(), function () { var n = arguments.length > 0 && void 0 !== arguments[0] ? arguments[0] : c; e.lineWidth = l, "self" !== o.edgeType || a ? e.lineCap = "round" : e.lineCap = "butt", i.strokeStyle(e, u[0], u[1], u[2], n), i.drawEdgePath(t, e, o.allpts, "solid") }(), i.drawElementText(e, t, r), n && e.translate(s.x1, s.y1) } }, drawEdgePath: function (e, t, n, r) { var i = e._private.rscratch, o = t, a = void 0, s = !1, l = this.usePaths(); if (l) { var c = n.join("$"); i.pathCacheKey && i.pathCacheKey === c ? (a = t = i.pathCache, s = !0) : (a = t = new Path2D, i.pathCacheKey = c, i.pathCache = a) } if (o.setLineDash) switch (r) { case "dotted": o.setLineDash([1, 1]); break; case "dashed": o.setLineDash([6, 3]); break; case "solid": o.setLineDash([]) }if (!s && !i.badLine) switch (t.beginPath && t.beginPath(), t.moveTo(n[0], n[1]), i.edgeType) { case "bezier": case "self": case "compound": case "multibezier": if (e.hasClass("horizontal")) { var u = n[4], d = n[5], p = (n[0] + n[4]) / 2; t.lineTo(n[0] + 10, n[1]), t.bezierCurveTo(p, n[1], p, n[5], n[4] - 10, n[5]), t.lineTo(u, d) } else if (e.hasClass("vertical")) { var f = n[4], h = n[5], g = (n[1] + n[5]) / 2; t.bezierCurveTo(n[0], g, n[4], g, n[4], n[5] - 10), t.lineTo(f, h) } else for (var m = 2; m + 3 < n.length; m += 4)t.quadraticCurveTo(n[m], n[m + 1], n[m + 2], n[m + 3]); break; case "straight": case "segments": case "haystack": for (var v = 2; v + 1 < n.length; v += 2)t.lineTo(n[v], n[v + 1]) }t = o, l ? t.stroke(a) : t.stroke(), t.setLineDash && t.setLineDash([]) }, drawArrowheads: function (e, t, n) { var r = t._private.rscratch, i = "haystack" === r.edgeType; i || this.drawArrowhead(e, t, "source", r.arrowStartX, r.arrowStartY, r.srcArrowAngle, n), this.drawArrowhead(e, t, "mid-target", r.midX, r.midY, r.midtgtArrowAngle, n), this.drawArrowhead(e, t, "mid-source", r.midX, r.midY, r.midsrcArrowAngle, n), i || (t.hasClass("horizontal") ? this.drawArrowhead(e, t, "target", r.arrowEndX, r.arrowEndY, -Math.PI / 2, n) : t.hasClass("vertical") ? this.drawArrowhead(e, t, "target", r.arrowEndX, r.arrowEndY, 0, n) : this.drawArrowhead(e, t, "target", r.arrowEndX, r.arrowEndY, r.tgtArrowAngle, n)) }, drawArrowhead: function (e, t, n, r, i, o, a) { if (!(isNaN(r) || null == r || isNaN(i) || null == i || isNaN(o) || null == o)) { var s = t.pstyle(n + "-arrow-shape").value; if ("none" !== s) { var l = "hollow" === t.pstyle(n + "-arrow-fill").value ? "both" : "filled", c = t.pstyle(n + "-arrow-fill").value, u = t.pstyle("width").pfValue, d = t.pstyle("opacity").value; void 0 === a && (a = d); var p = e.globalCompositeOperation; 1 === a && "hollow" !== c || (e.globalCompositeOperation = "destination-out", this.fillStyle(e, 255, 255, 255, 1), this.strokeStyle(e, 255, 255, 255, 1), this.drawArrowShape(t, n, e, l, u, s, r, i, o), e.globalCompositeOperation = p); var f = t.pstyle(n + "-arrow-color").value; this.fillStyle(e, f[0], f[1], f[2], a), this.strokeStyle(e, f[0], f[1], f[2], a), this.drawArrowShape(t, n, e, c, u, s, r, i, o) } } }, drawArrowShape: function (e, t, n, r, i, o, a, s, l) { var c = this.usePaths(), u = e._private.rscratch, d = !1, p = void 0, f = n, h = { x: a, y: s }, g = e.pstyle("arrow-scale").value, m = this.getArrowWidth(i, g), v = this.arrowShapes[o]; if (c) { var b = m + "$" + o + "$" + l + "$" + a + "$" + s; u.arrowPathCacheKey = u.arrowPathCacheKey || {}, u.arrowPathCache = u.arrowPathCache || {}, u.arrowPathCacheKey[t] === b ? (p = n = u.arrowPathCache[t], d = !0) : (p = n = new Path2D, u.arrowPathCacheKey[t] = b, u.arrowPathCache[t] = p) } n.beginPath && n.beginPath(), d || v.draw(n, m, l, h, i), !v.leavePathOpen && n.closePath && n.closePath(), n = f, "filled" !== r && "both" !== r || (c ? n.fill(p) : n.fill()), "hollow" !== r && "both" !== r || (n.lineWidth = v.matchEdgeWidth ? i : 1, n.lineJoin = "miter", c ? n.stroke(p) : n.stroke()) } }; e.exports = r }, function (e, t, n) { "use strict"; var r = { safeDrawImage: function (e, t, n, r, i, o, a, s, l, c) { i <= 0 || o <= 0 || l <= 0 || c <= 0 || e.drawImage(t, n, r, i, o, a, s, l, c) }, drawInscribedImage: function (e, t, n, r, i) { var o = this, a = n.position(), s = a.x, l = a.y, c = n.cy().style(), u = c.getIndexedStyle.bind(c), d = u(n, "background-fit", "value", r), p = u(n, "background-repeat", "value", r), f = n.width(), h = n.height(), g = 2 * n.padding(), m = f + ("inner" === u(n, "background-width-relative-to", "value", r) ? 0 : g), v = h + ("inner" === u(n, "background-height-relative-to", "value", r) ? 0 : g), b = n._private.rscratch, y = "node" === n.pstyle("background-clip").value, x = u(n, "background-image-opacity", "value", r) * i, w = t.width || t.cachedW, k = t.height || t.cachedH; null != w && null != k || (document.body.appendChild(t), w = t.cachedW = t.width || t.offsetWidth, k = t.cachedH = t.height || t.offsetHeight, document.body.removeChild(t)); var A = w, E = k; if ("auto" !== u(n, "background-width", "value", r) && (A = "%" === u(n, "background-width", "units", r) ? u(n, "background-width", "pfValue", r) * m : u(n, "background-width", "pfValue", r)), "auto" !== u(n, "background-height", "value", r) && (E = "%" === u(n, "background-height", "units", r) ? u(n, "background-height", "pfValue", r) * v : u(n, "background-height", "pfValue", r)), 0 !== A && 0 !== E) { if ("contain" === d) A *= S = Math.min(m / A, v / E), E *= S; else if ("cover" === d) { var S; A *= S = Math.max(m / A, v / E), E *= S } var $ = s - m / 2; "%" === u(n, "background-position-x", "units", r) ? $ += (m - A) * u(n, "background-position-x", "pfValue", r) : $ += u(n, "background-position-x", "pfValue", r); var C = l - v / 2; "%" === u(n, "background-position-y", "units", r) ? C += (v - E) * u(n, "background-position-y", "pfValue", r) : C += u(n, "background-position-y", "pfValue", r), b.pathCache && ($ -= s, C -= l, s = 0, l = 0); var _ = e.globalAlpha; if (e.globalAlpha = x, "no-repeat" === p) y && (e.save(), b.pathCache ? e.clip(b.pathCache) : (o.nodeShapes[o.getNodeShape(n)].draw(e, s, l, m, v), e.clip())), o.safeDrawImage(e, t, 0, 0, w, k, $, C, A, E), y && e.restore(); else { var O = e.createPattern(t, p); e.fillStyle = O, o.nodeShapes[o.getNodeShape(n)].draw(e, s, l, m, v), e.translate($, C), e.fill(), e.translate(-$, -C) } e.globalAlpha = _ } } }; e.exports = r }, function (e, t, n) { "use strict"; var r = n(1), i = n(2), o = { eleTextBiggerThanMin: function (e, t) { if (!t) { var n = e.cy().zoom(), r = this.getPixelRatio(), o = Math.ceil(i.log2(n * r)); t = Math.pow(2, o) } return !(e.pstyle("font-size").pfValue * t < e.pstyle("min-zoomed-font-size").pfValue) }, drawElementText: function (e, t, n) { var r = this; if (void 0 === n) { if (!r.eleTextBiggerThanMin(t)) return } else if (!n) return; if (t.isNode()) { if (!(o = t.pstyle("label")) || !o.value) return; var i = t.pstyle("text-halign").strValue; switch (t.pstyle("text-valign").strValue, i) { case "left": e.textAlign = "right"; break; case "right": e.textAlign = "left"; break; default: e.textAlign = "center" }e.textBaseline = "bottom" } else { var o = t.pstyle("label"), a = t.pstyle("source-label"), s = t.pstyle("target-label"); if (!(o && o.value || a && a.value || s && s.value)) return; e.textAlign = "center", e.textBaseline = "bottom" } r.drawText(e, t), t.isEdge() && (r.drawText(e, t, "source"), r.drawText(e, t, "target")) } }; o.drawNodeText = o.drawEdgeText = o.drawElementText, o.getFontCache = function (e) { var t; this.fontCaches = this.fontCaches || []; for (var n = 0; n < this.fontCaches.length; n++)if ((t = this.fontCaches[n]).context === e) return t; return t = { context: e }, this.fontCaches.push(t), t }, o.setupTextStyle = function (e, t) { var n = t.effectiveOpacity(), r = t.pstyle("font-style").strValue, i = t.pstyle("font-size").pfValue + "px", o = t.pstyle("font-family").strValue, a = t.pstyle("font-weight").strValue, s = t.pstyle("text-opacity").value * t.pstyle("opacity").value * n, l = t.pstyle("text-outline-opacity").value * s, c = t.pstyle("color").value, u = t.pstyle("text-outline-color").value, d = t._private.fontKey, p = this.getFontCache(e); p.key !== d && (e.font = r + " " + a + " " + i + " " + o, p.key = d), e.lineJoin = "round", this.fillStyle(e, c[0], c[1], c[2], s), this.strokeStyle(e, u[0], u[1], u[2], l) }, o.drawText = function (e, t, n) { var i = t._private.rscratch, o = t.effectiveOpacity(); if (0 !== o && 0 !== t.pstyle("text-opacity").value) { var a, s, l, c, u, d, p = r.getPrefixedProperty(i, "labelX", n), f = r.getPrefixedProperty(i, "labelY", n), h = this.getLabelText(t, n); if (null != h && "" !== h && !isNaN(p) && !isNaN(f)) { this.setupTextStyle(e, t); var g = n ? n + "-" : "", m = r.getPrefixedProperty(i, "labelWidth", n), v = r.getPrefixedProperty(i, "labelHeight", n), b = r.getPrefixedProperty(i, "labelAngle", n), y = t.pstyle(g + "text-margin-x").pfValue, x = t.pstyle(g + "text-margin-y").pfValue, w = t.isEdge(), k = (t.isNode(), t.pstyle("text-halign").value), A = t.pstyle("text-valign").value; w && (k = "center", A = "center"), p += y, f += x; var E, S = t.pstyle("text-rotation"); if (0 !== (E = "autorotate" === S.strValue ? w ? b : 0 : "none" === S.strValue ? 0 : S.pfValue)) { var $ = p, C = f; e.translate($, C), e.rotate(E), p = 0, f = 0 } switch (A) { case "top": break; case "center": f += v / 2; break; case "bottom": f += v }var _ = t.pstyle("text-background-opacity").value, O = t.pstyle("text-border-opacity").value, T = t.pstyle("text-border-width").pfValue, j = t.pstyle("text-background-padding").pfValue; if (_ > 0 || T > 0 && O > 0) { var P = p - j; switch (k) { case "left": P -= m; break; case "center": P -= m / 2 }var D = f - v - j, R = m + 2 * j, I = v + 2 * j; if (_ > 0) { var N = e.fillStyle, M = t.pstyle("text-background-color").value; e.fillStyle = "rgba(" + M[0] + "," + M[1] + "," + M[2] + "," + _ * o + ")", "roundrectangle" == t.pstyle("text-background-shape").strValue ? (s = P, l = D, c = R, u = I, d = (d = 2) || 5, (a = e).beginPath(), a.moveTo(s + d, l), a.lineTo(s + c - d, l), a.quadraticCurveTo(s + c, l, s + c, l + d), a.lineTo(s + c, l + u - d), a.quadraticCurveTo(s + c, l + u, s + c - d, l + u), a.lineTo(s + d, l + u), a.quadraticCurveTo(s, l + u, s, l + u - d), a.lineTo(s, l + d), a.quadraticCurveTo(s, l, s + d, l), a.closePath(), a.fill()) : e.fillRect(P, D, R, I), e.fillStyle = N } if (T > 0 && O > 0) { var z = e.strokeStyle, L = e.lineWidth, B = t.pstyle("text-border-color").value, F = t.pstyle("text-border-style").value; if (e.strokeStyle = "rgba(" + B[0] + "," + B[1] + "," + B[2] + "," + O * o + ")", e.lineWidth = T, e.setLineDash) switch (F) { case "dotted": e.setLineDash([1, 1]); break; case "dashed": e.setLineDash([4, 2]); break; case "double": e.lineWidth = T / 4, e.setLineDash([]); break; case "solid": e.setLineDash([]) }if (e.strokeRect(P, D, R, I), "double" === F) { var q = T / 2; e.strokeRect(P + q, D + q, R - 2 * q, I - 2 * q) } e.setLineDash && e.setLineDash([]), e.lineWidth = L, e.strokeStyle = z } } var V = 2 * t.pstyle("text-outline-width").pfValue; if (V > 0 && (e.lineWidth = V), "wrap" === t.pstyle("text-wrap").value) { var U = r.getPrefixedProperty(i, "labelWrapCachedLines", n), H = v / U.length; switch (A) { case "top": f -= (U.length - 1) * H; break; case "center": case "bottom": f -= (U.length - 1) * H }for (var G = 0; G < U.length; G++)V > 0 && e.strokeText(U[G], p, f), e.fillText(U[G], p, f), f += H } else V > 0 && e.strokeText(h, p, f), e.fillText(h, p, f); 0 !== E && (e.rotate(-E), e.translate(-$, -C)) } } }, e.exports = o }, function (e, t, n) { "use strict"; var r = n(0), i = { drawNode: function (e, t, n, i) { var o, a, s = this, l = t._private, c = l.rscratch, u = t.position(); if (r.number(u.x) && r.number(u.y) && t.visible()) { var d = t.effectiveOpacity(), p = s.usePaths(), f = void 0, h = !1, g = t.padding(); o = t.width() + 2 * g, a = t.height() + 2 * g; var m = void 0; n && (m = n, e.translate(-m.x1, -m.y1)); for (var v = t.pstyle("background-image").value, b = new Array(v.length), y = new Array(v.length), x = 0, w = 0; w < v.length; w++) { var k = v[w]; if (b[w] = null != k && "none" !== k) { var A = t.cy().style().getIndexedStyle(t, "background-image-crossorigin", "value", w); x++, y[w] = s.getCachedImage(k, A, (function () { t.emitAndNotify("background") })) } } var E = t.pstyle("background-blacken").value, S = t.pstyle("border-width").pfValue, $ = t.pstyle("background-color").value, C = t.pstyle("background-opacity").value * d, _ = t.pstyle("border-color").value, O = t.pstyle("border-style").value, T = t.pstyle("border-opacity").value * d; e.lineJoin = "miter"; var j = function () { var t = arguments.length > 0 && void 0 !== arguments[0] ? arguments[0] : C; s.fillStyle(e, $[0], $[1], $[2], t) }, P = function () { var t = arguments.length > 0 && void 0 !== arguments[0] ? arguments[0] : T; s.strokeStyle(e, _[0], _[1], _[2], t) }, D = t.pstyle("shape").strValue, R = t.pstyle("shape-polygon-points").pfValue; if (p) { var I = D + "$" + o + "$" + a + ("polygon" === D ? "$" + R.join("$") : ""); e.translate(u.x, u.y), c.pathCacheKey === I ? (f = c.pathCache, h = !0) : (f = new Path2D, c.pathCacheKey = I, c.pathCache = f) } var N, M, z, L = function () { if (!h) { var n = u; p && (n = { x: 0, y: 0 }), s.nodeShapes[s.getNodeShape(t)].draw(f || e, n.x, n.y, o, a) } p ? e.fill(f) : e.fill() }, B = function () { for (var n = arguments.length > 0 && void 0 !== arguments[0] ? arguments[0] : d, r = l.backgrounding, i = 0, o = 0; o < y.length; o++)b[o] && y[o].complete && !y[o].error && (i++, s.drawInscribedImage(e, y[o], t, o, n)); l.backgrounding = !(i === x), r !== l.backgrounding && t.updateStyle(!1) }, F = function () { var n = arguments.length > 0 && void 0 !== arguments[0] && arguments[0], r = arguments.length > 1 && void 0 !== arguments[1] ? arguments[1] : d; s.hasPie(t) && (s.drawPie(e, t, r), n && (p || s.nodeShapes[s.getNodeShape(t)].draw(e, u.x, u.y, o, a))) }, q = function () { var t = arguments.length > 0 && void 0 !== arguments[0] ? arguments[0] : d, n = (E > 0 ? E : -E) * t, r = E > 0 ? 0 : 255; 0 !== E && (s.fillStyle(e, r, r, r, n), p ? e.fill(f) : e.fill()) }, V = function () { if (S > 0) { if (e.lineWidth = S, e.lineCap = "butt", e.setLineDash) switch (O) { case "dotted": e.setLineDash([1, 1]); break; case "dashed": e.setLineDash([4, 2]); break; case "solid": case "double": e.setLineDash([]) }if (p ? e.stroke(f) : e.stroke(), "double" === O) { e.lineWidth = S / 3; var t = e.globalCompositeOperation; e.globalCompositeOperation = "destination-out", p ? e.stroke(f) : e.stroke(), e.globalCompositeOperation = t } e.setLineDash && e.setLineDash([]) } }; if ("yes" === t.pstyle("ghost").value) { var U = t.pstyle("ghost-offset-x").pfValue, H = t.pstyle("ghost-offset-y").pfValue, G = t.pstyle("ghost-opacity").value, W = G * d; e.translate(U, H), j(G * C), L(), B(W), F(0 !== E || 0 !== S), q(W), P(G * T), V(), e.translate(-U, -H) } j(), L(), B(), F(0 !== E || 0 !== S), q(), P(), V(), p && e.translate(-u.x, -u.y), s.drawElementText(e, t, i), N = t.pstyle("overlay-padding").pfValue, M = t.pstyle("overlay-opacity").value, z = t.pstyle("overlay-color").value, M > 0 && (s.fillStyle(e, z[0], z[1], z[2], M), s.nodeShapes.roundrectangle.draw(e, u.x, u.y, o + 2 * N, a + 2 * N), e.fill()), n && e.translate(m.x1, m.y1) } }, hasPie: function (e) { return (e = e[0])._private.hasPie }, drawPie: function (e, t, n, r) { t = t[0], r = r || t.position(); var i = t.cy().style(), o = t.pstyle("pie-size"), a = r.x, s = r.y, l = t.width(), c = t.height(), u = Math.min(l, c) / 2, d = 0; this.usePaths() && (a = 0, s = 0), "%" === o.units ? u *= o.pfValue : void 0 !== o.pfValue && (u = o.pfValue / 2); for (var p = 1; p <= i.pieBackgroundN; p++) { var f = t.pstyle("pie-" + p + "-background-size").value, h = t.pstyle("pie-" + p + "-background-color").value, g = t.pstyle("pie-" + p + "-background-opacity").value * n, m = f / 100; m + d > 1 && (m = 1 - d); var v = 1.5 * Math.PI + 2 * Math.PI * d, b = v + 2 * Math.PI * m; 0 === f || d >= 1 || d + m > 1 || (e.beginPath(), e.moveTo(a, s), e.arc(a, s, u, v, b), e.closePath(), this.fillStyle(e, h[0], h[1], h[2], g), e.fill(), d += m) } } }; e.exports = i }, function (e, t, n) { "use strict"; var r = {}, i = n(1); r.getPixelRatio = function () { var e = this.data.contexts[0]; if (null != this.forcedPixelRatio) return this.forcedPixelRatio; var t = e.backingStorePixelRatio || e.webkitBackingStorePixelRatio || e.mozBackingStorePixelRatio || e.msBackingStorePixelRatio || e.oBackingStorePixelRatio || e.backingStorePixelRatio || 1; return (window.devicePixelRatio || 1) / t }, r.paintCache = function (e) { for (var t, n = this.paintCaches = this.paintCaches || [], r = !0, i = 0; i < n.length; i++)if ((t = n[i]).context === e) { r = !1; break } return r && (t = { context: e }, n.push(t)), t }, r.fillStyle = function (e, t, n, r, i) { e.fillStyle = "rgba(" + t + "," + n + "," + r + "," + i + ")" }, r.strokeStyle = function (e, t, n, r, i) { e.strokeStyle = "rgba(" + t + "," + n + "," + r + "," + i + ")" }, r.matchCanvasSize = function (e) { var t = this, n = t.data, r = t.findContainerClientCoords(), i = r[2], o = r[3], a = t.getPixelRatio(), s = t.motionBlurPxRatio; e !== t.data.bufferCanvases[t.MOTIONBLUR_BUFFER_NODE] && e !== t.data.bufferCanvases[t.MOTIONBLUR_BUFFER_DRAG] || (a = s); var l, c = i * a, u = o * a; if (c !== t.canvasWidth || u !== t.canvasHeight) { t.fontCaches = null; var d = n.canvasContainer; d.style.width = i + "px", d.style.height = o + "px"; for (var p = 0; p < t.CANVAS_LAYERS; p++)(l = n.canvases[p]).width = c, l.height = u, l.style.width = i + "px", l.style.height = o + "px"; for (p = 0; p < t.BUFFER_COUNT; p++)(l = n.bufferCanvases[p]).width = c, l.height = u, l.style.width = i + "px", l.style.height = o + "px"; t.textureMult = 1, a <= 1 && (l = n.bufferCanvases[t.TEXTURE_BUFFER], t.textureMult = 2, l.width = c * t.textureMult, l.height = u * t.textureMult), t.canvasWidth = c, t.canvasHeight = u } }, r.renderTo = function (e, t, n, r) { this.render({ forcedContext: e, forcedZoom: t, forcedPan: n, drawAllLayers: !0, forcedPxRatio: r }) }, r.render = function (e) { var t = (e = e || i.staticEmptyObject()).forcedContext, n = e.drawAllLayers, r = e.drawOnlyNodeLayer, o = e.forcedZoom, a = e.forcedPan, s = this, l = void 0 === e.forcedPxRatio ? this.getPixelRatio() : e.forcedPxRatio, c = s.cy, u = s.data, d = u.canvasNeedsRedraw, p = s.textureOnViewport && !t && (s.pinching || s.hoverData.dragging || s.swipePanning || s.data.wheelZooming), f = void 0 !== e.motionBlur ? e.motionBlur : s.motionBlur, h = s.motionBlurPxRatio, g = c.hasCompoundNodes(), m = s.hoverData.draggingEles, v = !(!s.hoverData.selecting && !s.touchData.selecting), b = f = f && !t && s.motionBlurEnabled && !v; t || (s.prevPxRatio !== l && (s.invalidateContainerClientCoordsCache(), s.matchCanvasSize(s.container), s.redrawHint("eles", !0), s.redrawHint("drag", !0)), s.prevPxRatio = l), !t && s.motionBlurTimeout && clearTimeout(s.motionBlurTimeout), f && (null == s.mbFrames && (s.mbFrames = 0), s.mbFrames++, s.mbFrames < 3 && (b = !1), s.mbFrames > s.minMbLowQualFrames && (s.motionBlurPxRatio = s.mbPxRBlurry)), s.clearingMotionBlur && (s.motionBlurPxRatio = 1), s.textureDrawLastFrame && !p && (d[s.NODE] = !0, d[s.SELECT_BOX] = !0); var y = c.style()._private.coreStyle, x = c.zoom(), w = void 0 !== o ? o : x, k = c.pan(), A = { x: k.x, y: k.y }, E = { zoom: x, pan: { x: k.x, y: k.y } }, S = s.prevViewport; void 0 === S || E.zoom !== S.zoom || E.pan.x !== S.pan.x || E.pan.y !== S.pan.y || m && !g || (s.motionBlurPxRatio = 1), a && (A = a), w *= l, A.x *= l, A.y *= l; var $ = s.getCachedZSortedEles(); function C(e, t, n, r, i) { var o = e.globalCompositeOperation; e.globalCompositeOperation = "destination-out", s.fillStyle(e, 255, 255, 255, s.motionBlurTransparency), e.fillRect(t, n, r, i), e.globalCompositeOperation = o } function _(e, r) { var i, l, c, d; s.clearingMotionBlur || e !== u.bufferContexts[s.MOTIONBLUR_BUFFER_NODE] && e !== u.bufferContexts[s.MOTIONBLUR_BUFFER_DRAG] ? (i = A, l = w, c = s.canvasWidth, d = s.canvasHeight) : (i = { x: k.x * h, y: k.y * h }, l = x * h, c = s.canvasWidth * h, d = s.canvasHeight * h), e.setTransform(1, 0, 0, 1, 0, 0), "motionBlur" === r ? C(e, 0, 0, c, d) : t || void 0 !== r && !r || e.clearRect(0, 0, c, d), n || (e.translate(i.x, i.y), e.scale(l, l)), a && e.translate(a.x, a.y), o && e.scale(o, o) } if (p || (s.textureDrawLastFrame = !1), p) { if (s.textureDrawLastFrame = !0, !s.textureCache) { s.textureCache = {}, s.textureCache.bb = c.mutableElements().boundingBox(), s.textureCache.texture = s.data.bufferCanvases[s.TEXTURE_BUFFER]; var O = s.data.bufferContexts[s.TEXTURE_BUFFER]; O.setTransform(1, 0, 0, 1, 0, 0), O.clearRect(0, 0, s.canvasWidth * s.textureMult, s.canvasHeight * s.textureMult), s.render({ forcedContext: O, drawOnlyNodeLayer: !0, forcedPxRatio: l * s.textureMult }), (E = s.textureCache.viewport = { zoom: c.zoom(), pan: c.pan(), width: s.canvasWidth, height: s.canvasHeight }).mpan = { x: (0 - E.pan.x) / E.zoom, y: (0 - E.pan.y) / E.zoom } } d[s.DRAG] = !1, d[s.NODE] = !1; var T = u.contexts[s.NODE], j = s.textureCache.texture; E = s.textureCache.viewport, s.textureCache.bb, T.setTransform(1, 0, 0, 1, 0, 0), f ? C(T, 0, 0, E.width, E.height) : T.clearRect(0, 0, E.width, E.height); var P = y["outside-texture-bg-color"].value, D = y["outside-texture-bg-opacity"].value; s.fillStyle(T, P[0], P[1], P[2], D), T.fillRect(0, 0, E.width, E.height), x = c.zoom(), _(T, !1), T.clearRect(E.mpan.x, E.mpan.y, E.width / E.zoom / l, E.height / E.zoom / l), T.drawImage(j, E.mpan.x, E.mpan.y, E.width / E.zoom / l, E.height / E.zoom / l) } else s.textureOnViewport && !t && (s.textureCache = null); var R = c.extent(), I = s.pinching || s.hoverData.dragging || s.swipePanning || s.data.wheelZooming || s.hoverData.draggingEles, N = s.hideEdgesOnViewport && I, M = []; if (M[s.NODE] = !d[s.NODE] && f && !s.clearedForMotionBlur[s.NODE] || s.clearingMotionBlur, M[s.NODE] && (s.clearedForMotionBlur[s.NODE] = !0), M[s.DRAG] = !d[s.DRAG] && f && !s.clearedForMotionBlur[s.DRAG] || s.clearingMotionBlur, M[s.DRAG] && (s.clearedForMotionBlur[s.DRAG] = !0), d[s.NODE] || n || r || M[s.NODE]) { var z = f && !M[s.NODE] && 1 !== h; _(T = t || (z ? s.data.bufferContexts[s.MOTIONBLUR_BUFFER_NODE] : u.contexts[s.NODE]), f && !z ? "motionBlur" : void 0), N ? s.drawCachedNodes(T, $.nondrag, l, R) : s.drawLayeredElements(T, $.nondrag, l, R), s.debug && s.drawDebugPoints(T, $.nondrag), n || f || (d[s.NODE] = !1) } if (!r && (d[s.DRAG] || n || M[s.DRAG]) && (z = f && !M[s.DRAG] && 1 !== h, _(T = t || (z ? s.data.bufferContexts[s.MOTIONBLUR_BUFFER_DRAG] : u.contexts[s.DRAG]), f && !z ? "motionBlur" : void 0), N ? s.drawCachedNodes(T, $.drag, l, R) : s.drawCachedElements(T, $.drag, l, R), s.debug && s.drawDebugPoints(T, $.drag), n || f || (d[s.DRAG] = !1)), s.showFps || !r && d[s.SELECT_BOX] && !n) { if (_(T = t || u.contexts[s.SELECT_BOX]), 1 == s.selection[4] && (s.hoverData.selecting || s.touchData.selecting)) { x = s.cy.zoom(); var L = y["selection-box-border-width"].value / x; T.lineWidth = L, T.fillStyle = "rgba(" + y["selection-box-color"].value[0] + "," + y["selection-box-color"].value[1] + "," + y["selection-box-color"].value[2] + "," + y["selection-box-opacity"].value + ")", T.fillRect(s.selection[0], s.selection[1], s.selection[2] - s.selection[0], s.selection[3] - s.selection[1]), L > 0 && (T.strokeStyle = "rgba(" + y["selection-box-border-color"].value[0] + "," + y["selection-box-border-color"].value[1] + "," + y["selection-box-border-color"].value[2] + "," + y["selection-box-opacity"].value + ")", T.strokeRect(s.selection[0], s.selection[1], s.selection[2] - s.selection[0], s.selection[3] - s.selection[1])) } if (u.bgActivePosistion && !s.hoverData.selecting) { x = s.cy.zoom(); var B = u.bgActivePosistion; T.fillStyle = "rgba(" + y["active-bg-color"].value[0] + "," + y["active-bg-color"].value[1] + "," + y["active-bg-color"].value[2] + "," + y["active-bg-opacity"].value + ")", T.beginPath(), T.arc(B.x, B.y, y["active-bg-size"].pfValue / x, 0, 2 * Math.PI), T.fill() } var F = s.lastRedrawTime; if (s.showFps && F) { F = Math.round(F); var q = Math.round(1e3 / F); T.setTransform(1, 0, 0, 1, 0, 0), T.fillStyle = "rgba(255, 0, 0, 0.75)", T.strokeStyle = "rgba(255, 0, 0, 0.75)", T.lineWidth = 1, T.fillText("1 frame = " + F + " ms = " + q + " fps", 0, 20), T.strokeRect(0, 30, 250, 20), T.fillRect(0, 30, 250 * Math.min(q / 60, 1), 20) } n || (d[s.SELECT_BOX] = !1) } if (f && 1 !== h) { var V = u.contexts[s.NODE], U = s.data.bufferCanvases[s.MOTIONBLUR_BUFFER_NODE], H = u.contexts[s.DRAG], G = s.data.bufferCanvases[s.MOTIONBLUR_BUFFER_DRAG], W = function (e, t, n) { e.setTransform(1, 0, 0, 1, 0, 0), n || !b ? e.clearRect(0, 0, s.canvasWidth, s.canvasHeight) : C(e, 0, 0, s.canvasWidth, s.canvasHeight); var r = h; e.drawImage(t, 0, 0, s.canvasWidth * r, s.canvasHeight * r, 0, 0, s.canvasWidth, s.canvasHeight) }; (d[s.NODE] || M[s.NODE]) && (W(V, U, M[s.NODE]), d[s.NODE] = !1), (d[s.DRAG] || M[s.DRAG]) && (W(H, G, M[s.DRAG]), d[s.DRAG] = !1) } s.prevViewport = E, s.clearingMotionBlur && (s.clearingMotionBlur = !1, s.motionBlurCleared = !0, s.motionBlur = !0), f && (s.motionBlurTimeout = setTimeout((function () { s.motionBlurTimeout = null, s.clearedForMotionBlur[s.NODE] = !1, s.clearedForMotionBlur[s.DRAG] = !1, s.motionBlur = !1, s.clearingMotionBlur = !p, s.mbFrames = 0, d[s.NODE] = !0, d[s.DRAG] = !0, s.redraw() }), 100)), t || c.emit("render") }, e.exports = r }, function (e, t, n) { "use strict"; for (var r = n(2), i = { drawPolygonPath: function (e, t, n, r, i, o) { var a = r / 2, s = i / 2; e.beginPath && e.beginPath(), e.moveTo(t + a * o[0], n + s * o[1]); for (var l = 1; l < o.length / 2; l++)e.lineTo(t + a * o[2 * l], n + s * o[2 * l + 1]); e.closePath() }, drawRoundRectanglePath: function (e, t, n, i, o) { var a = i / 2, s = o / 2, l = r.getRoundRectangleRadius(i, o); e.beginPath && e.beginPath(), e.moveTo(t, n - s), e.arcTo(t + a, n - s, t + a, n, l), e.arcTo(t + a, n + s, t, n + s, l), e.arcTo(t - a, n + s, t - a, n, l), e.arcTo(t - a, n - s, t, n - s, l), e.lineTo(t, n - s), e.closePath() }, drawBottomRoundRectanglePath: function (e, t, n, i, o) { var a = i / 2, s = o / 2, l = r.getRoundRectangleRadius(i, o); e.beginPath && e.beginPath(), e.moveTo(t, n - s), e.lineTo(t + a, n - s), e.lineTo(t + a, n), e.arcTo(t + a, n + s, t, n + s, l), e.arcTo(t - a, n + s, t - a, n, l), e.lineTo(t - a, n - s), e.lineTo(t, n - s), e.closePath() }, drawCutRectanglePath: function (e, t, n, i, o) { var a = i / 2, s = o / 2, l = r.getCutRectangleCornerLength(); e.beginPath && e.beginPath(), e.moveTo(t - a + l, n - s), e.lineTo(t + a - l, n - s), e.lineTo(t + a, n - s + l), e.lineTo(t + a, n + s - l), e.lineTo(t + a - l, n + s), e.lineTo(t - a + l, n + s), e.lineTo(t - a, n + s - l), e.lineTo(t - a, n - s + l), e.closePath() }, drawBarrelPath: function (e, t, n, i, o) { var a = i / 2, s = o / 2, l = t - a, c = t + a, u = n - s, d = n + s, p = r.getBarrelCurveConstants(i, o), f = p.widthOffset, h = p.heightOffset, g = p.ctrlPtOffsetPct * f; e.beginPath && e.beginPath(), e.moveTo(l, u + h), e.lineTo(l, d - h), e.quadraticCurveTo(l + g, d, l + f, d), e.lineTo(c - f, d), e.quadraticCurveTo(c - g, d, c, d - h), e.lineTo(c, u + h), e.quadraticCurveTo(c - g, u, c - f, u), e.lineTo(l + f, u), e.quadraticCurveTo(l + g, u, l, u + h), e.closePath() } }, o = Math.sin(0), a = Math.cos(0), s = {}, l = {}, c = Math.PI / 40, u = 0 * Math.PI; u < 2 * Math.PI; u += c)s[u] = Math.sin(u), l[u] = Math.cos(u); i.drawEllipsePath = function (e, t, n, r, i) { if (e.beginPath && e.beginPath(), e.ellipse) e.ellipse(t, n, r / 2, i / 2, 0, 0, 2 * Math.PI); else for (var u, d, p = r / 2, f = i / 2, h = 0 * Math.PI; h < 2 * Math.PI; h += c)u = t - p * s[h] * o + p * l[h] * a, d = n + f * l[h] * o + f * s[h] * a, 0 === h ? e.moveTo(u, d) : e.lineTo(u, d); e.closePath() }, e.exports = i }, function (e, t, n) { "use strict"; var r = n(0), i = {}; function o(e) { var t = e.indexOf(","); return e.substr(t + 1) } function a(e, t, n) { var r = t.toDataURL(n, e.quality); switch (e.output) { case "blob": return function (e, t) { for (var n = atob(e), r = new ArrayBuffer(n.length), i = new Uint8Array(r), o = 0; o < n.length; o++)i[o] = n.charCodeAt(o); return new Blob([r], { type: t }) }(o(r), n); case "base64": return o(r); case "base64uri": default: return r } } i.createBuffer = function (e, t) { var n = document.createElement("canvas"); return n.width = e, n.height = t, [n, n.getContext("2d")] }, i.bufferCanvasImage = function (e) { var t = this.cy, n = t.mutableElements().boundingBox(), i = this.findContainerClientCoords(), o = e.full ? Math.ceil(n.w) : i[2], a = e.full ? Math.ceil(n.h) : i[3], s = r.number(e.maxWidth) || r.number(e.maxHeight), l = this.getPixelRatio(), c = 1; if (void 0 !== e.scale) o *= e.scale, a *= e.scale, c = e.scale; else if (s) { var u = 1 / 0, d = 1 / 0; r.number(e.maxWidth) && (u = c * e.maxWidth / o), r.number(e.maxHeight) && (d = c * e.maxHeight / a), o *= c = Math.min(u, d), a *= c } s || (o *= l, a *= l, c *= l); var p = document.createElement("canvas"); p.width = o, p.height = a, p.style.width = o + "px", p.style.height = a + "px"; var f = p.getContext("2d"); if (o > 0 && a > 0) { f.clearRect(0, 0, o, a), f.globalCompositeOperation = "source-over"; var h = this.getCachedZSortedEles(); if (e.full) f.translate(-n.x1 * c, -n.y1 * c), f.scale(c, c), this.drawElements(f, h), f.scale(1 / c, 1 / c), f.translate(n.x1 * c, n.y1 * c); else { var g = t.pan(), m = { x: g.x * c, y: g.y * c }; c *= t.zoom(), f.translate(m.x, m.y), f.scale(c, c), this.drawElements(f, h), f.scale(1 / c, 1 / c), f.translate(-m.x, -m.y) } e.bg && (f.globalCompositeOperation = "destination-over", f.fillStyle = e.bg, f.rect(0, 0, o, a), f.fill()) } return p }, i.png = function (e) { return a(e, this.bufferCanvasImage(e), "image/png") }, i.jpg = function (e) { return a(e, this.bufferCanvasImage(e), "image/jpeg") }, e.exports = i }, function (e, t, n) { "use strict"; var r = { nodeShapeImpl: function (e, t, n, r, i, o, a) { switch (e) { case "ellipse": return this.drawEllipsePath(t, n, r, i, o); case "polygon": return this.drawPolygonPath(t, n, r, i, o, a); case "roundrectangle": return this.drawRoundRectanglePath(t, n, r, i, o); case "cutrectangle": return this.drawCutRectanglePath(t, n, r, i, o); case "bottomroundrectangle": return this.drawBottomRoundRectanglePath(t, n, r, i, o); case "barrel": return this.drawBarrelPath(t, n, r, i, o) } } }; e.exports = r }, function (e, t, n) { "use strict"; var r = n(0), i = n(1), o = n(18), a = function e() { if (!(this instanceof e)) return new e; this.length = 0 }, s = a.prototype; s.instanceString = function () { return "stylesheet" }, s.selector = function (e) { return this[this.length++] = { selector: e, properties: [] }, this }, s.css = function (e, t) { var n = this.length - 1; if (r.string(e)) this[n].properties.push({ name: e, value: t }); else if (r.plainObject(e)) for (var a = e, s = 0; s < o.properties.length; s++) { var l = o.properties[s], c = a[l.name]; if (void 0 === c && (c = a[i.dash2camel(l.name)]), void 0 !== c) { var u = l.name, d = c; this[n].properties.push({ name: u, value: d }) } } return this }, s.style = s.css, s.generateStyle = function (e) { var t = new o(e); return this.appendToStyle(t) }, s.appendToStyle = function (e) { for (var t = 0; t < this.length; t++) { var n = this[t], r = n.selector, i = n.properties; e.selector(r); for (var o = 0; o < i.length; o++) { var a = i[o]; e.css(a.name, a.value) } } return e }, e.exports = a }, function (e, t, n) { "use strict"; e.exports = "snapshot-2fd4aa6cc2-1531011493999" }]) + }, e.exports = r(n(241), n(242)) + }).call(this, n(238).setImmediate) + }, function (e, t, n) { (function (e) { var r = void 0 !== e && e || "undefined" != typeof self && self || window, i = Function.prototype.apply; function o(e, t) { this._id = e, this._clearFn = t } t.setTimeout = function () { return new o(i.call(setTimeout, r, arguments), clearTimeout) }, t.setInterval = function () { return new o(i.call(setInterval, r, arguments), clearInterval) }, t.clearTimeout = t.clearInterval = function (e) { e && e.close() }, o.prototype.unref = o.prototype.ref = function () { }, o.prototype.close = function () { this._clearFn.call(r, this._id) }, t.enroll = function (e, t) { clearTimeout(e._idleTimeoutId), e._idleTimeout = t }, t.unenroll = function (e) { clearTimeout(e._idleTimeoutId), e._idleTimeout = -1 }, t._unrefActive = t.active = function (e) { clearTimeout(e._idleTimeoutId); var t = e._idleTimeout; t >= 0 && (e._idleTimeoutId = setTimeout((function () { e._onTimeout && e._onTimeout() }), t)) }, n(239), t.setImmediate = "undefined" != typeof self && self.setImmediate || void 0 !== e && e.setImmediate || this && this.setImmediate, t.clearImmediate = "undefined" != typeof self && self.clearImmediate || void 0 !== e && e.clearImmediate || this && this.clearImmediate }).call(this, n(35)) }, function (e, t, n) { (function (e, t) { !function (e, n) { "use strict"; if (!e.setImmediate) { var r, i, o, a, s, l = 1, c = {}, u = !1, d = e.document, p = Object.getPrototypeOf && Object.getPrototypeOf(e); p = p && p.setTimeout ? p : e, "[object process]" === {}.toString.call(e.process) ? r = function (e) { t.nextTick((function () { h(e) })) } : !function () { if (e.postMessage && !e.importScripts) { var t = !0, n = e.onmessage; return e.onmessage = function () { t = !1 }, e.postMessage("", "*"), e.onmessage = n, t } }() ? e.MessageChannel ? ((o = new MessageChannel).port1.onmessage = function (e) { h(e.data) }, r = function (e) { o.port2.postMessage(e) }) : d && "onreadystatechange" in d.createElement("script") ? (i = d.documentElement, r = function (e) { var t = d.createElement("script"); t.onreadystatechange = function () { h(e), t.onreadystatechange = null, i.removeChild(t), t = null }, i.appendChild(t) }) : r = function (e) { setTimeout(h, 0, e) } : (a = "setImmediate$" + Math.random() + "$", s = function (t) { t.source === e && "string" == typeof t.data && 0 === t.data.indexOf(a) && h(+t.data.slice(a.length)) }, e.addEventListener ? e.addEventListener("message", s, !1) : e.attachEvent("onmessage", s), r = function (t) { e.postMessage(a + t, "*") }), p.setImmediate = function (e) { "function" != typeof e && (e = new Function("" + e)); for (var t = new Array(arguments.length - 1), n = 0; n < t.length; n++)t[n] = arguments[n + 1]; var i = { callback: e, args: t }; return c[l] = i, r(l), l++ }, p.clearImmediate = f } function f(e) { delete c[e] } function h(e) { if (u) setTimeout(h, 0, e); else { var t = c[e]; if (t) { u = !0; try { !function (e) { var t = e.callback, n = e.args; switch (n.length) { case 0: t(); break; case 1: t(n[0]); break; case 2: t(n[0], n[1]); break; case 3: t(n[0], n[1], n[2]); break; default: t.apply(void 0, n) } }(t) } finally { f(e), u = !1 } } } } }("undefined" == typeof self ? void 0 === e ? this : e : self) }).call(this, n(35), n(240)) }, function (e, t) { var n, r, i = e.exports = {}; function o() { throw new Error("setTimeout has not been defined") } function a() { throw new Error("clearTimeout has not been defined") } function s(e) { if (n === setTimeout) return setTimeout(e, 0); if ((n === o || !n) && setTimeout) return n = setTimeout, setTimeout(e, 0); try { return n(e, 0) } catch (t) { try { return n.call(null, e, 0) } catch (t) { return n.call(this, e, 0) } } } !function () { try { n = "function" == typeof setTimeout ? setTimeout : o } catch (e) { n = o } try { r = "function" == typeof clearTimeout ? clearTimeout : a } catch (e) { r = a } }(); var l, c = [], u = !1, d = -1; function p() { u && l && (u = !1, l.length ? c = l.concat(c) : d = -1, c.length && f()) } function f() { if (!u) { var e = s(p); u = !0; for (var t = c.length; t;) { for (l = c, c = []; ++d < t;)l && l[d].run(); d = -1, t = c.length } l = null, u = !1, function (e) { if (r === clearTimeout) return clearTimeout(e); if ((r === a || !r) && clearTimeout) return r = clearTimeout, clearTimeout(e); try { r(e) } catch (t) { try { return r.call(null, e) } catch (t) { return r.call(this, e) } } }(e) } } function h(e, t) { this.fun = e, this.array = t } function g() { } i.nextTick = function (e) { var t = new Array(arguments.length - 1); if (arguments.length > 1) for (var n = 1; n < arguments.length; n++)t[n - 1] = arguments[n]; c.push(new h(e, t)), 1 !== c.length || u || s(f) }, h.prototype.run = function () { this.fun.apply(null, this.array) }, i.title = "browser", i.browser = !0, i.env = {}, i.argv = [], i.version = "", i.versions = {}, i.on = g, i.addListener = g, i.once = g, i.off = g, i.removeListener = g, i.removeAllListeners = g, i.emit = g, i.prependListener = g, i.prependOnceListener = g, i.listeners = function (e) { return [] }, i.binding = function (e) { throw new Error("process.binding is not supported") }, i.cwd = function () { return "/" }, i.chdir = function (e) { throw new Error("process.chdir is not supported") }, i.umask = function () { return 0 } }, function (e, t, n) { (function (t) { var n = /^\s+|\s+$/g, r = /^[-+]0x[0-9a-f]+$/i, i = /^0b[01]+$/i, o = /^0o[0-7]+$/i, a = parseInt, s = "object" == typeof t && t && t.Object === Object && t, l = "object" == typeof self && self && self.Object === Object && self, c = s || l || Function("return this")(), u = Object.prototype.toString, d = Math.max, p = Math.min, f = function () { return c.Date.now() }; function h(e) { var t = typeof e; return !!e && ("object" == t || "function" == t) } function g(e) { if ("number" == typeof e) return e; if (function (e) { return "symbol" == typeof e || function (e) { return !!e && "object" == typeof e }(e) && "[object Symbol]" == u.call(e) }(e)) return NaN; if (h(e)) { var t = "function" == typeof e.valueOf ? e.valueOf() : e; e = h(t) ? t + "" : t } if ("string" != typeof e) return 0 === e ? e : +e; e = e.replace(n, ""); var s = i.test(e); return s || o.test(e) ? a(e.slice(2), s ? 2 : 8) : r.test(e) ? NaN : +e } e.exports = function (e, t, n) { var r, i, o, a, s, l, c = 0, u = !1, m = !1, v = !0; if ("function" != typeof e) throw new TypeError("Expected a function"); function b(t) { var n = r, o = i; return r = i = void 0, c = t, a = e.apply(o, n) } function y(e) { return c = e, s = setTimeout(w, t), u ? b(e) : a } function x(e) { var n = e - l; return void 0 === l || n >= t || n < 0 || m && e - c >= o } function w() { var e = f(); if (x(e)) return k(e); s = setTimeout(w, function (e) { var n = t - (e - l); return m ? p(n, o - (e - c)) : n }(e)) } function k(e) { return s = void 0, v && r ? b(e) : (r = i = void 0, a) } function A() { var e = f(), n = x(e); if (r = arguments, i = this, l = e, n) { if (void 0 === s) return y(l); if (m) return s = setTimeout(w, t), b(l) } return void 0 === s && (s = setTimeout(w, t)), a } return t = g(t) || 0, h(n) && (u = !!n.leading, o = (m = "maxWait" in n) ? d(g(n.maxWait) || 0, t) : o, v = "trailing" in n ? !!n.trailing : v), A.cancel = function () { void 0 !== s && clearTimeout(s), c = 0, r = l = i = s = void 0 }, A.flush = function () { return void 0 === s ? a : k(f()) }, A } }).call(this, n(35)) }, function (e, t, n) { e.exports = n(243) }, function (e, t, n) { var r, i, o; (function () { var n, a, s, l, c, u, d, p, f, h, g, m, v, b, y; s = Math.floor, h = Math.min, a = function (e, t) { return e < t ? -1 : e > t ? 1 : 0 }, f = function (e, t, n, r, i) { var o; if (null == n && (n = 0), null == i && (i = a), n < 0) throw new Error("lo must be non-negative"); for (null == r && (r = e.length); n < r;)i(t, e[o = s((n + r) / 2)]) < 0 ? r = o : n = o + 1; return [].splice.apply(e, [n, n - n].concat(t)), t }, u = function (e, t, n) { return null == n && (n = a), e.push(t), b(e, 0, e.length - 1, n) }, c = function (e, t) { var n, r; return null == t && (t = a), n = e.pop(), e.length ? (r = e[0], e[0] = n, y(e, 0, t)) : r = n, r }, p = function (e, t, n) { var r; return null == n && (n = a), r = e[0], e[0] = t, y(e, 0, n), r }, d = function (e, t, n) { var r; return null == n && (n = a), e.length && n(e[0], t) < 0 && (t = (r = [e[0], t])[0], e[0] = r[1], y(e, 0, n)), t }, l = function (e, t) { var n, r, i, o, l, c; for (null == t && (t = a), l = [], r = 0, i = (o = function () { c = []; for (var t = 0, n = s(e.length / 2); 0 <= n ? t < n : t > n; 0 <= n ? t++ : t--)c.push(t); return c }.apply(this).reverse()).length; r < i; r++)n = o[r], l.push(y(e, n, t)); return l }, v = function (e, t, n) { var r; if (null == n && (n = a), -1 !== (r = e.indexOf(t))) return b(e, 0, r, n), y(e, r, n) }, g = function (e, t, n) { var r, i, o, s, c; if (null == n && (n = a), !(i = e.slice(0, t)).length) return i; for (l(i, n), o = 0, s = (c = e.slice(t)).length; o < s; o++)r = c[o], d(i, r, n); return i.sort(n).reverse() }, m = function (e, t, n) { var r, i, o, s, u, d, p, g, m; if (null == n && (n = a), 10 * t <= e.length) { if (!(o = e.slice(0, t).sort(n)).length) return o; for (i = o[o.length - 1], s = 0, d = (p = e.slice(t)).length; s < d; s++)n(r = p[s], i) < 0 && (f(o, r, 0, null, n), o.pop(), i = o[o.length - 1]); return o } for (l(e, n), m = [], u = 0, g = h(t, e.length); 0 <= g ? u < g : u > g; 0 <= g ? ++u : --u)m.push(c(e, n)); return m }, b = function (e, t, n, r) { var i, o, s; for (null == r && (r = a), i = e[n]; n > t && r(i, o = e[s = n - 1 >> 1]) < 0;)e[n] = o, n = s; return e[n] = i }, y = function (e, t, n) { var r, i, o, s, l; for (null == n && (n = a), i = e.length, l = t, o = e[t], r = 2 * t + 1; r < i;)(s = r + 1) < i && !(n(e[r], e[s]) < 0) && (r = s), e[t] = e[r], r = 2 * (t = r) + 1; return e[t] = o, b(e, l, t, n) }, n = function () { function e(e) { this.cmp = null != e ? e : a, this.nodes = [] } return e.push = u, e.pop = c, e.replace = p, e.pushpop = d, e.heapify = l, e.updateItem = v, e.nlargest = g, e.nsmallest = m, e.prototype.push = function (e) { return u(this.nodes, e, this.cmp) }, e.prototype.pop = function () { return c(this.nodes, this.cmp) }, e.prototype.peek = function () { return this.nodes[0] }, e.prototype.contains = function (e) { return -1 !== this.nodes.indexOf(e) }, e.prototype.replace = function (e) { return p(this.nodes, e, this.cmp) }, e.prototype.pushpop = function (e) { return d(this.nodes, e, this.cmp) }, e.prototype.heapify = function () { return l(this.nodes, this.cmp) }, e.prototype.updateItem = function (e) { return v(this.nodes, e, this.cmp) }, e.prototype.clear = function () { return this.nodes = [] }, e.prototype.empty = function () { return 0 === this.nodes.length }, e.prototype.size = function () { return this.nodes.length }, e.prototype.clone = function () { var t; return (t = new e).nodes = this.nodes.slice(0), t }, e.prototype.toArray = function () { return this.nodes.slice(0) }, e.prototype.insert = e.prototype.push, e.prototype.top = e.prototype.peek, e.prototype.front = e.prototype.peek, e.prototype.has = e.prototype.contains, e.prototype.copy = e.prototype.clone, e }(), i = [], void 0 === (o = "function" == typeof (r = function () { return n }) ? r.apply(t, i) : r) || (e.exports = o) }).call(this) }, function (e, t, n) { var r; !function () { "use strict"; var i = "undefined" == typeof jQuery ? null : jQuery, o = function (e, t) { if (e) { var n, r = { menuItems: [], menuItemClasses: [], contextMenuClasses: [] }; e("core", "contextMenus", (function (e) { var i = this; i.scratch("cycontextmenus") || i.scratch("cycontextmenus", {}); var o, a = l("options"), s = l("cxtMenu"); function l(e) { return i.scratch("cycontextmenus")[e] } function c(e, t) { i.scratch("cycontextmenus")[e] = t } function u(e) { for (var t = "", n = 0; n < e.length; n++) { t += e[n], n !== e.length - 1 && (t += " ") } return t } function d(e) { e.css("display", "block") } function p(e) { e.css("display", "none") } function f(e, n, r) { function o(n) { c("currentCyEvent", n), function (e) { var n = l("cxtMenuPosition"), r = e.position || e.cyPosition; if (n != r) { s.children().css("display", "none"), c("anyVisibleChild", !1), c("cxtMenuPosition", r); var o = t(i.container()).offset(), a = e.renderedPosition || e.cyRenderedPosition, u = o.left + a.x, d = o.top + a.y; s.css("left", u), s.css("top", d) } }(n), e.data("show") && (s.is(":visible") || d(s), c("anyVisibleChild", !0), d(e)), !l("anyVisibleChild") && s.is(":visible") && p(s) } var a, u; r && i.on("cxttap", u = function (e) { (e.target || e.cyTarget) == i && o(e) }), n && i.on("cxttap", n, a = function (e) { o(e) }), e.data("cy-context-menus-cxtfcn", a), e.data("cy-context-menus-cxtcorefcn", u) } function h(e, t, n, r) { !function (e, t) { var n; e.on("click", n = function () { t(l("currentCyEvent")) }), e.data("call-on-click-function", n) }(e, t), f(e, n, r) } function g(e) { for (var t = 0; t < e.length; t++)m(e[t]) } function m(e) { var t, n = b(e); t = n, s.append(t), function (e) { e.click((function () { p(s), c("cxtMenuPosition", void 0) })) }(t), h(n, e.onClickFunction, e.selector, e.coreAsWell) } function v(e, n) { var r, i, o = b(e); r = o, i = t("#" + n), r.insertBefore(i), h(o, e.onClickFunction, e.selector, e.coreAsWell) } function b(e) { var n, r, i, o = (n = a.menuItemClasses, r = e.hasTrailingDivider, i = u(n), i += " cy-context-menus-cxt-menuitem", r && (i += " cy-context-menus-divider"), i), s = '" : s += ">" + e.content + ""; var l = t(s); return l.data("selector", e.selector), l.data("on-click-function", e.onClickFunction), l.data("show", void 0 === e.show || e.show), l } function y() { var e; l("active") && (e = s.children(), t(e).each((function () { x(t(this)) })), i.off("tapstart", n), s.remove(), c(s = void 0, void 0), c("active", !1), c("anyVisibleChild", !1)) } function x(e) { var n = "string" == typeof e ? t("#" + e) : e, r = n.data("cy-context-menus-cxtfcn"), o = n.data("selector"), a = n.data("call-on-click-function"), s = n.data("cy-context-menus-cxtcorefcn"); r && i.off("cxttap", o, r), s && i.off("cxttap", s), a && n.off("click", a), n.remove() } "get" !== e && (c("options", a = function (e, t) { var n = {}; for (var r in e) n[r] = e[r]; for (var r in t) n[r] = t[r]; return n }(r, e)), l("active") && y(), c("active", !0), o = u(a.contextMenuClasses), (s = t("
")).addClass("cy-context-menus-cxt-menu"), c("cxtMenu", s), t("body").append(s), s = s, g(a.menuItems), i.on("tapstart", n = function () { p(s), c("cxtMenuPosition", void 0), c("currentCyEvent", void 0) }), t(".cy-context-menus-cxt-menu").contextmenu((function () { return !1 }))); return function (e) { return { isActive: function () { return l("active") }, appendMenuItem: function (t) { return m(t), e }, appendMenuItems: function (t) { return g(t), e }, removeMenuItem: function (t) { return x(t), e }, setTrailingDivider: function (n, r) { return function (e, n) { var r = t("#" + e); n ? r.addClass("cy-context-menus-divider") : r.removeClass("cy-context-menus-divider") }(n, r), e }, insertBeforeMenuItem: function (t, n) { return v(t, n), e }, moveBeforeOtherMenuItem: function (n, r) { return function (e, n) { if (e !== n) { var r = t("#" + e).detach(), i = t("#" + n); r.insertBefore(i) } }(n, r), e }, disableMenuItem: function (n) { return t("#" + n).attr("disabled", !0), e }, enableMenuItem: function (n) { return t("#" + n).attr("disabled", !1), e }, hideMenuItem: function (n) { return t("#" + n).data("show", !1), p(t("#" + n)), e }, showMenuItem: function (n) { return t("#" + n).data("show", !0), d(t("#" + n)), e }, destroy: function () { return y(), e } } }(this) })) } }; e.exports && (e.exports = o), void 0 === (r = function () { return o }.call(t, n, t, e)) || (e.exports = r), "undefined" != typeof cytoscape && i && o(cytoscape, i) }() }, function (e, t, n) { var r; r = function (e) { return function (e) { var t = {}; function n(r) { if (t[r]) return t[r].exports; var i = t[r] = { i: r, l: !1, exports: {} }; return e[r].call(i.exports, i, i.exports, n), i.l = !0, i.exports } return n.m = e, n.c = t, n.d = function (e, t, r) { n.o(e, t) || Object.defineProperty(e, t, { enumerable: !0, get: r }) }, n.r = function (e) { "undefined" != typeof Symbol && Symbol.toStringTag && Object.defineProperty(e, Symbol.toStringTag, { value: "Module" }), Object.defineProperty(e, "__esModule", { value: !0 }) }, n.t = function (e, t) { if (1 & t && (e = n(e)), 8 & t) return e; if (4 & t && "object" == typeof e && e && e.__esModule) return e; var r = Object.create(null); if (n.r(r), Object.defineProperty(r, "default", { enumerable: !0, value: e }), 2 & t && "string" != typeof e) for (var i in e) n.d(r, i, function (t) { return e[t] }.bind(null, i)); return r }, n.n = function (e) { var t = e && e.__esModule ? function () { return e.default } : function () { return e }; return n.d(t, "a", t), t }, n.o = function (e, t) { return Object.prototype.hasOwnProperty.call(e, t) }, n.p = "", n(n.s = 0) }([function (e, t, n) { var r = n(1), i = function (e) { e && e("layout", "dagre", r) }; "undefined" != typeof cytoscape && i(cytoscape), e.exports = i }, function (e, t, n) { function r(e) { return (r = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (e) { return typeof e } : function (e) { return e && "function" == typeof Symbol && e.constructor === Symbol && e !== Symbol.prototype ? "symbol" : typeof e })(e) } var i = n(2), o = n(3), a = n(4); function s(e) { this.options = o({}, i, e) } s.prototype.run = function () { var e = this.options, t = e.cy, n = e.eles, i = function (e, t) { return "function" == typeof t ? t.apply(e, [e]) : t }, o = e.boundingBox || { x1: 0, y1: 0, w: t.width(), h: t.height() }; void 0 === o.x2 && (o.x2 = o.x1 + o.w), void 0 === o.w && (o.w = o.x2 - o.x1), void 0 === o.y2 && (o.y2 = o.y1 + o.h), void 0 === o.h && (o.h = o.y2 - o.y1); var s = new a.graphlib.Graph({ multigraph: !0, compound: !0 }), l = {}, c = function (e, t) { null != t && (l[e] = t) }; c("nodesep", e.nodeSep), c("edgesep", e.edgeSep), c("ranksep", e.rankSep), c("rankdir", e.rankDir), c("ranker", e.ranker), s.setGraph(l), s.setDefaultEdgeLabel((function () { return {} })), s.setDefaultNodeLabel((function () { return {} })); for (var u = n.nodes(), d = 0; d < u.length; d++) { var p = u[d], f = p.layoutDimensions(e); s.setNode(p.id(), { width: f.w, height: f.h, name: p.id() }) } for (var h = 0; h < u.length; h++) { var g = u[h]; g.isChild() && s.setParent(g.id(), g.parent().id()) } for (var m = n.edges().stdFilter((function (e) { return !e.source().isParent() && !e.target().isParent() })), v = 0; v < m.length; v++) { var b = m[v]; s.setEdge(b.source().id(), b.target().id(), { minlen: i(b, e.minLen), weight: i(b, e.edgeWeight), name: b.id() }, b.id()) } a.layout(s); for (var y, x = s.nodes(), w = 0; w < x.length; w++) { var k = x[w], A = s.node(k); t.getElementById(k).scratch().dagre = A } return e.boundingBox ? (y = { x1: 1 / 0, x2: -1 / 0, y1: 1 / 0, y2: -1 / 0 }, u.forEach((function (e) { var t = e.scratch().dagre; y.x1 = Math.min(y.x1, t.x), y.x2 = Math.max(y.x2, t.x), y.y1 = Math.min(y.y1, t.y), y.y2 = Math.max(y.y2, t.y) })), y.w = y.x2 - y.x1, y.h = y.y2 - y.y1) : y = o, u.layoutPositions(this, e, (function (t) { var n = (t = "object" === r(t) ? t : this).scratch().dagre; return function (t) { if (e.boundingBox) { var n = 0 === y.w ? 0 : (t.x - y.x1) / y.w, r = 0 === y.h ? 0 : (t.y - y.y1) / y.h; return { x: o.x1 + n * o.w, y: o.y1 + r * o.h } } return t }({ x: n.x, y: n.y }) })), this }, e.exports = s }, function (e, t) { var n = { nodeSep: void 0, edgeSep: void 0, rankSep: void 0, rankDir: void 0, ranker: void 0, minLen: function (e) { return 1 }, edgeWeight: function (e) { return 1 }, fit: !0, padding: 30, spacingFactor: void 0, nodeDimensionsIncludeLabels: !1, animate: !1, animateFilter: function (e, t) { return !0 }, animationDuration: 500, animationEasing: void 0, boundingBox: void 0, transform: function (e, t) { return t }, ready: function () { }, stop: function () { } }; e.exports = n }, function (e, t) { e.exports = null != Object.assign ? Object.assign.bind(Object) : function (e) { for (var t = arguments.length, n = new Array(t > 1 ? t - 1 : 0), r = 1; r < t; r++)n[r - 1] = arguments[r]; return n.forEach((function (t) { Object.keys(t).forEach((function (n) { return e[n] = t[n] })) })), e } }, function (t, n) { t.exports = e }]) }, e.exports = r(n(246)) }, function (e, t, n) { e.exports = { graphlib: n(30), layout: n(359), debug: n(420), util: { time: n(20).time, notime: n(20).notime }, version: n(421) } }, function (e, t, n) { e.exports = { Graph: n(116), version: n(348) } }, function (e, t, n) { var r = n(149); e.exports = function (e) { return r(e, 4) } }, function (e, t) { e.exports = function () { this.__data__ = [], this.size = 0 } }, function (e, t, n) { var r = n(75), i = Array.prototype.splice; e.exports = function (e) { var t = this.__data__, n = r(t, e); return !(n < 0) && (n == t.length - 1 ? t.pop() : i.call(t, n, 1), --this.size, !0) } }, function (e, t, n) { var r = n(75); e.exports = function (e) { var t = this.__data__, n = r(t, e); return n < 0 ? void 0 : t[n][1] } }, function (e, t, n) { var r = n(75); e.exports = function (e) { return r(this.__data__, e) > -1 } }, function (e, t, n) { var r = n(75); e.exports = function (e, t) { var n = this.__data__, i = r(n, e); return i < 0 ? (++this.size, n.push([e, t])) : n[i][1] = t, this } }, function (e, t, n) { var r = n(74); e.exports = function () { this.__data__ = new r, this.size = 0 } }, function (e, t) { e.exports = function (e) { var t = this.__data__, n = t.delete(e); return this.size = t.size, n } }, function (e, t) { e.exports = function (e) { return this.__data__.get(e) } }, function (e, t) { e.exports = function (e) { return this.__data__.has(e) } }, function (e, t, n) { var r = n(74), i = n(117), o = n(118); e.exports = function (e, t) { var n = this.__data__; if (n instanceof r) { var a = n.__data__; if (!i || a.length < 199) return a.push([e, t]), this.size = ++n.size, this; n = this.__data__ = new o(a) } return n.set(e, t), this.size = n.size, this } }, function (e, t, n) { var r = n(64), i = n(262), o = n(23), a = n(151), s = /^\[object .+?Constructor\]$/, l = Function.prototype, c = Object.prototype, u = l.toString, d = c.hasOwnProperty, p = RegExp("^" + u.call(d).replace(/[\\^$.*+?()[\]{}|]/g, "\\$&").replace(/hasOwnProperty|(function).*?(?=\\\()| for .+?(?=\\\])/g, "$1.*?") + "$"); e.exports = function (e) { return !(!o(e) || i(e)) && (r(e) ? p : s).test(a(e)) } }, function (e, t, n) { var r = n(58), i = Object.prototype, o = i.hasOwnProperty, a = i.toString, s = r ? r.toStringTag : void 0; e.exports = function (e) { var t = o.call(e, s), n = e[s]; try { e[s] = void 0; var r = !0 } catch (e) { } var i = a.call(e); return r && (t ? e[s] = n : delete e[s]), i } }, function (e, t) { var n = Object.prototype.toString; e.exports = function (e) { return n.call(e) } }, function (e, t, n) { var r, i = n(263), o = (r = /[^.]+$/.exec(i && i.keys && i.keys.IE_PROTO || "")) ? "Symbol(src)_1." + r : ""; e.exports = function (e) { return !!o && o in e } }, function (e, t, n) { var r = n(31)["__core-js_shared__"]; e.exports = r }, function (e, t) { e.exports = function (e, t) { return null == e ? void 0 : e[t] } }, function (e, t, n) { var r = n(266), i = n(74), o = n(117); e.exports = function () { this.size = 0, this.__data__ = { hash: new r, map: new (o || i), string: new r } } }, function (e, t, n) { var r = n(267), i = n(268), o = n(269), a = n(270), s = n(271); function l(e) { var t = -1, n = null == e ? 0 : e.length; for (this.clear(); ++t < n;) { var r = e[t]; this.set(r[0], r[1]) } } l.prototype.clear = r, l.prototype.delete = i, l.prototype.get = o, l.prototype.has = a, l.prototype.set = s, e.exports = l }, function (e, t, n) { var r = n(76); e.exports = function () { this.__data__ = r ? r(null) : {}, this.size = 0 } }, function (e, t) { e.exports = function (e) { var t = this.has(e) && delete this.__data__[e]; return this.size -= t ? 1 : 0, t } }, function (e, t, n) { var r = n(76), i = Object.prototype.hasOwnProperty; e.exports = function (e) { var t = this.__data__; if (r) { var n = t[e]; return "__lodash_hash_undefined__" === n ? void 0 : n } return i.call(t, e) ? t[e] : void 0 } }, function (e, t, n) { var r = n(76), i = Object.prototype.hasOwnProperty; e.exports = function (e) { var t = this.__data__; return r ? void 0 !== t[e] : i.call(t, e) } }, function (e, t, n) { var r = n(76); e.exports = function (e, t) { var n = this.__data__; return this.size += this.has(e) ? 0 : 1, n[e] = r && void 0 === t ? "__lodash_hash_undefined__" : t, this } }, function (e, t, n) { var r = n(77); e.exports = function (e) { var t = r(this, e).delete(e); return this.size -= t ? 1 : 0, t } }, function (e, t) { e.exports = function (e) { var t = typeof e; return "string" == t || "number" == t || "symbol" == t || "boolean" == t ? "__proto__" !== e : null === e } }, function (e, t, n) { var r = n(77); e.exports = function (e) { return r(this, e).get(e) } }, function (e, t, n) { var r = n(77); e.exports = function (e) { return r(this, e).has(e) } }, function (e, t, n) { var r = n(77); e.exports = function (e, t) { var n = r(this, e), i = n.size; return n.set(e, t), this.size += n.size == i ? 0 : 1, this } }, function (e, t, n) { var r = n(65), i = n(41); e.exports = function (e, t) { return e && r(t, i(t), e) } }, function (e, t) { e.exports = function (e, t) { for (var n = -1, r = Array(e); ++n < e;)r[n] = t(n); return r } }, function (e, t, n) { var r = n(47), i = n(34); e.exports = function (e) { return i(e) && "[object Arguments]" == r(e) } }, function (e, t) { e.exports = function () { return !1 } }, function (e, t, n) { var r = n(47), i = n(120), o = n(34), a = {}; a["[object Float32Array]"] = a["[object Float64Array]"] = a["[object Int8Array]"] = a["[object Int16Array]"] = a["[object Int32Array]"] = a["[object Uint8Array]"] = a["[object Uint8ClampedArray]"] = a["[object Uint16Array]"] = a["[object Uint32Array]"] = !0, a["[object Arguments]"] = a["[object Array]"] = a["[object ArrayBuffer]"] = a["[object Boolean]"] = a["[object DataView]"] = a["[object Date]"] = a["[object Error]"] = a["[object Function]"] = a["[object Map]"] = a["[object Number]"] = a["[object Object]"] = a["[object RegExp]"] = a["[object Set]"] = a["[object String]"] = a["[object WeakMap]"] = !1, e.exports = function (e) { return o(e) && i(e.length) && !!a[r(e)] } }, function (e, t, n) { var r = n(154)(Object.keys, Object); e.exports = r }, function (e, t, n) { var r = n(65), i = n(48); e.exports = function (e, t) { return e && r(t, i(t), e) } }, function (e, t, n) { var r = n(23), i = n(83), o = n(285), a = Object.prototype.hasOwnProperty; e.exports = function (e) { if (!r(e)) return o(e); var t = i(e), n = []; for (var s in e) ("constructor" != s || !t && a.call(e, s)) && n.push(s); return n } }, function (e, t) { e.exports = function (e) { var t = []; if (null != e) for (var n in Object(e)) t.push(n); return t } }, function (e, t, n) { var r = n(65), i = n(123); e.exports = function (e, t) { return r(e, i(e), t) } }, function (e, t, n) { var r = n(65), i = n(159); e.exports = function (e, t) { return r(e, i(e), t) } }, function (e, t, n) { var r = n(161), i = n(159), o = n(48); e.exports = function (e) { return r(e, o, i) } }, function (e, t, n) { var r = n(46)(n(31), "DataView"); e.exports = r }, function (e, t, n) { var r = n(46)(n(31), "Promise"); e.exports = r }, function (e, t, n) { var r = n(46)(n(31), "WeakMap"); e.exports = r }, function (e, t) { var n = Object.prototype.hasOwnProperty; e.exports = function (e) { var t = e.length, r = new e.constructor(t); return t && "string" == typeof e[0] && n.call(e, "index") && (r.index = e.index, r.input = e.input), r } }, function (e, t, n) { var r = n(125), i = n(294), o = n(295), a = n(296), s = n(164); e.exports = function (e, t, n) { var l = e.constructor; switch (t) { case "[object ArrayBuffer]": return r(e); case "[object Boolean]": case "[object Date]": return new l(+e); case "[object DataView]": return i(e, n); case "[object Float32Array]": case "[object Float64Array]": case "[object Int8Array]": case "[object Int16Array]": case "[object Int32Array]": case "[object Uint8Array]": case "[object Uint8ClampedArray]": case "[object Uint16Array]": case "[object Uint32Array]": return s(e, n); case "[object Map]": return new l; case "[object Number]": case "[object String]": return new l(e); case "[object RegExp]": return o(e); case "[object Set]": return new l; case "[object Symbol]": return a(e) } } }, function (e, t, n) { var r = n(125); e.exports = function (e, t) { var n = t ? r(e.buffer) : e.buffer; return new e.constructor(n, e.byteOffset, e.byteLength) } }, function (e, t) { var n = /\w*$/; e.exports = function (e) { var t = new e.constructor(e.source, n.exec(e)); return t.lastIndex = e.lastIndex, t } }, function (e, t, n) { var r = n(58), i = r ? r.prototype : void 0, o = i ? i.valueOf : void 0; e.exports = function (e) { return o ? Object(o.call(e)) : {} } }, function (e, t, n) { var r = n(298), i = n(82), o = n(121), a = o && o.isMap, s = a ? i(a) : r; e.exports = s }, function (e, t, n) { var r = n(60), i = n(34); e.exports = function (e) { return i(e) && "[object Map]" == r(e) } }, function (e, t, n) { var r = n(300), i = n(82), o = n(121), a = o && o.isSet, s = a ? i(a) : r; e.exports = s }, function (e, t, n) { var r = n(60), i = n(34); e.exports = function (e) { return i(e) && "[object Set]" == r(e) } }, function (e, t) { e.exports = function (e) { return function (t, n, r) { for (var i = -1, o = Object(t), a = r(t), s = a.length; s--;) { var l = a[e ? s : ++i]; if (!1 === n(o[l], l, o)) break } return t } } }, function (e, t, n) { var r = n(36); e.exports = function (e, t) { return function (n, i) { if (null == n) return n; if (!r(n)) return e(n, i); for (var o = n.length, a = t ? o : -1, s = Object(n); (t ? a-- : ++a < o) && !1 !== i(s[a], a, s);); return n } } }, function (e, t, n) { var r = n(85); e.exports = function (e, t) { var n = []; return r(e, (function (e, r, i) { t(e, r, i) && n.push(e) })), n } }, function (e, t, n) { var r = n(305), i = n(313), o = n(176); e.exports = function (e) { var t = i(e); return 1 == t.length && t[0][2] ? o(t[0][0], t[0][1]) : function (n) { return n === e || r(n, e, t) } } }, function (e, t, n) { var r = n(73), i = n(171); e.exports = function (e, t, n, o) { var a = n.length, s = a, l = !o; if (null == e) return !s; for (e = Object(e); a--;) { var c = n[a]; if (l && c[2] ? c[1] !== e[c[0]] : !(c[0] in e)) return !1 } for (; ++a < s;) { var u = (c = n[a])[0], d = e[u], p = c[1]; if (l && c[2]) { if (void 0 === d && !(u in e)) return !1 } else { var f = new r; if (o) var h = o(d, p, u, e, t, f); if (!(void 0 === h ? i(p, d, 3, o, f) : h)) return !1 } } return !0 } }, function (e, t, n) { var r = n(73), i = n(172), o = n(310), a = n(312), s = n(60), l = n(13), c = n(59), u = n(67), d = "[object Object]", p = Object.prototype.hasOwnProperty; e.exports = function (e, t, n, f, h, g) { var m = l(e), v = l(t), b = m ? "[object Array]" : s(e), y = v ? "[object Array]" : s(t), x = (b = "[object Arguments]" == b ? d : b) == d, w = (y = "[object Arguments]" == y ? d : y) == d, k = b == y; if (k && c(e)) { if (!c(t)) return !1; m = !0, x = !1 } if (k && !x) return g || (g = new r), m || u(e) ? i(e, t, n, f, h, g) : o(e, t, b, n, f, h, g); if (!(1 & n)) { var A = x && p.call(e, "__wrapped__"), E = w && p.call(t, "__wrapped__"); if (A || E) { var S = A ? e.value() : e, $ = E ? t.value() : t; return g || (g = new r), h(S, $, n, f, g) } } return !!k && (g || (g = new r), a(e, t, n, f, h, g)) } }, function (e, t) { e.exports = function (e) { return this.__data__.set(e, "__lodash_hash_undefined__"), this } }, function (e, t) { e.exports = function (e) { return this.__data__.has(e) } }, function (e, t) { e.exports = function (e, t) { for (var n = -1, r = null == e ? 0 : e.length; ++n < r;)if (t(e[n], n, e)) return !0; return !1 } }, function (e, t, n) { var r = n(58), i = n(163), o = n(57), a = n(172), s = n(311), l = n(129), c = r ? r.prototype : void 0, u = c ? c.valueOf : void 0; e.exports = function (e, t, n, r, c, d, p) { switch (n) { case "[object DataView]": if (e.byteLength != t.byteLength || e.byteOffset != t.byteOffset) return !1; e = e.buffer, t = t.buffer; case "[object ArrayBuffer]": return !(e.byteLength != t.byteLength || !d(new i(e), new i(t))); case "[object Boolean]": case "[object Date]": case "[object Number]": return o(+e, +t); case "[object Error]": return e.name == t.name && e.message == t.message; case "[object RegExp]": case "[object String]": return e == t + ""; case "[object Map]": var f = s; case "[object Set]": var h = 1 & r; if (f || (f = l), e.size != t.size && !h) return !1; var g = p.get(e); if (g) return g == t; r |= 2, p.set(e, t); var m = a(f(e), f(t), r, c, d, p); return p.delete(e), m; case "[object Symbol]": if (u) return u.call(e) == u.call(t) }return !1 } }, function (e, t) { e.exports = function (e) { var t = -1, n = Array(e.size); return e.forEach((function (e, r) { n[++t] = [r, e] })), n } }, function (e, t, n) { var r = n(160), i = Object.prototype.hasOwnProperty; e.exports = function (e, t, n, o, a, s) { var l = 1 & n, c = r(e), u = c.length; if (u != r(t).length && !l) return !1; for (var d = u; d--;) { var p = c[d]; if (!(l ? p in t : i.call(t, p))) return !1 } var f = s.get(e), h = s.get(t); if (f && h) return f == t && h == e; var g = !0; s.set(e, t), s.set(t, e); for (var m = l; ++d < u;) { var v = e[p = c[d]], b = t[p]; if (o) var y = l ? o(b, v, p, t, e, s) : o(v, b, p, e, t, s); if (!(void 0 === y ? v === b || a(v, b, n, o, s) : y)) { g = !1; break } m || (m = "constructor" == p) } if (g && !m) { var x = e.constructor, w = t.constructor; x == w || !("constructor" in e) || !("constructor" in t) || "function" == typeof x && x instanceof x && "function" == typeof w && w instanceof w || (g = !1) } return s.delete(e), s.delete(t), g } }, function (e, t, n) { var r = n(175), i = n(41); e.exports = function (e) { for (var t = i(e), n = t.length; n--;) { var o = t[n], a = e[o]; t[n] = [o, a, r(a)] } return t } }, function (e, t, n) { var r = n(171), i = n(315), o = n(178), a = n(130), s = n(175), l = n(176), c = n(68); e.exports = function (e, t) { return a(e) && s(t) ? l(c(e), t) : function (n) { var a = i(n, e); return void 0 === a && a === t ? o(n, e) : r(t, a, 3) } } }, function (e, t, n) { var r = n(86); e.exports = function (e, t, n) { var i = null == e ? void 0 : r(e, t); return void 0 === i ? n : i } }, function (e, t, n) { var r = n(317), i = /[^.[\]]+|\[(?:(-?\d+(?:\.\d+)?)|(["'])((?:(?!\2)[^\\]|\\.)*?)\2)\]|(?=(?:\.|\[\])(?:\.|\[\]|$))/g, o = /\\(\\)?/g, a = r((function (e) { var t = []; return 46 === e.charCodeAt(0) && t.push(""), e.replace(i, (function (e, n, r, i) { t.push(r ? i.replace(o, "$1") : n || e) })), t })); e.exports = a }, function (e, t, n) { var r = n(318); e.exports = function (e) { var t = r(e, (function (e) { return 500 === n.size && n.clear(), e })), n = t.cache; return t } }, function (e, t, n) { var r = n(118); function i(e, t) { if ("function" != typeof e || null != t && "function" != typeof t) throw new TypeError("Expected a function"); var n = function () { var r = arguments, i = t ? t.apply(this, r) : r[0], o = n.cache; if (o.has(i)) return o.get(i); var a = e.apply(this, r); return n.cache = o.set(i, a) || o, a }; return n.cache = new (i.Cache || r), n } i.Cache = r, e.exports = i }, function (e, t, n) { var r = n(58), i = n(88), o = n(13), a = n(61), s = r ? r.prototype : void 0, l = s ? s.toString : void 0; e.exports = function e(t) { if ("string" == typeof t) return t; if (o(t)) return i(t, e) + ""; if (a(t)) return l ? l.call(t) : ""; var n = t + ""; return "0" == n && 1 / t == -1 / 0 ? "-0" : n } }, function (e, t) { e.exports = function (e, t) { return null != e && t in Object(e) } }, function (e, t, n) { var r = n(180), i = n(322), o = n(130), a = n(68); e.exports = function (e) { return o(e) ? r(a(e)) : i(e) } }, function (e, t, n) { var r = n(86); e.exports = function (e) { return function (t) { return r(t, e) } } }, function (e, t) { var n = Object.prototype.hasOwnProperty; e.exports = function (e, t) { return null != e && n.call(e, t) } }, function (e, t, n) { var r = n(122), i = n(60), o = n(66), a = n(13), s = n(36), l = n(59), c = n(83), u = n(67), d = Object.prototype.hasOwnProperty; e.exports = function (e) { if (null == e) return !0; if (s(e) && (a(e) || "string" == typeof e || "function" == typeof e.splice || l(e) || u(e) || o(e))) return !e.length; var t = i(e); if ("[object Map]" == t || "[object Set]" == t) return !e.size; if (c(e)) return !r(e).length; for (var n in e) if (d.call(e, n)) return !1; return !0 } }, function (e, t) { e.exports = function (e, t, n, r) { var i = -1, o = null == e ? 0 : e.length; for (r && o && (n = e[++i]); ++i < o;)n = t(n, e[i], i, e); return n } }, function (e, t) { e.exports = function (e, t, n, r, i) { return i(e, (function (e, i, o) { n = r ? (r = !1, e) : t(n, e, i, o) })), n } }, function (e, t, n) { var r = n(122), i = n(60), o = n(36), a = n(328), s = n(329); e.exports = function (e) { if (null == e) return 0; if (o(e)) return a(e) ? s(e) : e.length; var t = i(e); return "[object Map]" == t || "[object Set]" == t ? e.size : r(e).length } }, function (e, t, n) { var r = n(47), i = n(13), o = n(34); e.exports = function (e) { return "string" == typeof e || !i(e) && o(e) && "[object String]" == r(e) } }, function (e, t, n) { var r = n(330), i = n(331), o = n(332); e.exports = function (e) { return i(e) ? o(e) : r(e) } }, function (e, t, n) { var r = n(180)("length"); e.exports = r }, function (e, t) { var n = RegExp("[\\u200d\\ud800-\\udfff\\u0300-\\u036f\\ufe20-\\ufe2f\\u20d0-\\u20ff\\ufe0e\\ufe0f]"); e.exports = function (e) { return n.test(e) } }, function (e, t) { var n = "[\\ud800-\\udfff]", r = "[\\u0300-\\u036f\\ufe20-\\ufe2f\\u20d0-\\u20ff]", i = "\\ud83c[\\udffb-\\udfff]", o = "[^\\ud800-\\udfff]", a = "(?:\\ud83c[\\udde6-\\uddff]){2}", s = "[\\ud800-\\udbff][\\udc00-\\udfff]", l = "(?:" + r + "|" + i + ")" + "?", c = "[\\ufe0e\\ufe0f]?" + l + ("(?:\\u200d(?:" + [o, a, s].join("|") + ")[\\ufe0e\\ufe0f]?" + l + ")*"), u = "(?:" + [o + r + "?", r, a, s, n].join("|") + ")", d = RegExp(i + "(?=" + i + ")|" + u + c, "g"); e.exports = function (e) { for (var t = d.lastIndex = 0; d.test(e);)++t; return t } }, function (e, t, n) { var r = n(119), i = n(166), o = n(127), a = n(37), s = n(84), l = n(13), c = n(59), u = n(64), d = n(23), p = n(67); e.exports = function (e, t, n) { var f = l(e), h = f || c(e) || p(e); if (t = a(t, 4), null == n) { var g = e && e.constructor; n = h ? f ? new g : [] : d(e) && u(g) ? i(s(e)) : {} } return (h ? r : o)(e, (function (e, r, i) { return t(n, e, r, i) })), n } }, function (e, t, n) { var r = n(131), i = n(89), o = n(339), a = n(189), s = i((function (e) { return o(r(e, 1, a, !0)) })); e.exports = s }, function (e, t, n) { var r = n(58), i = n(66), o = n(13), a = r ? r.isConcatSpreadable : void 0; e.exports = function (e) { return o(e) || i(e) || !!(a && e && e[a]) } }, function (e, t) { e.exports = function (e, t, n) { switch (n.length) { case 0: return e.call(t); case 1: return e.call(t, n[0]); case 2: return e.call(t, n[0], n[1]); case 3: return e.call(t, n[0], n[1], n[2]) }return e.apply(t, n) } }, function (e, t, n) { var r = n(126), i = n(152), o = n(49), a = i ? function (e, t) { return i(e, "toString", { configurable: !0, enumerable: !1, value: r(t), writable: !0 }) } : o; e.exports = a }, function (e, t) { var n = Date.now; e.exports = function (e) { var t = 0, r = 0; return function () { var i = n(), o = 16 - (i - r); if (r = i, o > 0) { if (++t >= 800) return arguments[0] } else t = 0; return e.apply(void 0, arguments) } } }, function (e, t, n) { var r = n(173), i = n(340), o = n(344), a = n(174), s = n(345), l = n(129); e.exports = function (e, t, n) { var c = -1, u = i, d = e.length, p = !0, f = [], h = f; if (n) p = !1, u = o; else if (d >= 200) { var g = t ? null : s(e); if (g) return l(g); p = !1, u = a, h = new r } else h = t ? [] : f; e: for (; ++c < d;) { var m = e[c], v = t ? t(m) : m; if (m = n || 0 !== m ? m : 0, p && v == v) { for (var b = h.length; b--;)if (h[b] === v) continue e; t && h.push(v), f.push(m) } else u(h, v, n) || (h !== f && h.push(v), f.push(m)) } return f } }, function (e, t, n) { var r = n(341); e.exports = function (e, t) { return !!(null == e ? 0 : e.length) && r(e, t, 0) > -1 } }, function (e, t, n) { var r = n(188), i = n(342), o = n(343); e.exports = function (e, t, n) { return t == t ? o(e, t, n) : r(e, i, n) } }, function (e, t) { e.exports = function (e) { return e != e } }, function (e, t) { e.exports = function (e, t, n) { for (var r = n - 1, i = e.length; ++r < i;)if (e[r] === t) return r; return -1 } }, function (e, t) { e.exports = function (e, t, n) { for (var r = -1, i = null == e ? 0 : e.length; ++r < i;)if (n(t, e[r])) return !0; return !1 } }, function (e, t, n) { var r = n(162), i = n(346), o = n(129), a = r && 1 / o(new r([, -0]))[1] == 1 / 0 ? function (e) { return new r(e) } : i; e.exports = a }, function (e, t) { e.exports = function () { } }, function (e, t, n) { var r = n(88); e.exports = function (e, t) { return r(t, (function (t) { return e[t] })) } }, function (e, t) { e.exports = "2.1.8" }, function (e, t, n) { var r = n(22), i = n(116); function o(e) { return r.map(e.nodes(), (function (t) { var n = e.node(t), i = e.parent(t), o = { v: t }; return r.isUndefined(n) || (o.value = n), r.isUndefined(i) || (o.parent = i), o })) } function a(e) { return r.map(e.edges(), (function (t) { var n = e.edge(t), i = { v: t.v, w: t.w }; return r.isUndefined(t.name) || (i.name = t.name), r.isUndefined(n) || (i.value = n), i })) } e.exports = { write: function (e) { var t = { options: { directed: e.isDirected(), multigraph: e.isMultigraph(), compound: e.isCompound() }, nodes: o(e), edges: a(e) }; r.isUndefined(e.graph()) || (t.value = r.clone(e.graph())); return t }, read: function (e) { var t = new i(e.options).setGraph(e.value); return r.each(e.nodes, (function (e) { t.setNode(e.v, e.value), e.parent && t.setParent(e.v, e.parent) })), r.each(e.edges, (function (e) { t.setEdge({ v: e.v, w: e.w, name: e.name }, e.value) })), t } } }, function (e, t, n) { e.exports = { components: n(351), dijkstra: n(191), dijkstraAll: n(352), findCycles: n(353), floydWarshall: n(354), isAcyclic: n(355), postorder: n(356), preorder: n(357), prim: n(358), tarjan: n(193), topsort: n(194) } }, function (e, t, n) { var r = n(22); e.exports = function (e) { var t, n = {}, i = []; function o(i) { r.has(n, i) || (n[i] = !0, t.push(i), r.each(e.successors(i), o), r.each(e.predecessors(i), o)) } return r.each(e.nodes(), (function (e) { t = [], o(e), t.length && i.push(t) })), i } }, function (e, t, n) { var r = n(191), i = n(22); e.exports = function (e, t, n) { return i.transform(e.nodes(), (function (i, o) { i[o] = r(e, o, t, n) }), {}) } }, function (e, t, n) { var r = n(22), i = n(193); e.exports = function (e) { return r.filter(i(e), (function (t) { return t.length > 1 || 1 === t.length && e.hasEdge(t[0], t[0]) })) } }, function (e, t, n) { var r = n(22); e.exports = function (e, t, n) { return function (e, t, n) { var r = {}, i = e.nodes(); return i.forEach((function (e) { r[e] = {}, r[e][e] = { distance: 0 }, i.forEach((function (t) { e !== t && (r[e][t] = { distance: Number.POSITIVE_INFINITY }) })), n(e).forEach((function (n) { var i = n.v === e ? n.w : n.v, o = t(n); r[e][i] = { distance: o, predecessor: e } })) })), i.forEach((function (e) { var t = r[e]; i.forEach((function (n) { var o = r[n]; i.forEach((function (n) { var r = o[e], i = t[n], a = o[n], s = r.distance + i.distance; s < a.distance && (a.distance = s, a.predecessor = i.predecessor) })) })) })), r }(e, t || i, n || function (t) { return e.outEdges(t) }) }; var i = r.constant(1) }, function (e, t, n) { var r = n(194); e.exports = function (e) { try { r(e) } catch (e) { if (e instanceof r.CycleException) return !1; throw e } return !0 } }, function (e, t, n) { var r = n(195); e.exports = function (e, t) { return r(e, t, "post") } }, function (e, t, n) { var r = n(195); e.exports = function (e, t) { return r(e, t, "pre") } }, function (e, t, n) { var r = n(22), i = n(116), o = n(192); e.exports = function (e, t) { var n, a = new i, s = {}, l = new o; function c(e) { var r = e.v === n ? e.w : e.v, i = l.priority(r); if (void 0 !== i) { var o = t(e); o < i && (s[r] = n, l.decrease(r, o)) } } if (0 === e.nodeCount()) return a; r.each(e.nodes(), (function (e) { l.add(e, Number.POSITIVE_INFINITY), a.setNode(e) })), l.decrease(e.nodes()[0], 0); var u = !1; for (; l.size() > 0;) { if (n = l.removeMin(), r.has(s, n)) a.setEdge(n, s[n]); else { if (u) throw new Error("Input graph is not connected: " + e); u = !0 } e.nodeEdges(n).forEach(c) } return a } }, function (e, t, n) { "use strict"; var r = n(11), i = n(399), o = n(402), a = n(403), s = n(20).normalizeRanks, l = n(405), c = n(20).removeEmptyRanks, u = n(406), d = n(407), p = n(408), f = n(409), h = n(418), g = n(20), m = n(30).Graph; e.exports = function (e, t) { var n = t && t.debugTiming ? g.time : g.notime; n("layout", (function () { var t = n(" buildLayoutGraph", (function () { return function (e) { var t = new m({ multigraph: !0, compound: !0 }), n = $(e.graph()); return t.setGraph(r.merge({}, b, S(n, v), r.pick(n, y))), r.forEach(e.nodes(), (function (n) { var i = $(e.node(n)); t.setNode(n, r.defaults(S(i, x), w)), t.setParent(n, e.parent(n)) })), r.forEach(e.edges(), (function (n) { var i = $(e.edge(n)); t.setEdge(n, r.merge({}, A, S(i, k), r.pick(i, E))) })), t }(e) })); n(" runLayout", (function () { !function (e, t) { t(" makeSpaceForEdgeLabels", (function () { !function (e) { var t = e.graph(); t.ranksep /= 2, r.forEach(e.edges(), (function (n) { var r = e.edge(n); r.minlen *= 2, "c" !== r.labelpos.toLowerCase() && ("TB" === t.rankdir || "BT" === t.rankdir ? r.width += r.labeloffset : r.height += r.labeloffset) })) }(e) })), t(" removeSelfEdges", (function () { !function (e) { r.forEach(e.edges(), (function (t) { if (t.v === t.w) { var n = e.node(t.v); n.selfEdges || (n.selfEdges = []), n.selfEdges.push({ e: t, label: e.edge(t) }), e.removeEdge(t) } })) }(e) })), t(" acyclic", (function () { i.run(e) })), t(" nestingGraph.run", (function () { u.run(e) })), t(" rank", (function () { a(g.asNonCompoundGraph(e)) })), t(" injectEdgeLabelProxies", (function () { !function (e) { r.forEach(e.edges(), (function (t) { var n = e.edge(t); if (n.width && n.height) { var r = e.node(t.v), i = { rank: (e.node(t.w).rank - r.rank) / 2 + r.rank, e: t }; g.addDummyNode(e, "edge-proxy", i, "_ep") } })) }(e) })), t(" removeEmptyRanks", (function () { c(e) })), t(" nestingGraph.cleanup", (function () { u.cleanup(e) })), t(" normalizeRanks", (function () { s(e) })), t(" assignRankMinMax", (function () { !function (e) { var t = 0; r.forEach(e.nodes(), (function (n) { var i = e.node(n); i.borderTop && (i.minRank = e.node(i.borderTop).rank, i.maxRank = e.node(i.borderBottom).rank, t = r.max(t, i.maxRank)) })), e.graph().maxRank = t }(e) })), t(" removeEdgeLabelProxies", (function () { !function (e) { r.forEach(e.nodes(), (function (t) { var n = e.node(t); "edge-proxy" === n.dummy && (e.edge(n.e).labelRank = n.rank, e.removeNode(t)) })) }(e) })), t(" normalize.run", (function () { o.run(e) })), t(" parentDummyChains", (function () { l(e) })), t(" addBorderSegments", (function () { d(e) })), t(" order", (function () { f(e) })), t(" insertSelfEdges", (function () { !function (e) { var t = g.buildLayerMatrix(e); r.forEach(t, (function (t) { var n = 0; r.forEach(t, (function (t, i) { var o = e.node(t); o.order = i + n, r.forEach(o.selfEdges, (function (t) { g.addDummyNode(e, "selfedge", { width: t.label.width, height: t.label.height, rank: o.rank, order: i + ++n, e: t.e, label: t.label }, "_se") })), delete o.selfEdges })) })) }(e) })), t(" adjustCoordinateSystem", (function () { p.adjust(e) })), t(" position", (function () { h(e) })), t(" positionSelfEdges", (function () { !function (e) { r.forEach(e.nodes(), (function (t) { var n = e.node(t); if ("selfedge" === n.dummy) { var r = e.node(n.e.v), i = r.x + r.width / 2, o = r.y, a = n.x - i, s = r.height / 2; e.setEdge(n.e, n.label), e.removeNode(t), n.label.points = [{ x: i + 2 * a / 3, y: o - s }, { x: i + 5 * a / 6, y: o - s }, { x: i + a, y: o }, { x: i + 5 * a / 6, y: o + s }, { x: i + 2 * a / 3, y: o + s }], n.label.x = n.x, n.label.y = n.y } })) }(e) })), t(" removeBorderNodes", (function () { !function (e) { r.forEach(e.nodes(), (function (t) { if (e.children(t).length) { var n = e.node(t), i = e.node(n.borderTop), o = e.node(n.borderBottom), a = e.node(r.last(n.borderLeft)), s = e.node(r.last(n.borderRight)); n.width = Math.abs(s.x - a.x), n.height = Math.abs(o.y - i.y), n.x = a.x + n.width / 2, n.y = i.y + n.height / 2 } })), r.forEach(e.nodes(), (function (t) { "border" === e.node(t).dummy && e.removeNode(t) })) }(e) })), t(" normalize.undo", (function () { o.undo(e) })), t(" fixupEdgeLabelCoords", (function () { !function (e) { r.forEach(e.edges(), (function (t) { var n = e.edge(t); if (r.has(n, "x")) switch ("l" !== n.labelpos && "r" !== n.labelpos || (n.width -= n.labeloffset), n.labelpos) { case "l": n.x -= n.width / 2 + n.labeloffset; break; case "r": n.x += n.width / 2 + n.labeloffset } })) }(e) })), t(" undoCoordinateSystem", (function () { p.undo(e) })), t(" translateGraph", (function () { !function (e) { var t = Number.POSITIVE_INFINITY, n = 0, i = Number.POSITIVE_INFINITY, o = 0, a = e.graph(), s = a.marginx || 0, l = a.marginy || 0; function c(e) { var r = e.x, a = e.y, s = e.width, l = e.height; t = Math.min(t, r - s / 2), n = Math.max(n, r + s / 2), i = Math.min(i, a - l / 2), o = Math.max(o, a + l / 2) } r.forEach(e.nodes(), (function (t) { c(e.node(t)) })), r.forEach(e.edges(), (function (t) { var n = e.edge(t); r.has(n, "x") && c(n) })), t -= s, i -= l, r.forEach(e.nodes(), (function (n) { var r = e.node(n); r.x -= t, r.y -= i })), r.forEach(e.edges(), (function (n) { var o = e.edge(n); r.forEach(o.points, (function (e) { e.x -= t, e.y -= i })), r.has(o, "x") && (o.x -= t), r.has(o, "y") && (o.y -= i) })), a.width = n - t + s, a.height = o - i + l }(e) })), t(" assignNodeIntersects", (function () { !function (e) { r.forEach(e.edges(), (function (t) { var n, r, i = e.edge(t), o = e.node(t.v), a = e.node(t.w); i.points ? (n = i.points[0], r = i.points[i.points.length - 1]) : (i.points = [], n = a, r = o), i.points.unshift(g.intersectRect(o, n)), i.points.push(g.intersectRect(a, r)) })) }(e) })), t(" reversePoints", (function () { !function (e) { r.forEach(e.edges(), (function (t) { var n = e.edge(t); n.reversed && n.points.reverse() })) }(e) })), t(" acyclic.undo", (function () { i.undo(e) })) }(t, n) })), n(" updateInputGraph", (function () { !function (e, t) { r.forEach(e.nodes(), (function (n) { var r = e.node(n), i = t.node(n); r && (r.x = i.x, r.y = i.y, t.children(n).length && (r.width = i.width, r.height = i.height)) })), r.forEach(e.edges(), (function (n) { var i = e.edge(n), o = t.edge(n); i.points = o.points, r.has(o, "x") && (i.x = o.x, i.y = o.y) })), e.graph().width = t.graph().width, e.graph().height = t.graph().height }(e, t) })) })) }; var v = ["nodesep", "edgesep", "ranksep", "marginx", "marginy"], b = { ranksep: 50, edgesep: 20, nodesep: 50, rankdir: "tb" }, y = ["acyclicer", "ranker", "rankdir", "align"], x = ["width", "height"], w = { width: 0, height: 0 }, k = ["minlen", "weight", "width", "height", "labeloffset"], A = { minlen: 1, weight: 1, width: 0, height: 0, labeloffset: 10, labelpos: "r" }, E = ["labelpos"]; function S(e, t) { return r.mapValues(r.pick(e, t), Number) } function $(e) { var t = {}; return r.forEach(e, (function (e, n) { t[n.toLowerCase()] = e })), t } }, function (e, t, n) { var r = n(149); e.exports = function (e) { return r(e, 5) } }, function (e, t, n) { var r = n(89), i = n(57), o = n(90), a = n(48), s = Object.prototype, l = s.hasOwnProperty, c = r((function (e, t) { e = Object(e); var n = -1, r = t.length, c = r > 2 ? t[2] : void 0; for (c && o(t[0], t[1], c) && (r = 1); ++n < r;)for (var u = t[n], d = a(u), p = -1, f = d.length; ++p < f;) { var h = d[p], g = e[h]; (void 0 === g || i(g, s[h]) && !l.call(e, h)) && (e[h] = u[h]) } return e })); e.exports = c }, function (e, t, n) { var r = n(363)(n(364)); e.exports = r }, function (e, t, n) { var r = n(37), i = n(36), o = n(41); e.exports = function (e) { return function (t, n, a) { var s = Object(t); if (!i(t)) { var l = r(n, 3); t = o(t), n = function (e) { return l(s[e], e, s) } } var c = e(t, n, a); return c > -1 ? s[l ? t[c] : c] : void 0 } } }, function (e, t, n) { var r = n(188), i = n(37), o = n(365), a = Math.max; e.exports = function (e, t, n) { var s = null == e ? 0 : e.length; if (!s) return -1; var l = null == n ? 0 : o(n); return l < 0 && (l = a(s + l, 0)), r(e, i(t, 3), l) } }, function (e, t, n) { var r = n(196); e.exports = function (e) { var t = r(e), n = t % 1; return t == t ? n ? t - n : t : 0 } }, function (e, t, n) { var r = n(367), i = n(23), o = n(61), a = /^[-+]0x[0-9a-f]+$/i, s = /^0b[01]+$/i, l = /^0o[0-7]+$/i, c = parseInt; e.exports = function (e) { if ("number" == typeof e) return e; if (o(e)) return NaN; if (i(e)) { var t = "function" == typeof e.valueOf ? e.valueOf() : e; e = i(t) ? t + "" : t } if ("string" != typeof e) return 0 === e ? e : +e; e = r(e); var n = s.test(e); return n || l.test(e) ? c(e.slice(2), n ? 2 : 8) : a.test(e) ? NaN : +e } }, function (e, t, n) { var r = n(368), i = /^\s+/; e.exports = function (e) { return e ? e.slice(0, r(e) + 1).replace(i, "") : e } }, function (e, t) { var n = /\s/; e.exports = function (e) { for (var t = e.length; t-- && n.test(e.charAt(t));); return t } }, function (e, t, n) { var r = n(128), i = n(169), o = n(48); e.exports = function (e, t) { return null == e ? e : r(e, i(t), o) } }, function (e, t) { e.exports = function (e) { var t = null == e ? 0 : e.length; return t ? e[t - 1] : void 0 } }, function (e, t, n) { var r = n(79), i = n(127), o = n(37); e.exports = function (e, t) { var n = {}; return t = o(t, 3), i(e, (function (e, i, o) { r(n, i, t(e, i, o)) })), n } }, function (e, t, n) { var r = n(132), i = n(373), o = n(49); e.exports = function (e) { return e && e.length ? r(e, o, i) : void 0 } }, function (e, t) { e.exports = function (e, t) { return e > t } }, function (e, t, n) { var r = n(375), i = n(379)((function (e, t, n) { r(e, t, n) })); e.exports = i }, function (e, t, n) { var r = n(73), i = n(198), o = n(128), a = n(376), s = n(23), l = n(48), c = n(199); e.exports = function e(t, n, u, d, p) { t !== n && o(n, (function (o, l) { if (p || (p = new r), s(o)) a(t, n, l, u, e, d, p); else { var f = d ? d(c(t, l), o, l + "", t, n, p) : void 0; void 0 === f && (f = o), i(t, l, f) } }), l) } }, function (e, t, n) { var r = n(198), i = n(155), o = n(164), a = n(156), s = n(165), l = n(66), c = n(13), u = n(189), d = n(59), p = n(64), f = n(23), h = n(377), g = n(67), m = n(199), v = n(378); e.exports = function (e, t, n, b, y, x, w) { var k = m(e, n), A = m(t, n), E = w.get(A); if (E) r(e, n, E); else { var S = x ? x(k, A, n + "", e, t, w) : void 0, $ = void 0 === S; if ($) { var C = c(A), _ = !C && d(A), O = !C && !_ && g(A); S = A, C || _ || O ? c(k) ? S = k : u(k) ? S = a(k) : _ ? ($ = !1, S = i(A, !0)) : O ? ($ = !1, S = o(A, !0)) : S = [] : h(A) || l(A) ? (S = k, l(k) ? S = v(k) : f(k) && !p(k) || (S = s(A))) : $ = !1 } $ && (w.set(A, S), y(S, A, b, x, w), w.delete(A)), r(e, n, S) } } }, function (e, t, n) { var r = n(47), i = n(84), o = n(34), a = Function.prototype, s = Object.prototype, l = a.toString, c = s.hasOwnProperty, u = l.call(Object); e.exports = function (e) { if (!o(e) || "[object Object]" != r(e)) return !1; var t = i(e); if (null === t) return !0; var n = c.call(t, "constructor") && t.constructor; return "function" == typeof n && n instanceof n && l.call(n) == u } }, function (e, t, n) { var r = n(65), i = n(48); e.exports = function (e) { return r(e, i(e)) } }, function (e, t, n) { var r = n(89), i = n(90); e.exports = function (e) { return r((function (t, n) { var r = -1, o = n.length, a = o > 1 ? n[o - 1] : void 0, s = o > 2 ? n[2] : void 0; for (a = e.length > 3 && "function" == typeof a ? (o--, a) : void 0, s && i(n[0], n[1], s) && (a = o < 3 ? void 0 : a, o = 1), t = Object(t); ++r < o;) { var l = n[r]; l && e(t, l, r, a) } return t })) } }, function (e, t, n) { var r = n(132), i = n(200), o = n(49); e.exports = function (e) { return e && e.length ? r(e, o, i) : void 0 } }, function (e, t, n) { var r = n(132), i = n(37), o = n(200); e.exports = function (e, t) { return e && e.length ? r(e, i(t, 2), o) : void 0 } }, function (e, t, n) { var r = n(31); e.exports = function () { return r.Date.now() } }, function (e, t, n) { var r = n(384), i = n(387)((function (e, t) { return null == e ? {} : r(e, t) })); e.exports = i }, function (e, t, n) { var r = n(385), i = n(178); e.exports = function (e, t) { return r(e, t, (function (t, n) { return i(e, n) })) } }, function (e, t, n) { var r = n(86), i = n(386), o = n(87); e.exports = function (e, t, n) { for (var a = -1, s = t.length, l = {}; ++a < s;) { var c = t[a], u = r(e, c); n(u, c) && i(l, o(c, e), u) } return l } }, function (e, t, n) { var r = n(78), i = n(87), o = n(81), a = n(23), s = n(68); e.exports = function (e, t, n, l) { if (!a(e)) return e; for (var c = -1, u = (t = i(t, e)).length, d = u - 1, p = e; null != p && ++c < u;) { var f = s(t[c]), h = n; if ("__proto__" === f || "constructor" === f || "prototype" === f) return e; if (c != d) { var g = p[f]; void 0 === (h = l ? l(g, f, p) : void 0) && (h = a(g) ? g : o(t[c + 1]) ? [] : {}) } r(p, f, h), p = p[f] } return e } }, function (e, t, n) { var r = n(197), i = n(186), o = n(187); e.exports = function (e) { return o(i(e, void 0, r), e + "") } }, function (e, t, n) { var r = n(389)(); e.exports = r }, function (e, t, n) { var r = n(390), i = n(90), o = n(196); e.exports = function (e) { return function (t, n, a) { return a && "number" != typeof a && i(t, n, a) && (n = a = void 0), t = o(t), void 0 === n ? (n = t, t = 0) : n = o(n), a = void 0 === a ? t < n ? 1 : -1 : o(a), r(t, n, a, e) } } }, function (e, t) { var n = Math.ceil, r = Math.max; e.exports = function (e, t, i, o) { for (var a = -1, s = r(n((t - e) / (i || 1)), 0), l = Array(s); s--;)l[o ? s : ++a] = e, e += i; return l } }, function (e, t, n) { var r = n(131), i = n(392), o = n(89), a = n(90), s = o((function (e, t) { if (null == e) return []; var n = t.length; return n > 1 && a(e, t[0], t[1]) ? t = [] : n > 2 && a(t[0], t[1], t[2]) && (t = [t[0]]), i(e, r(t, 1), []) })); e.exports = s }, function (e, t, n) { var r = n(88), i = n(86), o = n(37), a = n(184), s = n(393), l = n(82), c = n(394), u = n(49), d = n(13); e.exports = function (e, t, n) { t = t.length ? r(t, (function (e) { return d(e) ? function (t) { return i(t, 1 === e.length ? e[0] : e) } : e })) : [u]; var p = -1; t = r(t, l(o)); var f = a(e, (function (e, n, i) { return { criteria: r(t, (function (t) { return t(e) })), index: ++p, value: e } })); return s(f, (function (e, t) { return c(e, t, n) })) } }, function (e, t) { e.exports = function (e, t) { var n = e.length; for (e.sort(t); n--;)e[n] = e[n].value; return e } }, function (e, t, n) { var r = n(395); e.exports = function (e, t, n) { for (var i = -1, o = e.criteria, a = t.criteria, s = o.length, l = n.length; ++i < s;) { var c = r(o[i], a[i]); if (c) return i >= l ? c : c * ("desc" == n[i] ? -1 : 1) } return e.index - t.index } }, function (e, t, n) { var r = n(61); e.exports = function (e, t) { if (e !== t) { var n = void 0 !== e, i = null === e, o = e == e, a = r(e), s = void 0 !== t, l = null === t, c = t == t, u = r(t); if (!l && !u && !a && e > t || a && s && c && !l && !u || i && s && c || !n && c || !o) return 1; if (!i && !a && !u && e < t || u && n && o && !i && !a || l && n && o || !s && o || !c) return -1 } return 0 } }, function (e, t, n) { var r = n(177), i = 0; e.exports = function (e) { var t = ++i; return r(e) + t } }, function (e, t, n) { var r = n(78), i = n(398); e.exports = function (e, t) { return i(e || [], t || [], r) } }, function (e, t) { e.exports = function (e, t, n) { for (var r = -1, i = e.length, o = t.length, a = {}; ++r < i;) { var s = r < o ? t[r] : void 0; n(a, e[r], s) } return a } }, function (e, t, n) { "use strict"; var r = n(11), i = n(400); e.exports = { run: function (e) { var t = "greedy" === e.graph().acyclicer ? i(e, function (e) { return function (t) { return e.edge(t).weight } }(e)) : function (e) { var t = [], n = {}, i = {}; function o(a) { r.has(i, a) || (i[a] = !0, n[a] = !0, r.forEach(e.outEdges(a), (function (e) { r.has(n, e.w) ? t.push(e) : o(e.w) })), delete n[a]) } return r.forEach(e.nodes(), o), t }(e); r.forEach(t, (function (t) { var n = e.edge(t); e.removeEdge(t), n.forwardName = t.name, n.reversed = !0, e.setEdge(t.w, t.v, n, r.uniqueId("rev")) })) }, undo: function (e) { r.forEach(e.edges(), (function (t) { var n = e.edge(t); if (n.reversed) { e.removeEdge(t); var r = n.forwardName; delete n.reversed, delete n.forwardName, e.setEdge(t.w, t.v, n, r) } })) } } }, function (e, t, n) { var r = n(11), i = n(30).Graph, o = n(401); e.exports = function (e, t) { if (e.nodeCount() <= 1) return []; var n = function (e, t) { var n = new i, a = 0, s = 0; r.forEach(e.nodes(), (function (e) { n.setNode(e, { v: e, in: 0, out: 0 }) })), r.forEach(e.edges(), (function (e) { var r = n.edge(e.v, e.w) || 0, i = t(e), o = r + i; n.setEdge(e.v, e.w, o), s = Math.max(s, n.node(e.v).out += i), a = Math.max(a, n.node(e.w).in += i) })); var c = r.range(s + a + 3).map((function () { return new o })), u = a + 1; return r.forEach(n.nodes(), (function (e) { l(c, u, n.node(e)) })), { graph: n, buckets: c, zeroIdx: u } }(e, t || a), c = function (e, t, n) { var r, i = [], o = t[t.length - 1], a = t[0]; for (; e.nodeCount();) { for (; r = a.dequeue();)s(e, t, n, r); for (; r = o.dequeue();)s(e, t, n, r); if (e.nodeCount()) for (var l = t.length - 2; l > 0; --l)if (r = t[l].dequeue()) { i = i.concat(s(e, t, n, r, !0)); break } } return i }(n.graph, n.buckets, n.zeroIdx); return r.flatten(r.map(c, (function (t) { return e.outEdges(t.v, t.w) })), !0) }; var a = r.constant(1); function s(e, t, n, i, o) { var a = o ? [] : void 0; return r.forEach(e.inEdges(i.v), (function (r) { var i = e.edge(r), s = e.node(r.v); o && a.push({ v: r.v, w: r.w }), s.out -= i, l(t, n, s) })), r.forEach(e.outEdges(i.v), (function (r) { var i = e.edge(r), o = r.w, a = e.node(o); a.in -= i, l(t, n, a) })), e.removeNode(i.v), a } function l(e, t, n) { n.out ? n.in ? e[n.out - n.in + t].enqueue(n) : e[e.length - 1].enqueue(n) : e[0].enqueue(n) } }, function (e, t) { function n() { var e = {}; e._next = e._prev = e, this._sentinel = e } function r(e) { e._prev._next = e._next, e._next._prev = e._prev, delete e._next, delete e._prev } function i(e, t) { if ("_next" !== e && "_prev" !== e) return t } e.exports = n, n.prototype.dequeue = function () { var e = this._sentinel, t = e._prev; if (t !== e) return r(t), t }, n.prototype.enqueue = function (e) { var t = this._sentinel; e._prev && e._next && r(e), e._next = t._next, t._next._prev = e, t._next = e, e._prev = t }, n.prototype.toString = function () { for (var e = [], t = this._sentinel, n = t._prev; n !== t;)e.push(JSON.stringify(n, i)), n = n._prev; return "[" + e.join(", ") + "]" } }, function (e, t, n) { "use strict"; var r = n(11), i = n(20); e.exports = { run: function (e) { e.graph().dummyChains = [], r.forEach(e.edges(), (function (t) { !function (e, t) { var n, r, o, a = t.v, s = e.node(a).rank, l = t.w, c = e.node(l).rank, u = t.name, d = e.edge(t), p = d.labelRank; if (c === s + 1) return; for (e.removeEdge(t), o = 0, ++s; s < c; ++o, ++s)d.points = [], r = { width: 0, height: 0, edgeLabel: d, edgeObj: t, rank: s }, n = i.addDummyNode(e, "edge", r, "_d"), s === p && (r.width = d.width, r.height = d.height, r.dummy = "edge-label", r.labelpos = d.labelpos), e.setEdge(a, n, { weight: d.weight }, u), 0 === o && e.graph().dummyChains.push(n), a = n; e.setEdge(a, l, { weight: d.weight }, u) }(e, t) })) }, undo: function (e) { r.forEach(e.graph().dummyChains, (function (t) { var n, r = e.node(t), i = r.edgeLabel; for (e.setEdge(r.edgeObj, i); r.dummy;)n = e.successors(t)[0], e.removeNode(t), i.points.push({ x: r.x, y: r.y }), "edge-label" === r.dummy && (i.x = r.x, i.y = r.y, i.width = r.width, i.height = r.height), t = n, r = e.node(t) })) } } }, function (e, t, n) { "use strict"; var r = n(91).longestPath, i = n(201), o = n(404); e.exports = function (e) { switch (e.graph().ranker) { case "network-simplex": s(e); break; case "tight-tree": !function (e) { r(e), i(e) }(e); break; case "longest-path": a(e); break; default: s(e) } }; var a = r; function s(e) { o(e) } }, function (e, t, n) { "use strict"; var r = n(11), i = n(201), o = n(91).slack, a = n(91).longestPath, s = n(30).alg.preorder, l = n(30).alg.postorder, c = n(20).simplify; function u(e) { e = c(e), a(e); var t, n = i(e); for (f(n), d(n, e); t = g(n);)v(n, e, t, m(n, e, t)) } function d(e, t) { var n = l(e, e.nodes()); n = n.slice(0, n.length - 1), r.forEach(n, (function (n) { !function (e, t, n) { var r = e.node(n).parent; e.edge(n, r).cutvalue = p(e, t, n) }(e, t, n) })) } function p(e, t, n) { var i = e.node(n).parent, o = !0, a = t.edge(n, i), s = 0; return a || (o = !1, a = t.edge(i, n)), s = a.weight, r.forEach(t.nodeEdges(n), (function (r) { var a, l, c = r.v === n, u = c ? r.w : r.v; if (u !== i) { var d = c === o, p = t.edge(r).weight; if (s += d ? p : -p, a = n, l = u, e.hasEdge(a, l)) { var f = e.edge(n, u).cutvalue; s += d ? -f : f } } })), s } function f(e, t) { arguments.length < 2 && (t = e.nodes()[0]), h(e, {}, 1, t) } function h(e, t, n, i, o) { var a = n, s = e.node(i); return t[i] = !0, r.forEach(e.neighbors(i), (function (o) { r.has(t, o) || (n = h(e, t, n, o, i)) })), s.low = a, s.lim = n++, o ? s.parent = o : delete s.parent, n } function g(e) { return r.find(e.edges(), (function (t) { return e.edge(t).cutvalue < 0 })) } function m(e, t, n) { var i = n.v, a = n.w; t.hasEdge(i, a) || (i = n.w, a = n.v); var s = e.node(i), l = e.node(a), c = s, u = !1; s.lim > l.lim && (c = l, u = !0); var d = r.filter(t.edges(), (function (t) { return u === b(e, e.node(t.v), c) && u !== b(e, e.node(t.w), c) })); return r.minBy(d, (function (e) { return o(t, e) })) } function v(e, t, n, i) { var o = n.v, a = n.w; e.removeEdge(o, a), e.setEdge(i.v, i.w, {}), f(e), d(e, t), function (e, t) { var n = r.find(e.nodes(), (function (e) { return !t.node(e).parent })), i = s(e, n); i = i.slice(1), r.forEach(i, (function (n) { var r = e.node(n).parent, i = t.edge(n, r), o = !1; i || (i = t.edge(r, n), o = !0), t.node(n).rank = t.node(r).rank + (o ? i.minlen : -i.minlen) })) }(e, t) } function b(e, t, n) { return n.low <= t.lim && t.lim <= n.lim } e.exports = u, u.initLowLimValues = f, u.initCutValues = d, u.calcCutValue = p, u.leaveEdge = g, u.enterEdge = m, u.exchangeEdges = v }, function (e, t, n) { var r = n(11); e.exports = function (e) { var t = function (e) { var t = {}, n = 0; function i(o) { var a = n; r.forEach(e.children(o), i), t[o] = { low: a, lim: n++ } } return r.forEach(e.children(), i), t }(e); r.forEach(e.graph().dummyChains, (function (n) { for (var r = e.node(n), i = r.edgeObj, o = function (e, t, n, r) { var i, o, a = [], s = [], l = Math.min(t[n].low, t[r].low), c = Math.max(t[n].lim, t[r].lim); i = n; do { i = e.parent(i), a.push(i) } while (i && (t[i].low > l || c > t[i].lim)); o = i, i = r; for (; (i = e.parent(i)) !== o;)s.push(i); return { path: a.concat(s.reverse()), lca: o } }(e, t, i.v, i.w), a = o.path, s = o.lca, l = 0, c = a[l], u = !0; n !== i.w;) { if (r = e.node(n), u) { for (; (c = a[l]) !== s && e.node(c).maxRank < r.rank;)l++; c === s && (u = !1) } if (!u) { for (; l < a.length - 1 && e.node(c = a[l + 1]).minRank <= r.rank;)l++; c = a[l] } e.setParent(n, c), n = e.successors(n)[0] } })) } }, function (e, t, n) { var r = n(11), i = n(20); e.exports = { run: function (e) { var t = i.addDummyNode(e, "root", {}, "_root"), n = function (e) { var t = {}; return r.forEach(e.children(), (function (n) { !function n(i, o) { var a = e.children(i); a && a.length && r.forEach(a, (function (e) { n(e, o + 1) })); t[i] = o }(n, 1) })), t }(e), o = r.max(r.values(n)) - 1, a = 2 * o + 1; e.graph().nestingRoot = t, r.forEach(e.edges(), (function (t) { e.edge(t).minlen *= a })); var s = function (e) { return r.reduce(e.edges(), (function (t, n) { return t + e.edge(n).weight }), 0) }(e) + 1; r.forEach(e.children(), (function (l) { !function e(t, n, o, a, s, l, c) { var u = t.children(c); if (!u.length) return void (c !== n && t.setEdge(n, c, { weight: 0, minlen: o })); var d = i.addBorderNode(t, "_bt"), p = i.addBorderNode(t, "_bb"), f = t.node(c); t.setParent(d, c), f.borderTop = d, t.setParent(p, c), f.borderBottom = p, r.forEach(u, (function (r) { e(t, n, o, a, s, l, r); var i = t.node(r), u = i.borderTop ? i.borderTop : r, f = i.borderBottom ? i.borderBottom : r, h = i.borderTop ? a : 2 * a, g = u !== f ? 1 : s - l[c] + 1; t.setEdge(d, u, { weight: h, minlen: g, nestingEdge: !0 }), t.setEdge(f, p, { weight: h, minlen: g, nestingEdge: !0 }) })), t.parent(c) || t.setEdge(n, d, { weight: 0, minlen: s + l[c] }) }(e, t, a, s, o, n, l) })), e.graph().nodeRankFactor = a }, cleanup: function (e) { var t = e.graph(); e.removeNode(t.nestingRoot), delete t.nestingRoot, r.forEach(e.edges(), (function (t) { e.edge(t).nestingEdge && e.removeEdge(t) })) } } }, function (e, t, n) { var r = n(11), i = n(20); function o(e, t, n, r, o, a) { var s = { width: 0, height: 0, rank: a, borderType: t }, l = o[t][a - 1], c = i.addDummyNode(e, "border", s, n); o[t][a] = c, e.setParent(c, r), l && e.setEdge(l, c, { weight: 1 }) } e.exports = function (e) { r.forEach(e.children(), (function t(n) { var i = e.children(n), a = e.node(n); if (i.length && r.forEach(i, t), r.has(a, "minRank")) { a.borderLeft = [], a.borderRight = []; for (var s = a.minRank, l = a.maxRank + 1; s < l; ++s)o(e, "borderLeft", "_bl", n, a, s), o(e, "borderRight", "_br", n, a, s) } })) } }, function (e, t, n) { "use strict"; var r = n(11); function i(e) { r.forEach(e.nodes(), (function (t) { o(e.node(t)) })), r.forEach(e.edges(), (function (t) { o(e.edge(t)) })) } function o(e) { var t = e.width; e.width = e.height, e.height = t } function a(e) { e.y = -e.y } function s(e) { var t = e.x; e.x = e.y, e.y = t } e.exports = { adjust: function (e) { var t = e.graph().rankdir.toLowerCase(); "lr" !== t && "rl" !== t || i(e) }, undo: function (e) { var t = e.graph().rankdir.toLowerCase(); "bt" !== t && "rl" !== t || function (e) { r.forEach(e.nodes(), (function (t) { a(e.node(t)) })), r.forEach(e.edges(), (function (t) { var n = e.edge(t); r.forEach(n.points, a), r.has(n, "y") && a(n) })) }(e); "lr" !== t && "rl" !== t || (!function (e) { r.forEach(e.nodes(), (function (t) { s(e.node(t)) })), r.forEach(e.edges(), (function (t) { var n = e.edge(t); r.forEach(n.points, s), r.has(n, "x") && s(n) })) }(e), i(e)) } } }, function (e, t, n) { "use strict"; var r = n(11), i = n(410), o = n(411), a = n(412), s = n(416), l = n(417), c = n(30).Graph, u = n(20); function d(e, t, n) { return r.map(t, (function (t) { return s(e, t, n) })) } function p(e, t) { var n = new c; r.forEach(e, (function (e) { var i = e.graph().root, o = a(e, i, n, t); r.forEach(o.vs, (function (t, n) { e.node(t).order = n })), l(e, n, o.vs) })) } function f(e, t) { r.forEach(t, (function (t) { r.forEach(t, (function (t, n) { e.node(t).order = n })) })) } e.exports = function (e) { var t = u.maxRank(e), n = d(e, r.range(1, t + 1), "inEdges"), a = d(e, r.range(t - 1, -1, -1), "outEdges"), s = i(e); f(e, s); for (var l, c = Number.POSITIVE_INFINITY, h = 0, g = 0; g < 4; ++h, ++g) { p(h % 2 ? n : a, h % 4 >= 2), s = u.buildLayerMatrix(e); var m = o(e, s); m < c && (g = 0, l = r.cloneDeep(s), c = m) } f(e, l) } }, function (e, t, n) { "use strict"; var r = n(11); e.exports = function (e) { var t = {}, n = r.filter(e.nodes(), (function (t) { return !e.children(t).length })), i = r.max(r.map(n, (function (t) { return e.node(t).rank }))), o = r.map(r.range(i + 1), (function () { return [] })); var a = r.sortBy(n, (function (t) { return e.node(t).rank })); return r.forEach(a, (function n(i) { if (r.has(t, i)) return; t[i] = !0; var a = e.node(i); o[a.rank].push(i), r.forEach(e.successors(i), n) })), o } }, function (e, t, n) { "use strict"; var r = n(11); function i(e, t, n) { for (var i = r.zipObject(n, r.map(n, (function (e, t) { return t }))), o = r.flatten(r.map(t, (function (t) { return r.sortBy(r.map(e.outEdges(t), (function (t) { return { pos: i[t.w], weight: e.edge(t).weight } })), "pos") })), !0), a = 1; a < n.length;)a <<= 1; var s = 2 * a - 1; a -= 1; var l = r.map(new Array(s), (function () { return 0 })), c = 0; return r.forEach(o.forEach((function (e) { var t = e.pos + a; l[t] += e.weight; for (var n = 0; t > 0;)t % 2 && (n += l[t + 1]), l[t = t - 1 >> 1] += e.weight; c += e.weight * n }))), c } e.exports = function (e, t) { for (var n = 0, r = 1; r < t.length; ++r)n += i(e, t[r - 1], t[r]); return n } }, function (e, t, n) { var r = n(11), i = n(413), o = n(414), a = n(415); e.exports = function e(t, n, s, l) { var c = t.children(n), u = t.node(n), d = u ? u.borderLeft : void 0, p = u ? u.borderRight : void 0, f = {}; d && (c = r.filter(c, (function (e) { return e !== d && e !== p }))); var h = i(t, c); r.forEach(h, (function (n) { if (t.children(n.v).length) { var i = e(t, n.v, s, l); f[n.v] = i, r.has(i, "barycenter") && (o = n, a = i, r.isUndefined(o.barycenter) ? (o.barycenter = a.barycenter, o.weight = a.weight) : (o.barycenter = (o.barycenter * o.weight + a.barycenter * a.weight) / (o.weight + a.weight), o.weight += a.weight)) } var o, a })); var g = o(h, s); !function (e, t) { r.forEach(e, (function (e) { e.vs = r.flatten(e.vs.map((function (e) { return t[e] ? t[e].vs : e })), !0) })) }(g, f); var m = a(g, l); if (d && (m.vs = r.flatten([d, m.vs, p], !0), t.predecessors(d).length)) { var v = t.node(t.predecessors(d)[0]), b = t.node(t.predecessors(p)[0]); r.has(m, "barycenter") || (m.barycenter = 0, m.weight = 0), m.barycenter = (m.barycenter * m.weight + v.order + b.order) / (m.weight + 2), m.weight += 2 } return m } }, function (e, t, n) { var r = n(11); e.exports = function (e, t) { return r.map(t, (function (t) { var n = e.inEdges(t); if (n.length) { var i = r.reduce(n, (function (t, n) { var r = e.edge(n), i = e.node(n.v); return { sum: t.sum + r.weight * i.order, weight: t.weight + r.weight } }), { sum: 0, weight: 0 }); return { v: t, barycenter: i.sum / i.weight, weight: i.weight } } return { v: t } })) } }, function (e, t, n) { "use strict"; var r = n(11); e.exports = function (e, t) { var n = {}; return r.forEach(e, (function (e, t) { var i = n[e.v] = { indegree: 0, in: [], out: [], vs: [e.v], i: t }; r.isUndefined(e.barycenter) || (i.barycenter = e.barycenter, i.weight = e.weight) })), r.forEach(t.edges(), (function (e) { var t = n[e.v], i = n[e.w]; r.isUndefined(t) || r.isUndefined(i) || (i.indegree++, t.out.push(n[e.w])) })), function (e) { var t = []; function n(e) { return function (t) { t.merged || (r.isUndefined(t.barycenter) || r.isUndefined(e.barycenter) || t.barycenter >= e.barycenter) && function (e, t) { var n = 0, r = 0; e.weight && (n += e.barycenter * e.weight, r += e.weight); t.weight && (n += t.barycenter * t.weight, r += t.weight); e.vs = t.vs.concat(e.vs), e.barycenter = n / r, e.weight = r, e.i = Math.min(t.i, e.i), t.merged = !0 }(e, t) } } function i(t) { return function (n) { n.in.push(t), 0 == --n.indegree && e.push(n) } } for (; e.length;) { var o = e.pop(); t.push(o), r.forEach(o.in.reverse(), n(o)), r.forEach(o.out, i(o)) } return r.map(r.filter(t, (function (e) { return !e.merged })), (function (e) { return r.pick(e, ["vs", "i", "barycenter", "weight"]) })) }(r.filter(n, (function (e) { return !e.indegree }))) } }, function (e, t, n) { var r = n(11), i = n(20); function o(e, t, n) { for (var i; t.length && (i = r.last(t)).i <= n;)t.pop(), e.push(i.vs), n++; return n } e.exports = function (e, t) { var n = i.partition(e, (function (e) { return r.has(e, "barycenter") })), a = n.lhs, s = r.sortBy(n.rhs, (function (e) { return -e.i })), l = [], c = 0, u = 0, d = 0; a.sort((p = !!t, function (e, t) { return e.barycenter < t.barycenter ? -1 : e.barycenter > t.barycenter ? 1 : p ? t.i - e.i : e.i - t.i })), d = o(l, s, d), r.forEach(a, (function (e) { d += e.vs.length, l.push(e.vs), c += e.barycenter * e.weight, u += e.weight, d = o(l, s, d) })); var p; var f = { vs: r.flatten(l, !0) }; u && (f.barycenter = c / u, f.weight = u); return f } }, function (e, t, n) { var r = n(11), i = n(30).Graph; e.exports = function (e, t, n) { var o = function (e) { var t; for (; e.hasNode(t = r.uniqueId("_root"));); return t }(e), a = new i({ compound: !0 }).setGraph({ root: o }).setDefaultNodeLabel((function (t) { return e.node(t) })); return r.forEach(e.nodes(), (function (i) { var s = e.node(i), l = e.parent(i); (s.rank === t || s.minRank <= t && t <= s.maxRank) && (a.setNode(i), a.setParent(i, l || o), r.forEach(e[n](i), (function (t) { var n = t.v === i ? t.w : t.v, o = a.edge(n, i), s = r.isUndefined(o) ? 0 : o.weight; a.setEdge(n, i, { weight: e.edge(t).weight + s }) })), r.has(s, "minRank") && a.setNode(i, { borderLeft: s.borderLeft[t], borderRight: s.borderRight[t] })) })), a } }, function (e, t, n) { var r = n(11); e.exports = function (e, t, n) { var i, o = {}; r.forEach(n, (function (n) { for (var r, a, s = e.parent(n); s;) { if ((r = e.parent(s)) ? (a = o[r], o[r] = s) : (a = i, i = s), a && a !== s) return void t.setEdge(a, s); s = r } })) } }, function (e, t, n) { "use strict"; var r = n(11), i = n(20), o = n(419).positionX; e.exports = function (e) { (function (e) { var t = i.buildLayerMatrix(e), n = e.graph().ranksep, o = 0; r.forEach(t, (function (t) { var i = r.max(r.map(t, (function (t) { return e.node(t).height }))); r.forEach(t, (function (t) { e.node(t).y = o + i / 2 })), o += i + n })) })(e = i.asNonCompoundGraph(e)), r.forEach(o(e), (function (t, n) { e.node(n).x = t })) } }, function (e, t, n) { "use strict"; var r = n(11), i = n(30).Graph, o = n(20); function a(e, t) { var n = {}; return r.reduce(t, (function (t, i) { var o = 0, a = 0, s = t.length, c = r.last(i); return r.forEach(i, (function (t, u) { var d = function (e, t) { if (e.node(t).dummy) return r.find(e.predecessors(t), (function (t) { return e.node(t).dummy })) }(e, t), p = d ? e.node(d).order : s; (d || t === c) && (r.forEach(i.slice(a, u + 1), (function (t) { r.forEach(e.predecessors(t), (function (r) { var i = e.node(r), a = i.order; !(a < o || p < a) || i.dummy && e.node(t).dummy || l(n, r, t) })) })), a = u + 1, o = p) })), i })), n } function s(e, t) { var n = {}; function i(t, i, o, a, s) { var c; r.forEach(r.range(i, o), (function (i) { c = t[i], e.node(c).dummy && r.forEach(e.predecessors(c), (function (t) { var r = e.node(t); r.dummy && (r.order < a || r.order > s) && l(n, t, c) })) })) } return r.reduce(t, (function (t, n) { var o, a = -1, s = 0; return r.forEach(n, (function (r, l) { if ("border" === e.node(r).dummy) { var c = e.predecessors(r); c.length && (o = e.node(c[0]).order, i(n, s, l, a, o), s = l, a = o) } i(n, s, n.length, o, t.length) })), n })), n } function l(e, t, n) { if (t > n) { var r = t; t = n, n = r } var i = e[t]; i || (e[t] = i = {}), i[n] = !0 } function c(e, t, n) { if (t > n) { var i = t; t = n, n = i } return r.has(e[t], n) } function u(e, t, n, i) { var o = {}, a = {}, s = {}; return r.forEach(t, (function (e) { r.forEach(e, (function (e, t) { o[e] = e, a[e] = e, s[e] = t })) })), r.forEach(t, (function (e) { var t = -1; r.forEach(e, (function (e) { var l = i(e); if (l.length) for (var u = ((l = r.sortBy(l, (function (e) { return s[e] }))).length - 1) / 2, d = Math.floor(u), p = Math.ceil(u); d <= p; ++d) { var f = l[d]; a[e] === e && t < s[f] && !c(n, e, f) && (a[f] = e, a[e] = o[e] = o[f], t = s[f]) } })) })), { root: o, align: a } } function d(e, t, n, o, a) { var s = {}, l = function (e, t, n, o) { var a = new i, s = e.graph(), l = function (e, t, n) { return function (i, o, a) { var s, l = i.node(o), c = i.node(a), u = 0; if (u += l.width / 2, r.has(l, "labelpos")) switch (l.labelpos.toLowerCase()) { case "l": s = -l.width / 2; break; case "r": s = l.width / 2 }if (s && (u += n ? s : -s), s = 0, u += (l.dummy ? t : e) / 2, u += (c.dummy ? t : e) / 2, u += c.width / 2, r.has(c, "labelpos")) switch (c.labelpos.toLowerCase()) { case "l": s = c.width / 2; break; case "r": s = -c.width / 2 }return s && (u += n ? s : -s), s = 0, u } }(s.nodesep, s.edgesep, o); return r.forEach(t, (function (t) { var i; r.forEach(t, (function (t) { var r = n[t]; if (a.setNode(r), i) { var o = n[i], s = a.edge(o, r); a.setEdge(o, r, Math.max(l(e, t, i), s || 0)) } i = t })) })), a }(e, t, n, a), c = a ? "borderLeft" : "borderRight"; function u(e, t) { for (var n = l.nodes(), r = n.pop(), i = {}; r;)i[r] ? e(r) : (i[r] = !0, n.push(r), n = n.concat(t(r))), r = n.pop() } return u((function (e) { s[e] = l.inEdges(e).reduce((function (e, t) { return Math.max(e, s[t.v] + l.edge(t)) }), 0) }), l.predecessors.bind(l)), u((function (t) { var n = l.outEdges(t).reduce((function (e, t) { return Math.min(e, s[t.w] - l.edge(t)) }), Number.POSITIVE_INFINITY), r = e.node(t); n !== Number.POSITIVE_INFINITY && r.borderType !== c && (s[t] = Math.max(s[t], n)) }), l.successors.bind(l)), r.forEach(o, (function (e) { s[e] = s[n[e]] })), s } function p(e, t) { return r.minBy(r.values(t), (function (t) { var n = Number.NEGATIVE_INFINITY, i = Number.POSITIVE_INFINITY; return r.forIn(t, (function (t, r) { var o = function (e, t) { return e.node(t).width }(e, r) / 2; n = Math.max(t + o, n), i = Math.min(t - o, i) })), n - i })) } function f(e, t) { var n = r.values(t), i = r.min(n), o = r.max(n); r.forEach(["u", "d"], (function (n) { r.forEach(["l", "r"], (function (a) { var s, l = n + a, c = e[l]; if (c !== t) { var u = r.values(c); (s = "l" === a ? i - r.min(u) : o - r.max(u)) && (e[l] = r.mapValues(c, (function (e) { return e + s }))) } })) })) } function h(e, t) { return r.mapValues(e.ul, (function (n, i) { if (t) return e[t.toLowerCase()][i]; var o = r.sortBy(r.map(e, i)); return (o[1] + o[2]) / 2 })) } e.exports = { positionX: function (e) { var t, n = o.buildLayerMatrix(e), i = r.merge(a(e, n), s(e, n)), l = {}; r.forEach(["u", "d"], (function (o) { t = "u" === o ? n : r.values(n).reverse(), r.forEach(["l", "r"], (function (n) { "r" === n && (t = r.map(t, (function (e) { return r.values(e).reverse() }))); var a = ("u" === o ? e.predecessors : e.successors).bind(e), s = u(e, t, i, a), c = d(e, t, s.root, s.align, "r" === n); "r" === n && (c = r.mapValues(c, (function (e) { return -e }))), l[o + n] = c })) })); var c = p(e, l); return f(l, c), h(l, e.graph().align) }, findType1Conflicts: a, findType2Conflicts: s, addConflict: l, hasConflict: c, verticalAlignment: u, horizontalCompaction: d, alignCoordinates: f, findSmallestWidthAlignment: p, balance: h } }, function (e, t, n) { var r = n(11), i = n(20), o = n(30).Graph; e.exports = { debugOrdering: function (e) { var t = i.buildLayerMatrix(e), n = new o({ compound: !0, multigraph: !0 }).setGraph({}); return r.forEach(e.nodes(), (function (t) { n.setNode(t, { label: t }), n.setParent(t, "layer" + e.node(t).rank) })), r.forEach(e.edges(), (function (e) { n.setEdge(e.v, e.w, {}, e.name) })), r.forEach(t, (function (e, t) { var i = "layer" + t; n.setNode(i, { rank: "same" }), r.reduce(e, (function (e, t) { return n.setEdge(e, t, { style: "invis" }), t })) })), n } } }, function (e, t) { e.exports = "0.8.5" }, function (e, t, n) { "use strict"; const r = n(423); angular.module("dbt").directive("modelTree", [function () { return { scope: { tree: "=" }, templateUrl: r, link: function (e) { e.nav_selected = "project" } } }]) }, function (e, t) { var n = "/components/model_tree/model_tree.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '\n\n\n') }]), e.exports = n }, function (e, t, n) { "use strict"; const r = n(425), i = n(21); n(426), angular.module("dbt").directive("modelTreeLine", ["$state", function (e) { return { scope: { item: "=", depth: "<", resourceType: "@" }, replace: !0, templateUrl: r, link: function (t, n, r, o) { t.depth || (t.depth = 0); var a = t.item.name; if (a) { var s = i.last(a, 15).join(""), l = i.initial(a, s.length).join(""); t.name = { name: a, start: l, end: s }, t.name_start = l, t.name_end = s, t.onFolderClick = function (n) { if (n.active = !n.active, "source" == t.resourceType) { var r = n.name; e.go("dbt.source_list", { source: r }) } else 0 === t.depth && "database" !== n.type && e.go("dbt.project_overview", { project_name: n.name }) }, t.activate = function (n) { t.$emit("clearSearch"), n.active = !0; var r = "dbt." + n.node.resource_type; e.go(r, { unique_id: n.unique_id }) }, t.getIcon = function (e, t) { return "#" + { header: { on: "icn-down", off: "icn-right" }, database: { on: "icn-db-on", off: "icn-db" }, schema: { on: "icn-tree-on", off: "icn-tree" }, table: { on: "icn-doc-on", off: "icn-doc" }, folder: { on: "icn-dir-on", off: "icn-dir" }, file: { on: "icn-doc-on", off: "icn-doc" }, group: { on: "icn-filter", off: "icn-filter" } }[e][t] }, t.getClass = function (e) { return { active: e.active, "menu-tree": "header" == e.type || "schema" == e.type || "folder" == e.type, "menu-main": "header" == e.type, "menu-node": "file" == e.type || "table" == e.type } } } } } }]) }, function (e, t) { var n = "/components/model_tree/model_tree_line.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '
  • \n\n \n \n \n \n \n \n {{name.start}}\n {{name.end}}\n \n \n\n \n \n \n \n \n \n {{name.start}}\n {{name.end}}\n \n \n\n
      \n \n
    \n
  • \n') }]), e.exports = n }, function (e, t, n) { var r = n(427); "string" == typeof r && (r = [[e.i, r, ""]]); var i = { hmr: !0, transform: void 0, insertInto: void 0 }; n(40)(r, i); r.locals && (e.exports = r.locals) }, function (e, t, n) { (e.exports = n(39)(!1)).push([e.i, "\n.unselectable{\n -webkit-user-select: none;\n -moz-user-select: none;\n -ms-user-select: none;\n user-select: none;\n}\n", ""]) }, function (e, t, n) { "use strict"; const r = n(8), i = n(429); n(33); n(206), r.module("dbt").directive("docsSearch", ["$sce", "project", function (e, t) { return { scope: { query: "=", results: "=", onSelect: "&" }, replace: !0, templateUrl: i, link: function (n) { n.max_results = 20, n.show_all = !1, n.max_results_columns = 3, n.limit_columns = {}, n.checkboxStatus = { show_names: !1, show_descriptions: !1, show_columns: !1, show_column_descriptions: !1, show_code: !1, show_tags: !1 }, n.limit_search = function (e, t, r) { return t < n.max_results || n.show_all }, n.getState = function (e) { return "dbt." + e.resource_type }, n.getModelName = function (e) { return "source" == e.resource_type ? e.source_name + "." + e.name : "macro" == e.resource_type ? e.package_name + "." + e.name : "metric" == e.resource_type ? e.label : "semantic_model" == e.resource_type || "saved_query" == e.resource_type ? e.name : "exposure" == e.resource_type || "model" == e.resource_type && null != e.version ? e.label : e.name }; function r(e) { return e.replace(/[.*+\-?^${}()|[\]\\]/g, "\\$&") } function i(e) { return _.words(e.toLowerCase()) } n.$watchGroup(["query", "checkboxStatus.show_names", "checkboxStatus.show_descriptions", "checkboxStatus.show_columns", "checkboxStatus.show_column_descriptions", "checkboxStatus.show_code", "checkboxStatus.show_tags"], (function () { n.results = function (e, t) { if (!_.some(_.values(t))) return e; let n = [], r = []; const { show_names: i, show_descriptions: o, show_columns: a, show_column_descriptions: s, show_code: l, show_tags: c } = t; return _.each(e, (function (e) { _.each(e.matches, (function (t) { if (!r.includes(e.model.unique_id)) { const u = i && ("name" === t.key || "label" == t.key), d = o && "description" == t.key, p = a && "columns" === t.key, f = s && "column_description" === t.key, h = l && "raw_code" === t.key, g = c && "tags" === t.key; (u || d || p || f || h || g) && (r.push(e.model.unique_id), n.push(e)) } })) })), n }(t.search(n.query), n.checkboxStatus) })), n.shorten = function (e) { if (null != e && e.trim().length > 0 && null != n.query && n.query.trim().length > 0) { let t = e.replace(/\s+/g, " "), o = r(i(n.query)[0]), a = t.search(new RegExp(o)), s = a - 75 < 0 ? 0 : a - 75, l = a + 75 > t.length ? t.length : a + 75; return "..." + t.substring(s, l) + "..." } return e }, n.highlight = function (t) { if (!n.query || !t) return e.trustAsHtml(t); let o = "(" + i(n.query).map(e => r(e)).join(")|(") + ")"; return e.trustAsHtml(t.replace(new RegExp(o, "gi"), '$&')) }, n.$watch("query", (function (e, t) { 0 == e.length && (n.show_all = !1, n.limit_columns = {}) })), n.columnFilter = function (e) { var t = []; let r = i(n.query); for (var o in e) r.every(e => -1 != o.toLowerCase().indexOf(e)) && t.push(o); return t }, n.limitColumns = function (e) { return void 0 !== n.limit_columns[e] ? n.limit_columns[e] : 3 } } } }]) }, function (e, t) { var n = "/components/search/search.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '
    \n \n
    \n
    \n

    \n {{ query }}\n {{ results.length }} search results\n

    \n \n \n \n \n \n \n \n \n \n \n \n \n
    \n
    \n
    \n
    \n
    \n
    \n
    \n
    \n \n
    \n
    \n

    \n \n {{result.model.resource_type}}\n

    \n

    \n
    \n
    \n
    \n \n columns:\n \n \n \n Show {{ columnFilter(result.model.columns).length - max_results_columns }} more\n
    \n
    \n \n \n \n
    \n
    \n \n tags:\n \n \n \n
    \n
    \n Show {{ results.length - max_results }} more\n
    \n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t) { e.exports = { getQuoteChar: function (e) { var t = (e || {}).adapter_type; return ["bigquery", "spark", "databricks"].indexOf(t) >= 0 ? "`" : '"' } } }, function (e, t, n) { "use strict"; const r = n(432); n(433); const i = n(21); angular.module("dbt").directive("tableDetails", ["$sce", "$filter", "project", function (e, t, n) { return { scope: { model: "=", extras: "=", exclude: "<" }, templateUrl: r, link: function (e) { function r(e, t) { if (0 == e) return "0 bytes"; if (e < 1 && (e *= 1e6), isNaN(parseFloat(e)) || !isFinite(e)) return "-"; void 0 === t && (t = 0); var n = Math.floor(Math.log(e) / Math.log(1024)); return (e / Math.pow(1024, Math.floor(n))).toFixed(t) + " " + ["bytes", "KB", "MB", "GB", "TB", "PB"][n] } function o(e, n) { return void 0 === n && (n = 2), t("number")(100 * e, n) + "%" } function a(e, n) { return void 0 === n && (n = 0), t("number")(e, n) } function s(e) { var t, r, o = !e.metadata, a = e.metadata || {}; t = e.database ? e.database + "." : "", r = o ? void 0 : "source" == e.resource_type ? t + e.schema + "." + e.identifier : t + e.schema + "." + e.alias; const s = e.group ? function (e) { const { name: t, email: r } = n.project.groups[e].owner, i = []; if (t && i.push(t), r) { const e = i.length > 0 ? `<${r}>` : r; i.push(e) } return i.join(" ") }(`group.${e.package_name}.${e.group}`) : a.owner, l = !(!e.hasOwnProperty("config") || !e.config.hasOwnProperty("contract")) && e.config.contract.enforced; var c, u = [{ name: "Owner", value: s }, { name: "Type", value: o ? void 0 : (c = a.type, "BASE TABLE" == c ? { type: "table", name: "table" } : "LATE BINDING VIEW" == c ? { type: "view", name: "late binding view" } : { type: c.toLowerCase(), name: c.toLowerCase() }).name }, { name: "Package", value: e.package_name }, { name: "Language", value: e.language }, { name: "Relation", value: r }, { name: "Access", value: e.access }, { name: "Version", value: e.version }, { name: "Contract", value: l ? "Enforced" : "Not Enforced" }]; return i.filter(u, (function (e) { return void 0 !== e.value })) } e.details = [], e.extended = [], e.exclude = e.exclude || [], e.meta = null, e._show_expanded = !1, e.show_expanded = function (t) { return void 0 !== t && (e._show_expanded = t), e._show_expanded }, e.hasData = function (e) { return !(!e || i.isEmpty(e)) && (1 != e.length || 0 != e[0].include) }, e.$watch("model", (function (t, n) { i.property(["metadata", "type"])(t); var l, c, u, d = t.hasOwnProperty("sources") && null != t.sources[0] ? t.sources[0].source_meta : null; if (e.meta = t.meta || d, e.details = s(t), e.extended = (l = t.stats, c = { rows: a, row_count: a, num_rows: a, max_varchar: a, pct_used: o, size: r, bytes: r, num_bytes: r }, u = i.sortBy(i.values(l), "label"), i.map(u, (function (e) { var t = i.clone(e), n = c[e.id]; return n && (t.value = n(e.value), t.label = e.label.replace("Approximate", "~"), t.label = e.label.replace("Utilization", "Used")), t }))), e.extras) { var p = i.filter(e.extras, (function (e) { return void 0 !== e.value && null !== e.value })); e.details = e.details.concat(p) } e.show_extended = i.where(e.extended, { include: !0 }).length > 0 })), e.queryTag = function (t) { e.$emit("query", t) } } } }]) }, function (e, t) { var n = "/components/table_details/table_details.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '
    \n
    Details
    \n
    \n
    \n
    \n
    \n
    \n
    \n
    \n
    {{ k }}
    \n
    {{ v }}
    \n
    \n
    \n
    \n
    \n
    \n
    \n
    Tags
    \n
    \n {{ tag }} \n
    \n
    untagged
    \n
    \n
    \n
    {{ item.name }}
    \n
    {{ item.value }}
    \n
    \n
    \n
    \n
    \n
    \n
    \n
    {{ item.label }}
    \n
    {{ item.value }}
    \n
    \n
    \n
    \n
    \n
    \n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t, n) { var r = n(434); "string" == typeof r && (r = [[e.i, r, ""]]); var i = { hmr: !0, transform: void 0, insertInto: void 0 }; n(40)(r, i); r.locals && (e.exports = r.locals) }, function (e, t, n) { (e.exports = n(39)(!1)).push([e.i, "\n\n.details-content {\n table-layout: fixed;\n}\n\n.detail-body {\n white-space: nowrap;\n overflow-x: scroll;\n}\n", ""]) }, function (e, t, n) { "use strict"; const r = n(436), i = n(21); angular.module("dbt").directive("columnDetails", ["project", function (e) { return { scope: { model: "=" }, templateUrl: r, link: function (t) { t.has_test = function (e, t) { return -1 != i.pluck(e.tests, "short").indexOf(t) }, t.has_constraint = function (e, t) { return !!e.hasOwnProperty("constraints") && e.constraints.some(e => e.type === t) }, t.has_more_info = function (e) { var t = e.tests || [], n = e.description || "", r = e.meta || {}, o = e.constraints || []; return t.length || n.length || o.length || !i.isEmpty(r) }, t.toggle_column_expanded = function (e) { t.has_more_info(e) && (e.expanded = !e.expanded) }, t.getState = function (e) { return "dbt." + e.resource_type }, t.get_col_name = function (t) { return e.caseColumn(t) }, t.get_columns = function (e) { var t = i.chain(e.columns).values().sortBy("index").value(); return i.each(t, (function (e, t) { e.index = t })), t } } } }]) }, function (e, t) { var n = "/components/column_details/column_details.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '
    \n
    \n
    \n Column information is not available for this seed\n
    \n
    \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
    ColumnTypeDescriptionConstraintsData TestsMore?
    \n
    \n {{ get_col_name(column.name) }}\n
    \n
    \n {{ column.type }}

    \n
    \n {{ column.description }}\n \n \n N\n PK\n FK\n C\n +\n \n \n \n U\n N\n F\n A\n +\n \n \n \n \n \n \n \n \n \n \n
    \n
    \n
    \n
    Details
    \n
    \n
    \n
    \n
    {{ k }}
    \n
    {{ v }}
    \n
    \n
    \n
    \n
    \n\n
    \n
    Description
    \n \n
    \n\n
    \n
    Constraints
    \n
    \n
    \n
    \n
    Name
    \n
    {{ constraint.name }}
    \n
    Type
    \n
    {{ constraint.type }}
    \n
    Expression
    \n
    {{ constraint.expression }}
    \n
    \n
    \n
    \n
    \n\n
    \n
    Generic Data Tests
    \n \n
    \n
    \n
    \n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t, n) { "use strict"; const r = n(438); n(33), n(439); function i(e) { return "python" === e ? "language-python" : "language-sql" } angular.module("dbt").directive("codeBlock", ["code", "$timeout", function (e, t) { return { scope: { versions: "=", default: "<", language: "=" }, restrict: "E", templateUrl: r, link: function (n, r) { n.selected_version = n.default, n.language_class = i(n.language), n.source = null, n.setSelected = function (r) { n.selected_version = r, n.source = n.versions[r] || ""; const i = n.source.trim(); n.highlighted = e.highlight(i, n.language), t((function () { Prism.highlightAll() })) }, n.titleCase = function (e) { return e.charAt(0).toUpperCase() + e.substring(1) }, n.copied = !1, n.copy_to_clipboard = function () { e.copy_to_clipboard(n.source), n.copied = !0, setTimeout((function () { n.$apply((function () { n.copied = !1 })) }), 1e3) }, n.$watch("language", (function (e, t) { e && e != t && (n.language_class = i(e)) }), !0), n.$watch("versions", (function (e, t) { if (e) if (n.default) n.setSelected(n.default); else { var r = Object.keys(n.versions); r.length > 0 && n.setSelected(r[0]) } }), !0) } } }]) }, function (e, t) { var n = "/components/code_block/code_block.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '
    Code
    \n\n') }]), e.exports = n }, function (e, t, n) { var r = n(440); "string" == typeof r && (r = [[e.i, r, ""]]); var i = { hmr: !0, transform: void 0, insertInto: void 0 }; n(40)(r, i); r.locals && (e.exports = r.locals) }, function (e, t, n) { (e.exports = n(39)(!1)).push([e.i, "pre.code {\n border: none !important;\n overflow-y: visible !important;\n overflow-x: scroll !important;\n padding-bottom: 10px;\n}\n\npre.code code {\n font-family: Monaco, monospace !important;\n font-weight: 400 !important;\n}\n\n.line-numbers-rows {\n border: none !important;\n}\n", ""]) }, function (e, t, n) { "use strict"; const r = n(442); angular.module("dbt").directive("macroArguments", [function () { return { scope: { macro: "=" }, templateUrl: r, link: function (e) { _.each(e.macro.arguments, (function (e) { e.expanded = !1 })) } } }]) }, function (e, t) { var n = "/components/macro_arguments/index.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '\n\n
    \n
    \n
    \n Details are not available for this macro. This may be due to the fact that this macro doesn\'t have any\n arguments or that they haven\'t been documented yet.\n
    \n
    \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
    ArgumentTypeDescriptionMore?
    \n
    \n {{ arg.name }}\n
    \n
    \n {{ arg.type }}

    \n
    \n {{ arg.description }}\n \n \n \n \n \n \n \n \n \n \n \n \n \n
    \n
    \n
    \n
    Description
    \n \n
    \n
    \n
    \n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t, n) { "use strict"; const r = n(444); angular.module("dbt").directive("referenceList", ["$state", function (e) { return { scope: { references: "=", node: "=" }, restrict: "E", templateUrl: r, link: function (t) { t.selected_type = null, t.setType = function (e) { t.selected_type = e, t.nodes = t.references[t.selected_type] }, t.getNodeUrl = function (t) { var n = "dbt." + t.resource_type; return e.href(n, { unique_id: t.unique_id, "#": null }) }, t.mapResourceType = function (e) { return "model" == e ? "Models" : "seed" == e ? "Seeds" : "test" == e ? "Data Tests" : "unit_test" == e ? "Unit Tests" : "snapshot" == e ? "Snapshots" : "analysis" == e ? "Analyses" : "macro" == e ? "Macros" : "exposure" == e ? "Exposures" : "metric" == e ? "Metrics" : "semantic_model" == e ? "Semantic Models" : "saved_query" == e ? "Saved Queries" : "operation" == e ? "Operations" : "Nodes" }, t.$watch("references", (function (e) { e && _.size(e) > 0 ? (t.selected_type = _.keys(e)[0], t.has_references = !0, t.nodes = t.references[t.selected_type]) : t.has_references = !1 })) } } }]) }, function (e, t) { var n = "/components/references/index.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '
    \n
    \n No resources reference this {{ node.resource_type }}\n
    \n
    \n \n
    \n \n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t, n) { n(446), n(448), n(449), n(450), n(451), n(452), n(453), n(454), n(455), n(456), n(457), n(458) }, function (e, t, n) { "use strict"; const r = n(8), i = n(24); n(25), r.module("dbt").controller("ModelCtrl", ["$scope", "$state", "project", "code", "$anchorScroll", "$location", function (e, t, n, r, o, a) { e.model_uid = t.params.unique_id, e.tab = t.params.tab, e.project = n, e.codeService = r, e.versions = {}, e.copied = !1, e.copy_to_clipboard = function (t) { r.copy_to_clipboard(t), e.copied = !0, setTimeout((function () { e.$apply((function () { e.copied = !1 })) }), 1e3) }, e.model = {}, n.ready((function (t) { let n = t.nodes[e.model_uid]; e.model = n, e.references = i.getReferences(t, n), e.referencesLength = Object.keys(e.references).length, e.parents = i.getParents(t, n), e.parentsLength = Object.keys(e.parents).length, e.language = n.language; e.versions = { Source: e.model.raw_code, Compiled: e.model.compiled_code || "\n-- compiled code not found for this model\n" }, setTimeout((function () { o() }), 0) })) }]) }, function (e, t, n) { (e.exports = n(39)(!1)).push([e.i, "\n.nav-tabs li.nav-pull-right {\n flex: 1 0 auto;\n text-align: right;\n}\n\ntr.column-row-selected {\n\n}\n\ntd.column-expanded{\n padding: 0px !important;\n}\n\ntd.column-expanded > div {\n padding: 5px 10px;\n margin-left: 20px;\n height: 100%;\n\n border-left: 1px solid #ccc !important;\n}\n", ""]) }, function (e, t, n) { "use strict"; const r = n(8), i = n(24); n(25), r.module("dbt").controller("SourceCtrl", ["$scope", "$state", "project", "code", "$anchorScroll", "$location", function (e, t, n, r, o, a) { e.model_uid = t.params.unique_id, e.project = n, e.codeService = r, e.extra_table_fields = [], e.versions = {}, e.model = {}, n.ready((function (t) { let n = t.nodes[e.model_uid]; e.model = n, e.references = i.getReferences(t, n), e.referencesLength = Object.keys(e.references).length, e.parents = i.getParents(t, n), e.parentsLength = Object.keys(e.parents).length, e.versions = { "Sample SQL": r.generateSourceSQL(e.model) }, e.extra_table_fields = [{ name: "Loader", value: e.model.loader }, { name: "Source", value: e.model.source_name }] })) }]) }, function (e, t, n) { "use strict"; const r = n(8), i = n(24); n(25), r.module("dbt").controller("SeedCtrl", ["$scope", "$state", "project", "code", "$transitions", "$anchorScroll", "$location", function (e, t, n, r, o, a, s) { e.model_uid = t.params.unique_id, e.tab = t.params.tab, e.project = n, e.codeService = r, e.versions = {}, e.model = {}, n.ready((function (t) { let n = t.nodes[e.model_uid]; e.model = n, e.references = i.getReferences(t, n), e.referencesLength = Object.keys(e.references).length, e.parents = i.getParents(t, n), e.parentsLength = Object.keys(e.parents).length, e.versions = { "Example SQL": r.generateSourceSQL(e.model) } })) }]) }, function (e, t, n) { "use strict"; const r = n(8), i = n(24); n(25), r.module("dbt").controller("SnapshotCtrl", ["$scope", "$state", "project", "code", "$anchorScroll", "$location", function (e, t, n, r, o, a) { e.model_uid = t.params.unique_id, e.tab = t.params.tab, e.project = n, e.codeService = r, e.versions = {}, e.model = {}, n.ready((function (t) { let n = t.nodes[e.model_uid]; e.model = n, e.references = i.getReferences(t, n), e.referencesLength = Object.keys(e.references).length, e.parents = i.getParents(t, n), e.parentsLength = Object.keys(e.parents).length, e.language = n.language; e.versions = { Source: e.model.raw_code, Compiled: e.model.compiled_code || "Compiled SQL is not available for this snapshot" }, setTimeout((function () { o() }), 0) })) }]) }, function (e, t, n) { "use strict"; const r = n(8), i = n(24); n(25), r.module("dbt").controller("TestCtrl", ["$scope", "$state", "project", "code", "$anchorScroll", "$location", function (e, t, n, r, o, a) { e.model_uid = t.params.unique_id, e.tab = t.params.tab, e.project = n, e.codeService = r, e.versions = {}, e.model = {}, n.ready((function (t) { let n = t.nodes[e.model_uid]; e.model = n, e.references = i.getReferences(t, n), e.referencesLength = Object.keys(e.references).length, e.parents = i.getParents(t, n), e.parentsLength = Object.keys(e.parents).length, e.language = n.language; e.versions = { Source: e.model.raw_code, Compiled: e.model.compiled_code || "\n-- compiled code not found for this model\n" }, setTimeout((function () { o() }), 0) })) }]) }, function (e, t, n) { "use strict"; const r = n(8), i = n(21), o = n(24); n(25), r.module("dbt").controller("MacroCtrl", ["$scope", "$state", "project", "code", "$transitions", "$anchorScroll", "$location", function (e, t, n, r, a, s, l) { e.model_uid = t.params.unique_id, e.tab = t.params.tab, e.project = n, e.codeService = r, e.macro = {}, n.ready((function (t) { let n = t.macros[e.model_uid]; if (e.macro = n, e.references = o.getMacroReferences(t, n), e.referencesLength = Object.keys(e.references).length, e.parents = o.getMacroParents(t, n), e.parentsLength = Object.keys(e.parents).length, e.macro.is_adapter_macro) { var r = t.metadata.adapter_type; e.versions = n.impls, n.impls[r] ? e.default_version = r : n.impls.default ? e.default_version = "default" : e.default_version = i.keys(n.impls)[0] } else e.default_version = "Source", e.versions = { Source: e.macro.macro_sql } })) }]) }, function (e, t, n) { "use strict"; const r = n(8), i = n(24); n(25), r.module("dbt").controller("AnalysisCtrl", ["$scope", "$state", "project", "code", "$transitions", "$anchorScroll", "$location", function (e, t, n, r, o, a, s) { e.model_uid = t.params.unique_id, e.project = n, e.codeService = r, e.default_version = "Source", e.versions = { Source: "", Compiled: "" }, e.model = {}, n.ready((function (t) { let n = t.nodes[e.model_uid]; e.model = n, e.references = i.getReferences(t, n), e.referencesLength = Object.keys(e.references).length, e.parents = i.getParents(t, n), e.parentsLength = Object.keys(e.parents).length, e.language = n.language, e.versions = { Source: e.model.raw_code, Compiled: e.model.compiled_code } })) }]) }, function (e, t, n) { "use strict"; const r = n(8), i = n(24); n(25), r.module("dbt").controller("ExposureCtrl", ["$scope", "$state", "project", "code", "$anchorScroll", "$location", function (e, t, n, r, o, a) { e.model_uid = t.params.unique_id, e.project = n, e.codeService = r, e.extra_table_fields = [], e.versions = {}, e.exposure = {}, n.ready((function (t) { let n = t.nodes[e.model_uid]; var r; e.exposure = n, e.parents = i.getParents(t, n), e.parentsLength = e.parents.length, e.language = n.language, e.exposure.owner.name && e.exposure.owner.email ? r = `${e.exposure.owner.name} <${e.exposure.owner.email}>` : e.exposure.owner.name ? r = "" + e.exposure.owner.name : e.exposure.owner.email && (r = "" + e.exposure.owner.email), e.extra_table_fields = [{ name: "Maturity", value: e.exposure.maturity }, { name: "Owner", value: r }, { name: "Exposure name", value: e.exposure.name }] })) }]) }, function (e, t, n) { "use strict"; const r = n(8), i = n(24); n(25), r.module("dbt").controller("MetricCtrl", ["$scope", "$state", "project", "code", "$anchorScroll", "$location", function (e, t, n, r, o, a) { e.model_uid = t.params.unique_id, e.project = n, e.codeService = r, e.extra_table_fields = [], e.versions = {}, e.metric = {}, n.ready((function (t) { let n = t.nodes[e.model_uid]; e.metric = n, e.parents = i.getParents(t, n), e.parentsLength = e.parents.length; const r = "expression" === e.metric.type ? "Expression metric" : "Aggregate metric"; e.extra_table_fields = [{ name: "Metric Type", value: r }, { name: "Metric name", value: e.metric.name }] })) }]) }, function (e, t, n) { "use strict"; const r = n(8), i = n(24); n(25), r.module("dbt").controller("SemanticModelCtrl", ["$scope", "$state", "project", "code", "$anchorScroll", "$location", function (e, t, n, r, o, a) { e.model_uid = t.params.unique_id, e.project = n, e.codeService = r, e.extra_table_fields = [], e.versions = {}, e.semantic_model = {}, n.ready((function (t) { let n = t.nodes[e.model_uid]; e.semantic_model = n, e.parents = i.getParents(t, n), e.parentsLength = e.parents.length; const r = "expression" === e.semantic_model.type ? "Expression semantic_model" : "Aggregate semantic_model"; e.extra_table_fields = [{ name: "Semantic Model Type", value: r }, { name: "Semantic Model name", value: e.semantic_model.name }] })) }]) }, function (e, t, n) { "use strict"; const r = n(8), i = n(24); n(25), r.module("dbt").controller("SavedQueryCtrl", ["$scope", "$state", "project", "code", "$anchorScroll", "$location", function (e, t, n, r, o, a) { e.model_uid = t.params.unique_id, e.project = n, e.codeService = r, e.extra_table_fields = [], e.versions = {}, e.saved_query = {}, n.ready((function (t) { let n = t.nodes[e.model_uid]; e.saved_query = n, e.parents = i.getParents(t, n), e.parentsLength = e.parents.length; const r = "expression" === e.saved_query.type ? "Expression saved_query" : "Aggregate saved_query"; e.extra_table_fields = [{ name: "Saved Query Type", value: r }, { name: "Saved Query name", value: e.saved_query.name }] })) }]) }, function (e, t, n) { "use strict"; const r = n(8), i = n(24); n(25), r.module("dbt").controller("OperationCtrl", ["$scope", "$state", "project", "code", "$anchorScroll", "$location", function (e, t, n, r, o, a) { e.model_uid = t.params.unique_id, e.tab = t.params.tab, e.project = n, e.codeService = r, e.versions = {}, e.model = {}, n.ready((function (t) { let n = t.nodes[e.model_uid]; e.model = n, e.references = i.getReferences(t, n), e.referencesLength = Object.keys(e.references).length, e.parents = i.getParents(t, n), e.parentsLength = Object.keys(e.parents).length, e.language = n.language; e.versions = { Source: e.model.raw_code, Compiled: e.model.compiled_code || "\n-- compiled code not found for this model\n" }, setTimeout((function () { o() }), 0) })) }]) }, function (e, t, n) { "use strict"; n(8).module("dbt").controller("GraphCtrl", ["$scope", "$state", "$window", "graph", "project", "selectorService", function (e, t, n, r, i, o) { function a(e) { return e && "source" == e.resource_type ? "source:" + e.source_name + "." + e.name : e && "exposure" == e.resource_type ? "exposure:" + e.name : e && "metric" == e.resource_type ? "metric:" + e.name : e && "semantic_model" == e.resource_type ? "semantic_model:" + e.name : e && "saved_query" == e.resource_type ? "saved_query:" + e.name : e && "model" == e.resource_type && null != e.version ? e.label : e.name ? e.name : "*" } e.graph = r.graph, e.graphService = r, e.graphRendered = function (e) { r.setGraphReady(e) }, e.$watch((function () { return t.params.unique_id }), (function (e, t) { e && e != t && i.find_by_id(e, (function (e) { e && ("sidebar" == r.orientation ? r.showVerticalGraph(a(e), !1) : r.showFullGraph(a(e))) })), e || o.clearViewNode() })) }]) }, function (e, t, n) { "use strict"; const r = n(8), i = n(21), o = n(33), a = n(461); n(462), n(206), n(470), n(472), n(475), n(479), r.module("dbt").controller("MainController", ["$scope", "$route", "$state", "project", "graph", "selectorService", "trackingService", "locationService", "$transitions", function (e, t, n, r, s, l, c, u, d) { function p(t) { e.model_uid = t; var n = r.node(t); n && l.resetSelection(n) } function f(e) { e && setTimeout((function () { var t = o("*[data-nav-unique-id='" + e + "']"); t.length && t[0].scrollIntoView && t[0].scrollIntoView({ behavior: "smooth", block: "center", inline: "center" }) }), 1) } e.tree = { database: {}, project: {}, sources: {} }, e.search = { query: "", results: [], is_focused: !1 }, e.logo = a, e.model_uid = null, e.project = {}, o("body").bind("keydown", (function (e) { "t" == event.key && "INPUT" != event.target.tagName && (o("#search").focus(), event.preventDefault()) })), e.onSearchFocus = function (t, n) { e.search.is_focused = n }, e.clearSearch = function () { e.search.is_focused = !1, e.search.query = "", e.search.results = [], o("#search").blur() }, e.$on("clearSearch", (function () { e.clearSearch() })), e.$on("query", (function (t, n) { e.search.is_focused = !0, e.search.query = n })), e.onSearchKeypress = function (t) { "Escape" == t.key && (e.clearSearch(), t.preventDefault()) }, r.getModelTree(n.params.unique_id, (function (t) { e.tree.database = t.database, e.tree.project = t.project, e.tree.sources = t.sources, e.tree.exposures = t.exposures, e.tree.metrics = t.metrics, e.tree.semantic_models = t.semantic_models, e.tree.saved_queries = t.saved_queries, e.tree.groups = t.groups, setTimeout((function () { f(e.model_uid) })) })), d.onSuccess({}, (function (t, n) { var i = t.router.globals.params, o = l.getViewNode(), a = o ? o.unique_id : null, s = i.unique_id, u = !0; if (t.from().name == t.to().name && a == s && (u = !1), u && i.unique_id) { var d = r.updateSelected(i.unique_id); e.tree.database = d.database, e.tree.groups = d.groups, e.tree.project = d.project, e.tree.sources = d.sources, e.search.query = "", p(i.unique_id), setTimeout((function () { f(i.unique_id) })) } u && c.track_pageview() })), e.$watch("search.query", (function (t) { e.search.results = function (t) { if ("" === e.search.query) return t; let n = { name: 10, tags: 5, description: 3, raw_code: 2, columns: 1 }; return i.each(t, (function (t) { t.overallWeight = 0, i.each(Object.keys(n), (function (r) { if (null != t.model[r]) { let o = 0, a = t.model[r], s = e.search.query.toLowerCase(); if ("columns" === r) i.each(a, (function (e) { if (e.name) { let t = e.name.toLowerCase(), n = 0; for (; -1 != n;)n = t.indexOf(s, n), -1 != n && (o++, n++) } })); else if ("tags" === r) i.each(a, (function (e) { let t = e.toLowerCase(), n = 0; for (; -1 != n;)n = t.indexOf(s, n), -1 != n && (o++, n++) })); else { a = a.toLowerCase(); let e = 0; for (; -1 != e;)e = a.indexOf(s, e), -1 != e && (o++, e++) } t.overallWeight += o * n[r] } })) })), t }(r.search(t)) })), r.init(), r.ready((function (t) { e.project = t, e.search.results = r.search(""); var o = i.unique(i.pluck(i.values(t.nodes), "package_name")).sort(), a = [null]; i.each(t.nodes, (function (e) { var t = e.tags; a = i.union(a, t).sort() })), l.init({ packages: o, tags: a }), p(n.params.unique_id); var d = u.parseState(n.params); d.show_graph && s.ready((function () { i.assign(l.selection.dirty, d.selected); var e = l.updateSelection(); s.updateGraph(e) })); var f = t.metadata || {}; c.init({ track: f.send_anonymous_usage_stats, project_id: f.project_id }) })) }]) }, function (e, t) { e.exports = "data:image/svg+xml,%3Csvg width='242' height='90' viewBox='0 0 242 90' fill='none' xmlns='http://www.w3.org/2000/svg'%3E %3Cpath d='M240.384 74.5122L239.905 75.8589H239.728L239.249 74.5156V75.8589H238.941V74.0234H239.324L239.816 75.3872L240.309 74.0234H240.691V75.8589H240.384V74.5122ZM238.671 74.3003H238.169V75.8589H237.858V74.3003H237.352V74.0234H238.671V74.3003Z' fill='%23262A38'/%3E %3Cpath d='M154.123 13.915V75.3527H141.672V69.0868C140.37 71.2839 138.499 73.0742 136.22 74.2134C133.779 75.434 131.012 76.085 128.246 76.085C124.828 76.1664 121.41 75.1899 118.562 73.2369C115.633 71.2839 113.354 68.5986 111.889 65.425C110.262 61.7631 109.448 57.8572 109.529 53.8698C109.448 49.8825 110.262 45.9765 111.889 42.3961C113.354 39.3038 115.633 36.6185 118.481 34.7469C121.41 32.8753 124.828 31.9801 128.246 32.0615C130.931 32.0615 133.616 32.6311 135.976 33.8517C138.255 34.991 140.126 36.6999 141.428 38.8156V18.0651L154.123 13.915ZM139.15 63.2279C140.777 61.1121 141.672 58.0199 141.672 54.0326C141.672 50.0452 140.859 47.0344 139.15 44.9187C137.441 42.8029 134.755 41.5823 131.989 41.6637C129.222 41.5009 126.537 42.7215 124.746 44.8373C123.038 46.953 122.142 49.9639 122.142 53.8698C122.142 57.8572 123.038 60.9494 124.746 63.1465C126.455 65.3436 129.222 66.5642 131.989 66.4828C135.081 66.4828 137.522 65.3436 139.15 63.2279Z' fill='%23262A38'/%3E %3Cpath d='M198.635 34.6655C201.564 36.5371 203.843 39.2225 205.226 42.3147C206.853 45.8952 207.667 49.8011 207.586 53.7885C207.667 57.7758 206.853 61.7632 205.226 65.3436C203.761 68.5172 201.483 71.2026 198.553 73.1556C195.705 75.0272 192.287 76.0037 188.87 75.9223C186.103 76.0037 183.336 75.3527 180.895 74.0507C178.617 72.9114 176.745 71.1212 175.524 68.9241V75.2713H162.993V18.0651L175.606 13.915V38.9783C176.826 36.7812 178.698 34.991 180.976 33.8517C183.418 32.5498 186.103 31.8988 188.87 31.9801C192.287 31.8988 195.705 32.8753 198.635 34.6655ZM192.45 63.1465C194.159 60.9494 194.973 57.8572 194.973 53.7885C194.973 49.8825 194.159 46.8716 192.45 44.7559C190.741 42.6402 188.381 41.5823 185.289 41.5823C182.523 41.4196 179.837 42.6402 178.047 44.8373C176.338 47.0344 175.524 50.0452 175.524 53.9512C175.524 57.9386 176.338 61.0308 178.047 63.1465C179.756 65.3436 182.441 66.5642 185.289 66.4015C188.056 66.5642 190.741 65.3436 192.45 63.1465Z' fill='%23262A38'/%3E %3Cpath d='M225 42.4774V58.915C225 61.2749 225.651 62.9838 226.791 64.0416C228.093 65.1809 229.801 65.7505 231.592 65.6691C232.975 65.6691 234.44 65.425 235.742 65.0995V74.8644C233.382 75.6782 230.941 76.085 228.499 76.0037C223.292 76.0037 219.304 74.5389 216.537 71.6094C213.771 68.68 212.387 64.5299 212.387 59.1592V23.1103L225 19.0416V33.038H235.742V42.4774H225Z' fill='%23262A38'/%3E %3Cpath d='M86.1754 3.74322C88.2911 5.77758 89.6745 8.46293 90 11.3924C90 12.613 89.6745 13.4268 88.9421 14.9729C88.2098 16.519 79.1772 32.1429 76.4919 36.4557C74.9458 38.9783 74.132 41.9892 74.132 44.9186C74.132 47.9295 74.9458 50.859 76.4919 53.3816C79.1772 57.6944 88.2098 73.3996 88.9421 74.9457C89.6745 76.4919 90 77.2242 90 78.4448C89.6745 81.3743 88.3725 84.0597 86.2568 86.0127C84.2224 88.1284 81.5371 89.5118 78.689 89.7559C77.4684 89.7559 76.6546 89.4304 75.1899 88.698C73.7251 87.9656 57.7758 79.1772 53.4629 76.4919C53.1374 76.3291 52.8119 76.085 52.4051 75.9222L31.085 63.3092C31.5732 67.3779 33.3635 71.2839 36.2929 74.132C36.8626 74.7016 37.4322 75.1899 38.0832 75.6781C37.5949 75.9222 37.0253 76.1664 36.5371 76.4919C32.2242 79.1772 16.519 88.2098 14.9729 88.9421C13.4268 89.6745 12.6944 90 11.3924 90C8.46293 89.6745 5.77758 88.3725 3.82459 86.2568C1.70886 84.2224 0.325497 81.5371 0 78.6076C0.0813743 77.387 0.406872 76.1664 1.05787 75.1085C1.79024 73.5624 10.8228 57.8571 13.5081 53.5443C15.0542 51.0217 15.868 48.0922 15.868 45.0814C15.868 42.0705 15.0542 39.141 13.5081 36.6184C10.8228 32.1429 1.70886 16.4376 1.05787 14.8915C0.406872 13.8336 0.0813743 12.613 0 11.3924C0.325497 8.46293 1.62749 5.77758 3.74322 3.74322C5.77758 1.62749 8.46293 0.325497 11.3924 0C12.613 0.0813743 13.8336 0.406872 14.9729 1.05787C16.2749 1.62749 27.7486 8.30018 33.8517 11.8807L35.2351 12.6944C35.7233 13.0199 36.1302 13.264 36.4557 13.4268L37.1067 13.8336L58.8336 26.6908C58.3454 21.8083 55.8228 17.3327 51.9168 14.3219C52.4051 14.0778 52.9747 13.8336 53.4629 13.5081C57.7758 10.8228 73.481 1.70886 75.0271 1.05787C76.085 0.406872 77.3056 0.0813743 78.6076 0C81.4557 0.325497 84.1411 1.62749 86.1754 3.74322ZM46.1392 50.7776L50.7776 46.1392C51.4286 45.4882 51.4286 44.5118 50.7776 43.8608L46.1392 39.2224C45.4882 38.5714 44.5118 38.5714 43.8608 39.2224L39.2224 43.8608C38.5714 44.5118 38.5714 45.4882 39.2224 46.1392L43.8608 50.7776C44.4304 51.3472 45.4882 51.3472 46.1392 50.7776Z' fill='%23FF694A'/%3E %3C/svg%3E" }, function (e, t, n) { "use strict"; n.r(t); var r = n(63), i = n.n(r); n(463), n(464), n(465), n(466), n(468); const o = n(8), a = (n(33), n(21)); window.Prism = i.a, o.module("dbt").factory("code", ["$sce", function (e) { var t = { copied: !1, highlight: function (t, n = "sql") { if ("sql" == n) var r = i.a.highlight(t, i.a.languages.sql, "sql"); else if ("python" == n) r = i.a.highlight(t, i.a.languages.python, "python"); return e.trustAsHtml(r) }, copy_to_clipboard: function (e) { var t = document.createElement("textarea"); t.value = e, t.setAttribute("readonly", ""), t.style.position = "absolute", t.style.left = "-9999px", document.body.appendChild(t), t.select(), document.execCommand("copy"), document.body.removeChild(t) }, generateSourceSQL: function (e) { var t = ["select"], n = a.size(e.columns), r = a.keys(e.columns); a.each(r, (function (e, r) { var i = " " + e; r + 1 != n && (i += ","), t.push(i) })); const i = (e.database ? e.database + "." : "") + e.schema + "." + e.identifier; return t.push("from " + i), t.join("\n") } }; return t }]) }, function (e, t) { Prism.languages.sql = { comment: { pattern: /(^|[^\\])(?:\/\*[\s\S]*?\*\/|(?:--|\/\/|#).*)/, lookbehind: !0 }, variable: [{ pattern: /@(["'`])(?:\\[\s\S]|(?!\1)[^\\])+\1/, greedy: !0 }, /@[\w.$]+/], string: { pattern: /(^|[^@\\])("|')(?:\\[\s\S]|(?!\2)[^\\]|\2\2)*\2/, greedy: !0, lookbehind: !0 }, identifier: { pattern: /(^|[^@\\])`(?:\\[\s\S]|[^`\\]|``)*`/, greedy: !0, lookbehind: !0, inside: { punctuation: /^`|`$/ } }, function: /\b(?:AVG|COUNT|FIRST|FORMAT|LAST|LCASE|LEN|MAX|MID|MIN|MOD|NOW|ROUND|SUM|UCASE)(?=\s*\()/i, keyword: /\b(?:ACTION|ADD|AFTER|ALGORITHM|ALL|ALTER|ANALYZE|ANY|APPLY|AS|ASC|AUTHORIZATION|AUTO_INCREMENT|BACKUP|BDB|BEGIN|BERKELEYDB|BIGINT|BINARY|BIT|BLOB|BOOL|BOOLEAN|BREAK|BROWSE|BTREE|BULK|BY|CALL|CASCADED?|CASE|CHAIN|CHAR(?:ACTER|SET)?|CHECK(?:POINT)?|CLOSE|CLUSTERED|COALESCE|COLLATE|COLUMNS?|COMMENT|COMMIT(?:TED)?|COMPUTE|CONNECT|CONSISTENT|CONSTRAINT|CONTAINS(?:TABLE)?|CONTINUE|CONVERT|CREATE|CROSS|CURRENT(?:_DATE|_TIME|_TIMESTAMP|_USER)?|CURSOR|CYCLE|DATA(?:BASES?)?|DATE(?:TIME)?|DAY|DBCC|DEALLOCATE|DEC|DECIMAL|DECLARE|DEFAULT|DEFINER|DELAYED|DELETE|DELIMITERS?|DENY|DESC|DESCRIBE|DETERMINISTIC|DISABLE|DISCARD|DISK|DISTINCT|DISTINCTROW|DISTRIBUTED|DO|DOUBLE|DROP|DUMMY|DUMP(?:FILE)?|DUPLICATE|ELSE(?:IF)?|ENABLE|ENCLOSED|END|ENGINE|ENUM|ERRLVL|ERRORS|ESCAPED?|EXCEPT|EXEC(?:UTE)?|EXISTS|EXIT|EXPLAIN|EXTENDED|FETCH|FIELDS|FILE|FILLFACTOR|FIRST|FIXED|FLOAT|FOLLOWING|FOR(?: EACH ROW)?|FORCE|FOREIGN|FREETEXT(?:TABLE)?|FROM|FULL|FUNCTION|GEOMETRY(?:COLLECTION)?|GLOBAL|GOTO|GRANT|GROUP|HANDLER|HASH|HAVING|HOLDLOCK|HOUR|IDENTITY(?:COL|_INSERT)?|IF|IGNORE|IMPORT|INDEX|INFILE|INNER|INNODB|INOUT|INSERT|INT|INTEGER|INTERSECT|INTERVAL|INTO|INVOKER|ISOLATION|ITERATE|JOIN|KEYS?|KILL|LANGUAGE|LAST|LEAVE|LEFT|LEVEL|LIMIT|LINENO|LINES|LINESTRING|LOAD|LOCAL|LOCK|LONG(?:BLOB|TEXT)|LOOP|MATCH(?:ED)?|MEDIUM(?:BLOB|INT|TEXT)|MERGE|MIDDLEINT|MINUTE|MODE|MODIFIES|MODIFY|MONTH|MULTI(?:LINESTRING|POINT|POLYGON)|NATIONAL|NATURAL|NCHAR|NEXT|NO|NONCLUSTERED|NULLIF|NUMERIC|OFF?|OFFSETS?|ON|OPEN(?:DATASOURCE|QUERY|ROWSET)?|OPTIMIZE|OPTION(?:ALLY)?|ORDER|OUT(?:ER|FILE)?|OVER|PARTIAL|PARTITION|PERCENT|PIVOT|PLAN|POINT|POLYGON|PRECEDING|PRECISION|PREPARE|PREV|PRIMARY|PRINT|PRIVILEGES|PROC(?:EDURE)?|PUBLIC|PURGE|QUICK|RAISERROR|READS?|REAL|RECONFIGURE|REFERENCES|RELEASE|RENAME|REPEAT(?:ABLE)?|REPLACE|REPLICATION|REQUIRE|RESIGNAL|RESTORE|RESTRICT|RETURN(?:ING|S)?|REVOKE|RIGHT|ROLLBACK|ROUTINE|ROW(?:COUNT|GUIDCOL|S)?|RTREE|RULE|SAVE(?:POINT)?|SCHEMA|SECOND|SELECT|SERIAL(?:IZABLE)?|SESSION(?:_USER)?|SET(?:USER)?|SHARE|SHOW|SHUTDOWN|SIMPLE|SMALLINT|SNAPSHOT|SOME|SONAME|SQL|START(?:ING)?|STATISTICS|STATUS|STRIPED|SYSTEM_USER|TABLES?|TABLESPACE|TEMP(?:ORARY|TABLE)?|TERMINATED|TEXT(?:SIZE)?|THEN|TIME(?:STAMP)?|TINY(?:BLOB|INT|TEXT)|TOP?|TRAN(?:SACTIONS?)?|TRIGGER|TRUNCATE|TSEQUAL|TYPES?|UNBOUNDED|UNCOMMITTED|UNDEFINED|UNION|UNIQUE|UNLOCK|UNPIVOT|UNSIGNED|UPDATE(?:TEXT)?|USAGE|USE|USER|USING|VALUES?|VAR(?:BINARY|CHAR|CHARACTER|YING)|VIEW|WAITFOR|WARNINGS|WHEN|WHERE|WHILE|WITH(?: ROLLUP|IN)?|WORK|WRITE(?:TEXT)?|YEAR)\b/i, boolean: /\b(?:FALSE|NULL|TRUE)\b/i, number: /\b0x[\da-f]+\b|\b\d+(?:\.\d*)?|\B\.\d+\b/i, operator: /[-+*\/=%^~]|&&?|\|\|?|!=?|<(?:=>?|<|>)?|>[>=]?|\b(?:AND|BETWEEN|DIV|ILIKE|IN|IS|LIKE|NOT|OR|REGEXP|RLIKE|SOUNDS LIKE|XOR)\b/i, punctuation: /[;[\]()`,.]/ } }, function (e, t) { Prism.languages.python = { comment: { pattern: /(^|[^\\])#.*/, lookbehind: !0, greedy: !0 }, "string-interpolation": { pattern: /(?:f|fr|rf)(?:("""|''')[\s\S]*?\1|("|')(?:\\.|(?!\2)[^\\\r\n])*\2)/i, greedy: !0, inside: { interpolation: { pattern: /((?:^|[^{])(?:\{\{)*)\{(?!\{)(?:[^{}]|\{(?!\{)(?:[^{}]|\{(?!\{)(?:[^{}])+\})+\})+\}/, lookbehind: !0, inside: { "format-spec": { pattern: /(:)[^:(){}]+(?=\}$)/, lookbehind: !0 }, "conversion-option": { pattern: /![sra](?=[:}]$)/, alias: "punctuation" }, rest: null } }, string: /[\s\S]+/ } }, "triple-quoted-string": { pattern: /(?:[rub]|br|rb)?("""|''')[\s\S]*?\1/i, greedy: !0, alias: "string" }, string: { pattern: /(?:[rub]|br|rb)?("|')(?:\\.|(?!\1)[^\\\r\n])*\1/i, greedy: !0 }, function: { pattern: /((?:^|\s)def[ \t]+)[a-zA-Z_]\w*(?=\s*\()/g, lookbehind: !0 }, "class-name": { pattern: /(\bclass\s+)\w+/i, lookbehind: !0 }, decorator: { pattern: /(^[\t ]*)@\w+(?:\.\w+)*/m, lookbehind: !0, alias: ["annotation", "punctuation"], inside: { punctuation: /\./ } }, keyword: /\b(?:_(?=\s*:)|and|as|assert|async|await|break|case|class|continue|def|del|elif|else|except|exec|finally|for|from|global|if|import|in|is|lambda|match|nonlocal|not|or|pass|print|raise|return|try|while|with|yield)\b/, builtin: /\b(?:__import__|abs|all|any|apply|ascii|basestring|bin|bool|buffer|bytearray|bytes|callable|chr|classmethod|cmp|coerce|compile|complex|delattr|dict|dir|divmod|enumerate|eval|execfile|file|filter|float|format|frozenset|getattr|globals|hasattr|hash|help|hex|id|input|int|intern|isinstance|issubclass|iter|len|list|locals|long|map|max|memoryview|min|next|object|oct|open|ord|pow|property|range|raw_input|reduce|reload|repr|reversed|round|set|setattr|slice|sorted|staticmethod|str|sum|super|tuple|type|unichr|unicode|vars|xrange|zip)\b/, boolean: /\b(?:False|None|True)\b/, number: /\b0(?:b(?:_?[01])+|o(?:_?[0-7])+|x(?:_?[a-f0-9])+)\b|(?:\b\d+(?:_\d+)*(?:\.(?:\d+(?:_\d+)*)?)?|\B\.\d+(?:_\d+)*)(?:e[+-]?\d+(?:_\d+)*)?j?(?!\w)/i, operator: /[-+%=]=?|!=|:=|\*\*?=?|\/\/?=?|<[<=>]?|>[=>]?|[&|^~]/, punctuation: /[{}[\];(),.:]/ }, Prism.languages.python["string-interpolation"].inside.interpolation.inside.rest = Prism.languages.python, Prism.languages.py = Prism.languages.python }, function (e, t) { !function () { if ("undefined" != typeof Prism && "undefined" != typeof document) { var e = /\n(?!$)/g, t = Prism.plugins.lineNumbers = { getLine: function (e, t) { if ("PRE" === e.tagName && e.classList.contains("line-numbers")) { var n = e.querySelector(".line-numbers-rows"); if (n) { var r = parseInt(e.getAttribute("data-start"), 10) || 1, i = r + (n.children.length - 1); t < r && (t = r), t > i && (t = i); var o = t - r; return n.children[o] } } }, resize: function (e) { r([e]) }, assumeViewportIndependence: !0 }, n = void 0; window.addEventListener("resize", (function () { t.assumeViewportIndependence && n === window.innerWidth || (n = window.innerWidth, r(Array.prototype.slice.call(document.querySelectorAll("pre.line-numbers")))) })), Prism.hooks.add("complete", (function (t) { if (t.code) { var n = t.element, i = n.parentNode; if (i && /pre/i.test(i.nodeName) && !n.querySelector(".line-numbers-rows") && Prism.util.isActive(n, "line-numbers")) { n.classList.remove("line-numbers"), i.classList.add("line-numbers"); var o, a = t.code.match(e), s = a ? a.length + 1 : 1, l = new Array(s + 1).join(""); (o = document.createElement("span")).setAttribute("aria-hidden", "true"), o.className = "line-numbers-rows", o.innerHTML = l, i.hasAttribute("data-start") && (i.style.counterReset = "linenumber " + (parseInt(i.getAttribute("data-start"), 10) - 1)), t.element.appendChild(o), r([i]), Prism.hooks.run("line-numbers", t) } } })), Prism.hooks.add("line-numbers", (function (e) { e.plugins = e.plugins || {}, e.plugins.lineNumbers = !0 })) } function r(t) { if (0 != (t = t.filter((function (e) { var t = function (e) { if (!e) return null; return window.getComputedStyle ? getComputedStyle(e) : e.currentStyle || null }(e)["white-space"]; return "pre-wrap" === t || "pre-line" === t }))).length) { var n = t.map((function (t) { var n = t.querySelector("code"), r = t.querySelector(".line-numbers-rows"); if (n && r) { var i = t.querySelector(".line-numbers-sizer"), o = n.textContent.split(e); i || ((i = document.createElement("span")).className = "line-numbers-sizer", n.appendChild(i)), i.innerHTML = "0", i.style.display = "block"; var a = i.getBoundingClientRect().height; return i.innerHTML = "", { element: t, lines: o, lineHeights: [], oneLinerHeight: a, sizer: i } } })).filter(Boolean); n.forEach((function (e) { var t = e.sizer, n = e.lines, r = e.lineHeights, i = e.oneLinerHeight; r[n.length - 1] = void 0, n.forEach((function (e, n) { if (e && e.length > 1) { var o = t.appendChild(document.createElement("span")); o.style.display = "block", o.textContent = e } else r[n] = i })) })), n.forEach((function (e) { for (var t = e.sizer, n = e.lineHeights, r = 0, i = 0; i < n.length; i++)void 0 === n[i] && (n[i] = t.children[r++].getBoundingClientRect().height) })), n.forEach((function (e) { var t = e.sizer, n = e.element.querySelector(".line-numbers-rows"); t.style.display = "none", t.innerHTML = "", e.lineHeights.forEach((function (e, t) { n.children[t].style.height = e + "px" })) })) } } }() }, function (e, t, n) { var r = n(467); "string" == typeof r && (r = [[e.i, r, ""]]); var i = { hmr: !0, transform: void 0, insertInto: void 0 }; n(40)(r, i); r.locals && (e.exports = r.locals) }, function (e, t, n) { (e.exports = n(39)(!1)).push([e.i, 'pre[class*="language-"].line-numbers {\n\tposition: relative;\n\tpadding-left: 3.8em;\n\tcounter-reset: linenumber;\n}\n\npre[class*="language-"].line-numbers > code {\n\tposition: relative;\n\twhite-space: inherit;\n}\n\n.line-numbers .line-numbers-rows {\n\tposition: absolute;\n\tpointer-events: none;\n\ttop: 0;\n\tfont-size: 100%;\n\tleft: -3.8em;\n\twidth: 3em; /* works for line-numbers below 1000 lines */\n\tletter-spacing: -1px;\n\tborder-right: 1px solid #999;\n\n\t-webkit-user-select: none;\n\t-moz-user-select: none;\n\t-ms-user-select: none;\n\tuser-select: none;\n\n}\n\n\t.line-numbers-rows > span {\n\t\tdisplay: block;\n\t\tcounter-increment: linenumber;\n\t}\n\n\t\t.line-numbers-rows > span:before {\n\t\t\tcontent: counter(linenumber);\n\t\t\tcolor: #999;\n\t\t\tdisplay: block;\n\t\t\tpadding-right: 0.8em;\n\t\t\ttext-align: right;\n\t\t}\n', ""]) }, function (e, t, n) { var r = n(469); "string" == typeof r && (r = [[e.i, r, ""]]); var i = { hmr: !0, transform: void 0, insertInto: void 0 }; n(40)(r, i); r.locals && (e.exports = r.locals) }, function (e, t, n) { (e.exports = n(39)(!1)).push([e.i, '/**\n * GHColors theme by Avi Aryan (http://aviaryan.in)\n * Inspired by Github syntax coloring\n */\n\ncode[class*="language-"],\npre[class*="language-"] {\n\tcolor: #393A34;\n\tfont-family: "Consolas", "Bitstream Vera Sans Mono", "Courier New", Courier, monospace;\n\tdirection: ltr;\n\ttext-align: left;\n\twhite-space: pre;\n\tword-spacing: normal;\n\tword-break: normal;\n\tfont-size: .9em;\n\tline-height: 1.2em;\n\n\t-moz-tab-size: 4;\n\t-o-tab-size: 4;\n\ttab-size: 4;\n\n\t-webkit-hyphens: none;\n\t-moz-hyphens: none;\n\t-ms-hyphens: none;\n\thyphens: none;\n}\n\npre > code[class*="language-"] {\n\tfont-size: 1em;\n}\n\npre[class*="language-"]::-moz-selection, pre[class*="language-"] ::-moz-selection,\ncode[class*="language-"]::-moz-selection, code[class*="language-"] ::-moz-selection {\n\tbackground: #b3d4fc;\n}\n\npre[class*="language-"]::selection, pre[class*="language-"] ::selection,\ncode[class*="language-"]::selection, code[class*="language-"] ::selection {\n\tbackground: #b3d4fc;\n}\n\n/* Code blocks */\npre[class*="language-"] {\n\tpadding: 1em;\n\tmargin: .5em 0;\n\toverflow: auto;\n\tborder: 1px solid #dddddd;\n\tbackground-color: white;\n}\n\n/* Inline code */\n:not(pre) > code[class*="language-"] {\n\tpadding: .2em;\n\tpadding-top: 1px;\n\tpadding-bottom: 1px;\n\tbackground: #f8f8f8;\n\tborder: 1px solid #dddddd;\n}\n\n.token.comment,\n.token.prolog,\n.token.doctype,\n.token.cdata {\n\tcolor: #999988;\n\tfont-style: italic;\n}\n\n.token.namespace {\n\topacity: .7;\n}\n\n.token.string,\n.token.attr-value {\n\tcolor: #e3116c;\n}\n\n.token.punctuation,\n.token.operator {\n\tcolor: #393A34; /* no highlight */\n}\n\n.token.entity,\n.token.url,\n.token.symbol,\n.token.number,\n.token.boolean,\n.token.variable,\n.token.constant,\n.token.property,\n.token.regex,\n.token.inserted {\n\tcolor: #36acaa;\n}\n\n.token.atrule,\n.token.keyword,\n.token.attr-name,\n.language-autohotkey .token.selector {\n\tcolor: #00a4db;\n}\n\n.token.function,\n.token.deleted,\n.language-autohotkey .token.tag {\n\tcolor: #9a050f;\n}\n\n.token.tag,\n.token.selector,\n.language-autohotkey .token.keyword {\n\tcolor: #00009f;\n}\n\n.token.important,\n.token.function,\n.token.bold {\n\tfont-weight: bold;\n}\n\n.token.italic {\n\tfont-style: italic;\n}\n', ""]) }, function (e, t, n) { n(33); const r = n(21), i = n(148), o = n(203), a = n(471); angular.module("dbt").factory("graph", ["$state", "$window", "$q", "selectorService", "project", "locationService", function (e, t, n, s, l, c) { var u = { vertical: { userPanningEnabled: !1, boxSelectionEnabled: !1, maxZoom: 1.5 }, horizontal: { userPanningEnabled: !0, boxSelectionEnabled: !1, maxZoom: 1, minZoom: .05 } }, d = { none: { name: "null" }, left_right: { name: "dagre", rankDir: "LR", rankSep: 200, edgeSep: 30, nodeSep: 50 }, top_down: { name: "preset", positions: function (t) { var n = e.params.unique_id; if (!n) return { x: 0, y: 0 }; var a = p.graph.pristine.dag, s = r.sortBy(o.ancestorNodes(a, n, 1)), l = r.sortBy(o.descendentNodes(a, n, 1)), c = r.partial(r.includes, s), u = r.partial(r.includes, l), d = a.filterNodes(c), f = a.filterNodes(u); return function (e, t, n, i) { var o, a = 100 / (1 + Math.max(t.length, n.length)); if (e == i) return { x: 0, y: 0 }; if (r.includes(t, i)) o = { set: t, index: r.indexOf(t, i), factor: -1, type: "parent" }; else { if (!r.includes(n, i)) return { x: 0, y: 0 }; o = { set: n, index: r.indexOf(n, i), factor: 1, type: "child" } } var s = o.set.length; if ("parent" == o.type) var l = { x: (0 + o.index) * a, y: -200 - 100 * (s - o.index - 1) }; else l = { x: (0 + o.index) * a, y: 200 + 100 * (s - o.index - 1) }; return l }(n, i.alg.topsort(d), i.alg.topsort(f).reverse(), t.data("id")) } } }, p = { loading: !0, loaded: n.defer(), graph_element: null, orientation: "sidebar", expanded: !1, graph: { options: u.vertical, pristine: { nodes: {}, edges: {}, dag: null }, elements: [], layout: d.none, style: [{ selector: "edge.vertical", style: { "curve-style": "unbundled-bezier", "target-arrow-shape": "triangle-backcurve", "target-arrow-color": "#027599", "arrow-scale": 1.5, "line-color": "#027599", width: 3, "target-distance-from-node": "5px", "source-endpoint": "0% 50%", "target-endpoint": "0deg" } }, { selector: "edge.horizontal", style: { "curve-style": "unbundled-bezier", "target-arrow-shape": "triangle-backcurve", "target-arrow-color": "#006f8a", "arrow-scale": 1.5, "target-distance-from-node": "10px", "source-distance-from-node": "5px", "line-color": "#006f8a", width: 3, "source-endpoint": "50% 0%", "target-endpoint": "270deg" } }, { selector: "edge[selected=1]", style: { "line-color": "#bd6bb6", "target-arrow-color": "#bd6bb6", "z-index": 1 } }, { selector: 'node[display="none"]', style: { display: "none" } }, { selector: "node.vertical", style: { "text-margin-x": "5px", "background-color": "#0094b3", "border-color": "#0094b3", "font-size": "16px", shape: "ellipse", color: "#fff", width: "5px", height: "5px", padding: "5px", content: "data(label)", "font-weight": 300, "text-valign": "center", "text-halign": "right" } }, { selector: "node.horizontal", style: { "background-color": "#0094b3", "border-color": "#0094b3", "font-size": "24px", shape: "roundrectangle", color: "#fff", width: "label", height: "label", padding: "12px", content: "data(label)", "font-weight": 300, "font-family": '-apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Oxygen, Ubuntu, Cantarell, "Fira Sans", "Droid Sans", "Helvetica Neue", Helvetica, Arial, sans-serif', "text-valign": "center", "text-halign": "center", ghost: "yes", "ghost-offset-x": "2px", "ghost-offset-y": "4px", "ghost-opacity": .5, "text-outline-color": "#000", "text-outline-width": "1px", "text-outline-opacity": .2 } }, { selector: 'node[resource_type="source"]', style: { "background-color": "#5fb825", "border-color": "#5fb825" } }, { selector: 'node[resource_type="exposure"]', style: { "background-color": "#ff694b", "border-color": "#ff694b" } }, { selector: 'node[resource_type="metric"]', style: { "background-color": "#ff5688", "border-color": "#ff5688" } }, { selector: 'node[resource_type="semantic_model"]', style: { "background-color": "#ffa8c2", "border-color": "#ffa8c2" } }, { selector: 'node[resource_type="saved_query"]', style: { "background-color": "#ff7f50", "border-color": "#ff7f50" } }, { selector: 'node[language="python"]', style: { "background-color": "#6a5acd", "border-color": "#6a5acd" } }, { selector: "node[node_color]", style: { "background-color": "data(node_color)", "border-color": "data(node_color)" } }, { selector: "node[selected=1]", style: { "background-color": "#bd6bb6", "border-color": "#bd6bb6" } }, { selector: "node.horizontal[selected=1]", style: { "background-color": "#88447d", "border-color": "#88447d" } }, { selector: "node.horizontal.dirty", style: { "background-color": "#919599", "border-color": "#919599" } }, { selector: "node[hidden=1]", style: { "background-color": "#919599", "border-color": "#919599", "background-opacity": .5 } }, { selector: 'node[access="private"]', style: { "background-opacity": .2, "border-width": 2, ghost: "no" } }], ready: function (e) { console.log("graph ready") } } }; function f(e, t, n) { var i = r.map(e, (function (e) { return p.graph.pristine.nodes[e] })), o = []; r.flatten(r.each(e, (function (t) { var n = p.graph.pristine.edges[t]; r.each(n, (function (t) { r.includes(e, t.data.target) && r.includes(e, t.data.source) && o.push(t) })) }))); var s = r.compact(i).concat(r.compact(o)); return r.each(p.graph.elements, (function (e) { e.data.display = "none", e.data.selected = 0, e.data.hidden = 0, e.classes = n })), r.each(s, (function (e) { e.data.display = "element", e.classes = n, t && r.includes(t, e.data.unique_id) && (e.data.selected = 1), r.get(e, ["data", "docs", "show"], !0) || (e.data.hidden = 1); var i = r.get(e, ["data", "docs", "node_color"]); i && a.isValidColor(i) && (e.data.node_color = i) })), p.graph.elements = r.filter(s, (function (e) { return "element" == e.data.display })), e } function h(e, t, n) { var r = p.graph.pristine.dag; if (r) { var i = p.graph.pristine.nodes, o = s.selectNodes(r, i, e), a = n ? o.matched : []; return f(o.selected, a, t) } } return p.setGraphReady = function (e) { p.loading = !1, p.loaded.resolve(), p.graph_element = e }, p.ready = function (e) { p.loaded.promise.then((function () { e(p) })) }, p.manifest = {}, p.packages = [], p.selected_node = null, p.getCanvasHeight = function () { return .8 * t.innerHeight + "px" }, l.ready((function (e) { p.manifest = e, p.packages = r.uniq(r.map(p.manifest.nodes, "package_name")), r.each(r.filter(p.manifest.nodes, (function (e) { var t = r.includes(["model", "seed", "source", "snapshot", "analysis", "exposure", "metric", "semantic_model", "operation", "saved_query"], e.resource_type), n = "test" === e.resource_type && !e.hasOwnProperty("test_metadata"), i = "unit_test" === e.resource_type; return t || n || i })), (function (e) { var t = { group: "nodes", data: r.assign(e, { parent: e.package_name, id: e.unique_id, is_group: "false" }) }; p.graph.pristine.nodes[e.unique_id] = t })), r.each(p.manifest.parent_map, (function (e, t) { r.each(e, (function (e) { var n = p.manifest.nodes[e], i = p.manifest.nodes[t]; if (r.includes(["model", "source", "seed", "snapshot", "metric", "semantic_model", "saved_query"], n.resource_type) && ("test" != i.resource_type || !i.hasOwnProperty("test_metadata"))) { var o = n.unique_id + "|" + i.unique_id, a = { group: "edges", data: { source: n.unique_id, target: i.unique_id, unique_id: o } }, s = i.unique_id; p.graph.pristine.edges[s] || (p.graph.pristine.edges[s] = []), p.graph.pristine.edges[s].push(a) } })) })); var t = new i.Graph({ directed: !0 }); r.each(p.graph.pristine.nodes, (function (e) { t.setNode(e.data.unique_id, e.data.name) })), r.each(p.graph.pristine.edges, (function (e) { r.each(e, (function (e) { t.setEdge(e.data.source, e.data.target) })) })), p.graph.pristine.dag = t, p.graph.elements = r.flatten(r.values(p.graph.pristine.nodes).concat(r.values(p.graph.pristine.edges))), f(t.nodes()) })), p.hideGraph = function () { p.orientation = "sidebar", p.expanded = !1 }, p.showVerticalGraph = function (e, t) { p.orientation = "sidebar", t && (p.expanded = !0); var n = h(r.assign({}, s.options, { include: "+" + e + "+", exclude: "", hops: 1 }), "vertical", !0); return p.graph.layout = d.top_down, p.graph.options = u.vertical, n }, p.showFullGraph = function (e) { p.orientation = "fullscreen", p.expanded = !0; var t = r.assign({}, s.options); e ? (t.include = "+" + e + "+", t.exclude = "") : (t.include = "", t.exclude = ""); var n = h(t, "horizontal", !0); return p.graph.layout = d.left_right, p.graph.options = u.horizontal, c.setState(t), n }, p.updateGraph = function (e) { p.orientation = "fullscreen", p.expanded = !0; var t = h(e, "horizontal", !1); return p.graph.layout = d.left_right, p.graph.options = u.horizontal, c.setState(e), t }, p.deselectNodes = function () { "fullscreen" == p.orientation && p.graph_element.elements().data("selected", 0) }, p.selectNode = function (e) { if ("fullscreen" == p.orientation) { p.graph.pristine.nodes[e]; var t = p.graph.pristine.dag, n = r.indexBy(o.ancestorNodes(t, e)), i = r.indexBy(o.descendentNodes(t, e)); n[e] = e, i[e] = e; var a = p.graph_element; r.each(p.graph.elements, (function (t) { var r = a.$id(t.data.id); n[t.data.source] && n[t.data.target] || i[t.data.source] && i[t.data.target] || t.data.unique_id == e ? r.data("selected", 1) : r.data("selected", 0) })) } }, p.markDirty = function (e) { p.markAllClean(), r.each(e, (function (e) { p.graph_element.$id(e).addClass("dirty") })) }, p.markAllClean = function () { p.graph_element && p.graph_element.elements().removeClass("dirty") }, p }]) }, function (e, t, n) { "use strict"; n.r(t), n.d(t, "isValidColor", (function () { return i })); const r = new Set(["aliceblue", "antiquewhite", "aqua", "aquamarine", "azure", "beige", "bisque", "black", "blanchedalmond", "blue", "blueviolet", "brown", "burlywood", "cadetblue", "chartreuse", "chocolate", "coral", "cornflowerblue", "cornsilk", "crimson", "cyan", "darkblue", "darkcyan", "darkgoldenrod", "darkgray", "darkgreen", "darkkhaki", "darkmagenta", "darkolivegreen", "darkorange", "darkorchid", "darkred", "darksalmon", "darkseagreen", "darkslateblue", "darkslategray", "darkturquoise", "darkviolet", "deeppink", "deepskyblue", "dimgray", "dodgerblue", "firebrick", "floralwhite", "forestgreen", "fuchsia", "ghostwhite", "gold", "goldenrod", "gray", "green", "greenyellow", "honeydew", "hotpink", "indianred", "indigo", "ivory", "khaki", "lavender", "lavenderblush", "lawngreen", "lemonchiffon", "lightblue", "lightcoral", "lightcyan", "lightgoldenrodyellow", "lightgray", "lightgreen", "lightpink", "lightsalmon", "lightsalmon", "lightseagreen", "lightskyblue", "lightslategray", "lightsteelblue", "lightyellow", "lime", "limegreen", "linen", "magenta", "maroon", "mediumaquamarine", "mediumblue", "mediumorchid", "mediumpurple", "mediumseagreen", "mediumslateblue", "mediumslateblue", "mediumspringgreen", "mediumturquoise", "mediumvioletred", "midnightblue", "mintcream", "mistyrose", "moccasin", "navajowhite", "navy", "oldlace", "olive", "olivedrab", "orange", "orangered", "orchid", "palegoldenrod", "palegreen", "paleturquoise", "palevioletred", "papayawhip", "peachpuff", "peru", "pink", "plum", "powderblue", "purple", "rebeccapurple", "red", "rosybrown", "royalblue", "saddlebrown", "salmon", "sandybrown", "seagreen", "seashell", "sienna", "silver", "skyblue", "slateblue", "slategray", "snow", "springgreen", "steelblue", "tan", "teal", "thistle", "tomato", "turquoise", "violet", "wheat", "white", "whitesmoke", "yellow", "yellowgreen"]); function i(e) { if (!e) return !1; const t = e.trim().toLowerCase(); if ("" === t) return !1; const n = t.match(/^#([A-Fa-f0-9]{3}){1,2}$/), i = r.has(t); return Boolean(n) || i } }, function (e, t, n) { n(33); const r = n(21), i = n(473); angular.module("dbt").factory("selectorService", ["$state", function (e) { var t = { include: "", exclude: "", packages: [], tags: [null], resource_types: ["model", "seed", "snapshot", "source", "test", "unit_test", "analysis", "exposure", "metric", "semantic_model", "saved_query"], depth: 1 }, n = { view_node: null, selection: { clean: r.clone(t), dirty: r.clone(t) }, options: { packages: [], tags: [null], resource_types: ["model", "seed", "snapshot", "source", "test", "analysis", "exposure", "metric", "semantic_model", "unit_test", "saved_query"] }, init: function (e) { r.each(e, (function (e, r) { n.options[r] = e, t[r] = e, n.selection.clean[r] = e, n.selection.dirty[r] = e })) }, resetSelection: function (e) { var i = { include: e && r.includes(["model", "seed", "snapshot"], e.resource_type) ? "+" + e.name + "+" : e && "source" == e.resource_type ? "+source:" + e.source_name + "." + e.name + "+" : e && "exposure" == e.resource_type ? "+exposure:" + e.name : e && "metric" == e.resource_type ? "+metric:" + e.name : e && "semantic_model" == e.resource_type ? "+semantic_model:" + e.name : e && "saved_query" == e.resource_type ? "+saved_query:" + e.name : e && r.includes(["analysis", "test", "unit_test"], e.resource_type) ? "+" + e.name : "" }, o = r.assign({}, t, i); n.selection.clean = r.clone(o), n.selection.dirty = r.clone(o), n.view_node = e }, getViewNode: function () { return n.view_node }, excludeNode: function (e, t) { var r, i = n.selection.dirty.exclude, o = t.parents ? "+" : "", a = t.children ? "+" : "", s = i.length > 0 ? " " : ""; "source" == e.resource_type ? (o += "source:", r = e.source_name + "." + e.name) : ["exposure", "metric", "semantic_model", "saved_query"].indexOf(e.resource_type) > -1 ? (o += e.resource_type + ":", r = e.name) : r = e.name; var l = i + s + o + r + a; return n.selection.dirty.exclude = l, n.updateSelection() }, selectSource: function (e, t) { var r = "source:" + e + (t.children ? "+" : ""); return n.selection.dirty.include = r, n.updateSelection() }, clearViewNode: function () { n.view_node = null }, isDirty: function () { return !r.isEqual(n.selection.clean, n.selection.dirty) }, updateSelection: function () { return n.selection.clean = r.clone(n.selection.dirty), n.selection.clean }, selectNodes: function (e, t, n) { return i.selectNodes(e, t, n) } }; return n }]) }, function (e, t, n) { const r = n(21), i = n(474); function o(e, t) { return t || (t = " "), r.filter(r.uniq(e.split(t)), (function (e) { return e.length > 0 })) } function a(e) { var t = { raw: e, select_at: !1, select_children: !1, children_depth: null, select_parents: !1, parents_depth: null }; const n = new RegExp("" + /^/.source + /(?(\@))?/.source + /(?((?(\d*))\+))?/.source + /((?([\w.]+)):)?/.source + /(?(.*?))/.source + /(?(\+(?(\d*))))?/.source + /$/.source).exec(e).groups; t.select_at = "@" == n.childs_parents, t.select_parents = !!n.parents, t.select_children = !!n.children, n.parents_depth && (t.parents_depth = parseInt(n.parents_depth)), n.children_depth && (t.children_depth = parseInt(n.children_depth)); var r = n.method, i = n.value; return r ? -1 != r.indexOf(".") && ([r, selector_modifier] = r.split(".", 2), i = { config: selector_modifier, value: i }) : r = "implicit", t.selector_type = r, t.selector_value = i, t } function s(e) { var t = o(e, " "); return r.map(t, (function (e) { var t = o(e, ","); return t.length > 1 ? { method: "intersect", selectors: r.map(t, a) } : { method: "none", selectors: r.map([e], a) } })) } function l(e, t) { var n = s(e), i = null, o = null; return r.each(n, (function (e) { var n = "intersect" == e.method ? r.intersection : r.union; r.each(e.selectors, (function (e) { var r = t(e); null === i ? (i = r.matched, o = r.selected) : (i = n(i, r.matched), o = n(o, r.selected)) })) })), { matched: i || [], selected: o || [] } } e.exports = { splitSpecs: o, parseSpec: a, parseSpecs: s, buildSpec: function (e, t, n) { return { include: s(e), exclude: s(t), hops: n } }, applySpec: l, selectNodes: function (e, t, n) { n.include, n.exclude; var o, a = r.partial(i.getNodesFromSpec, e, t, n.hops); r.values(t), o = 0 == n.include.trim().length ? { selected: e.nodes(), matched: [] } : l(n.include, a); var s = l(n.exclude, a), c = o.selected, u = o.matched; c = r.difference(c, s.selected), u = r.difference(u, s.matched); var d = []; return r.each(c, (function (e) { var i = t[e]; i.data.tags || (i.data.tags = []); var o = r.includes(n.packages, i.data.package_name), a = r.intersection(n.tags, i.data.tags).length > 0, s = r.includes(n.tags, null) && 0 == i.data.tags.length, l = r.includes(n.resource_types, i.data.resource_type); o && (a || s) && l || d.push(i.data.unique_id) })), { selected: r.difference(c, d), matched: r.difference(u, d) } } } }, function (e, t, n) { const r = n(21), i = n(203); var o = "fqn", a = "tag", s = "source", l = "exposure", c = "metric", u = "semantic_model", d = "saved_query", p = "group", f = "path", h = "file", g = "package", m = "config", v = "test_name", b = "test_type", y = {}; function x(e, t, n) { var r = e.slice(-1)[0], i = e.slice(-2, -1)[0]; if (t === r) return !0; if (version_options = [i, i + "_" + r, i + "." + r], n && version_options.includes(t)) return !0; var o = e.reduce((e, t) => e.concat(t.split(".")), []), a = t.split("."); if (o.length < a.length) return !1; for (var s = 0; s < a.length; s++) { var l = a[s]; if ("*" == l) return !0; if (o[s] != l) return !1 } return !0 } function w(e, t) { var n = []; return r.each(e, (function (e) { var i = e.data, o = i.fqn, a = null !== i.version; if (o && "source" != i.resource_type && "exposure" != i.resource_type && "metric" != i.resource_type && "semantic_model" != i.resource_type && "saved_query" != i.resource_type) { var s = r.rest(o); (x(o, t, a) || x(s, t, a)) && n.push(i) } })), r.uniq(n) } function k(e, t) { var n = [], i = t.split("/"); return r.each(e, (function (e) { var t = (e.data.original_file_path || "").split("/"), o = !0; r.each(i, (function (e, n) { "*" == e || "" == e || e != t[n] && (o = !1) })), o && n.push(e.data) })), n } function A(e, t) { var n = []; return r.each(e, (function (e) { var i = e.data.original_file_path.split("/"); r.last(i) == t && n.push(e.data) })), n } function E(e, t) { var n = []; return r.each(e, (function (e) { var i = e.data.tags; r.includes(i, t) && n.push(e.data) })), n } function S(e, t) { var n = []; return r.each(e, (function (e) { e.data.package_name == t && n.push(e.data) })), n } function $(e, t) { var n = []; return r.each(e, (function (e) { var r = e.data; r.config && r.config[t.config] == t.value && n.push(r) })), n } function C(e, t) { var n = []; return r.each(e, (function (e) { var r = e.data; r.test_metadata && r.test_metadata.name == t && n.push(r) })), n } function _(e, t) { var n = []; return r.each(e, (function (e) { var r = e.data; if ("test" != r.resource_type) return !1; (r.hasOwnProperty("test_metadata") && ["schema", "generic"].indexOf(t) > -1 || !r.hasOwnProperty("test_metadata") && ["data", "singular"].indexOf(t) > -1) && n.push(r) })), n } function O(e, t) { var n = []; return r.each(e, (function (e) { var r = e.data; if ("source" == r.resource_type) { var i, o, a = r.source_name, s = r.name; -1 != t.indexOf(".") ? [i, o] = t.split(".", 2) : (i = t, o = null), ("*" == i || i == a && "*" === o || i == a && o === s || i == a && null === o) && n.push(e.data) } })), n } y["implicit"] = function (e, t) { var n = w(e, t), i = k(e, t), o = []; t.toLowerCase().endsWith(".sql") && (o = A(e, t)); var a = r.uniq([].concat(r.map(n, "unique_id"), r.map(i, "unique_id"), r.map(o, "unique_id"))); return r.map(a, t => e[t].data) }, y[o] = w, y[a] = E, y[s] = O, y[l] = function (e, t) { var n = []; return r.each(e, (function (e) { var r = e.data; if ("exposure" == r.resource_type) { var i = r.name; ("*" == t || t == i) && n.push(e.data) } })), n }, y[c] = function (e, t) { var n = []; return r.each(e, (function (e) { var r = e.data; if ("metric" == r.resource_type) { var i = r.name; ("*" == t || t == i) && n.push(e.data) } })), n }, y[u] = function (e, t) { var n = []; return r.each(e, (function (e) { var r = e.data; if ("semantic_model" == r.resource_type) { var i = r.name; ("*" == t || t == i) && n.push(e.data) } })), n }, y[d] = function (e, t) { var n = []; return r.each(e, (function (e) { var r = e.data; if ("saved_query" == r.resource_type) { var i = r.name; ("*" == t || t == i) && n.push(e.data) } })), n }, y[p] = function (e, t) { var n = []; return r.each(e, (function (e) { var r = e.data; r.group == t && n.push(r) })), n }, y[f] = k, y[h] = A, y[g] = S, y[m] = $, y[v] = C, y[b] = _, e.exports = { isFQNMatch: x, getNodesByFQN: w, getNodesByTag: E, getNodesBySource: O, getNodesByPath: k, getNodesByPackage: S, getNodesByConfig: $, getNodesByTestName: C, getNodesByTestType: _, getNodesFromSpec: function (e, t, n, o) { const a = y[o.selector_type]; if (!a) return { selected: [], matched: [] }; var s = a(t, o.selector_value), l = [], c = []; return r.each(s, (function (t) { var a = t.unique_id; c.push(t.unique_id); var s = [], u = [], d = []; if (o.select_at && (d = r.union(i.selectAt(e, a))), o.select_parents) { var p = n || o.parents_depth; s = i.ancestorNodes(e, a, p) } if (o.select_children) { p = n || o.children_depth; u = i.descendentNodes(e, a, p) } l = r.union([a], l, u, s, d) })), { selected: l, matched: c } } } }, function (e, t, n) { const r = n(8); n(476); r.module("dbt").factory("trackingService", ["$location", "selectorService", "$rootScope", function (e, t, n) { var r = { initialized: !1, snowplow: null, project_id: null, init: function (e) { r.initialized || (r.initialized = !0, r.project_id = e.project_id, !0 === e.track && r.turn_on_tracking()) }, isHosted: function () { return hostedgetdbt = window.location.hostname.indexOf(".getdbt.com") > -1, hosteddbt = window.location.hostname.indexOf(".dbt.com") > -1, hostedgetdbt || hosteddbt }, turn_on_tracking: function () { var e, t, n, i, o, a; e = window, t = document, n = "script", e[i = "snowplow"] || (e.GlobalSnowplowNamespace = e.GlobalSnowplowNamespace || [], e.GlobalSnowplowNamespace.push(i), e[i] = function () { (e[i].q = e[i].q || []).push(arguments) }, e[i].q = e[i].q || [], o = t.createElement(n), a = t.getElementsByTagName(n)[0], o.async = 1, o.src = "//d1fc8wv8zag5ca.cloudfront.net/2.9.0/sp.js", a.parentNode.insertBefore(o, a)); var s = { appId: "dbt-docs", forceSecureTracker: !0, respectDoNotTrack: !0, userFingerprint: !1, contexts: { webPage: !0 } }; r.isHosted() && (window.location.hostname.indexOf(".getdbt.com") > -1 ? s.cookieDomain = ".getdbt.com" : s.cookieDomain = ".dbt.com"), r.snowplow = window.snowplow, r.snowplow("newTracker", "sp", "fishtownanalytics.sinter-collect.com", s), r.snowplow("enableActivityTracking", 30, 30), r.track_pageview() }, fuzzUrls: function () { r.isHosted() || (r.snowplow("setCustomUrl", "https://fuzzed.getdbt.com/"), r.snowplow("setReferrerUrl", "https://fuzzed.getdbt.com/")) }, getContext: function () { return [{ schema: "iglu:com.dbt/dbt_docs/jsonschema/1-0-0", data: { is_cloud_hosted: r.isHosted(), core_project_id: r.project_id } }] }, track_pageview: function () { if (r.snowplow) { r.fuzzUrls(); r.snowplow("trackPageView", null, r.getContext()) } }, track_event: function (e, t, n, i) { r.snowplow && (r.fuzzUrls(), r.snowplow("trackStructEvent", "dbt-docs", e, t, n, i, r.getContext())) }, track_graph_interaction: function (e, t) { r.snowplow && (r.fuzzUrls(), r.track_event("graph", "interact", e, t)) } }; return r }]) }, function (e, t, n) { var r, i, o, a, s; r = n(477), i = n(204).utf8, o = n(478), a = n(204).bin, (s = function (e, t) { e.constructor == String ? e = t && "binary" === t.encoding ? a.stringToBytes(e) : i.stringToBytes(e) : o(e) ? e = Array.prototype.slice.call(e, 0) : Array.isArray(e) || e.constructor === Uint8Array || (e = e.toString()); for (var n = r.bytesToWords(e), l = 8 * e.length, c = 1732584193, u = -271733879, d = -1732584194, p = 271733878, f = 0; f < n.length; f++)n[f] = 16711935 & (n[f] << 8 | n[f] >>> 24) | 4278255360 & (n[f] << 24 | n[f] >>> 8); n[l >>> 5] |= 128 << l % 32, n[14 + (l + 64 >>> 9 << 4)] = l; var h = s._ff, g = s._gg, m = s._hh, v = s._ii; for (f = 0; f < n.length; f += 16) { var b = c, y = u, x = d, w = p; c = h(c, u, d, p, n[f + 0], 7, -680876936), p = h(p, c, u, d, n[f + 1], 12, -389564586), d = h(d, p, c, u, n[f + 2], 17, 606105819), u = h(u, d, p, c, n[f + 3], 22, -1044525330), c = h(c, u, d, p, n[f + 4], 7, -176418897), p = h(p, c, u, d, n[f + 5], 12, 1200080426), d = h(d, p, c, u, n[f + 6], 17, -1473231341), u = h(u, d, p, c, n[f + 7], 22, -45705983), c = h(c, u, d, p, n[f + 8], 7, 1770035416), p = h(p, c, u, d, n[f + 9], 12, -1958414417), d = h(d, p, c, u, n[f + 10], 17, -42063), u = h(u, d, p, c, n[f + 11], 22, -1990404162), c = h(c, u, d, p, n[f + 12], 7, 1804603682), p = h(p, c, u, d, n[f + 13], 12, -40341101), d = h(d, p, c, u, n[f + 14], 17, -1502002290), c = g(c, u = h(u, d, p, c, n[f + 15], 22, 1236535329), d, p, n[f + 1], 5, -165796510), p = g(p, c, u, d, n[f + 6], 9, -1069501632), d = g(d, p, c, u, n[f + 11], 14, 643717713), u = g(u, d, p, c, n[f + 0], 20, -373897302), c = g(c, u, d, p, n[f + 5], 5, -701558691), p = g(p, c, u, d, n[f + 10], 9, 38016083), d = g(d, p, c, u, n[f + 15], 14, -660478335), u = g(u, d, p, c, n[f + 4], 20, -405537848), c = g(c, u, d, p, n[f + 9], 5, 568446438), p = g(p, c, u, d, n[f + 14], 9, -1019803690), d = g(d, p, c, u, n[f + 3], 14, -187363961), u = g(u, d, p, c, n[f + 8], 20, 1163531501), c = g(c, u, d, p, n[f + 13], 5, -1444681467), p = g(p, c, u, d, n[f + 2], 9, -51403784), d = g(d, p, c, u, n[f + 7], 14, 1735328473), c = m(c, u = g(u, d, p, c, n[f + 12], 20, -1926607734), d, p, n[f + 5], 4, -378558), p = m(p, c, u, d, n[f + 8], 11, -2022574463), d = m(d, p, c, u, n[f + 11], 16, 1839030562), u = m(u, d, p, c, n[f + 14], 23, -35309556), c = m(c, u, d, p, n[f + 1], 4, -1530992060), p = m(p, c, u, d, n[f + 4], 11, 1272893353), d = m(d, p, c, u, n[f + 7], 16, -155497632), u = m(u, d, p, c, n[f + 10], 23, -1094730640), c = m(c, u, d, p, n[f + 13], 4, 681279174), p = m(p, c, u, d, n[f + 0], 11, -358537222), d = m(d, p, c, u, n[f + 3], 16, -722521979), u = m(u, d, p, c, n[f + 6], 23, 76029189), c = m(c, u, d, p, n[f + 9], 4, -640364487), p = m(p, c, u, d, n[f + 12], 11, -421815835), d = m(d, p, c, u, n[f + 15], 16, 530742520), c = v(c, u = m(u, d, p, c, n[f + 2], 23, -995338651), d, p, n[f + 0], 6, -198630844), p = v(p, c, u, d, n[f + 7], 10, 1126891415), d = v(d, p, c, u, n[f + 14], 15, -1416354905), u = v(u, d, p, c, n[f + 5], 21, -57434055), c = v(c, u, d, p, n[f + 12], 6, 1700485571), p = v(p, c, u, d, n[f + 3], 10, -1894986606), d = v(d, p, c, u, n[f + 10], 15, -1051523), u = v(u, d, p, c, n[f + 1], 21, -2054922799), c = v(c, u, d, p, n[f + 8], 6, 1873313359), p = v(p, c, u, d, n[f + 15], 10, -30611744), d = v(d, p, c, u, n[f + 6], 15, -1560198380), u = v(u, d, p, c, n[f + 13], 21, 1309151649), c = v(c, u, d, p, n[f + 4], 6, -145523070), p = v(p, c, u, d, n[f + 11], 10, -1120210379), d = v(d, p, c, u, n[f + 2], 15, 718787259), u = v(u, d, p, c, n[f + 9], 21, -343485551), c = c + b >>> 0, u = u + y >>> 0, d = d + x >>> 0, p = p + w >>> 0 } return r.endian([c, u, d, p]) })._ff = function (e, t, n, r, i, o, a) { var s = e + (t & n | ~t & r) + (i >>> 0) + a; return (s << o | s >>> 32 - o) + t }, s._gg = function (e, t, n, r, i, o, a) { var s = e + (t & r | n & ~r) + (i >>> 0) + a; return (s << o | s >>> 32 - o) + t }, s._hh = function (e, t, n, r, i, o, a) { var s = e + (t ^ n ^ r) + (i >>> 0) + a; return (s << o | s >>> 32 - o) + t }, s._ii = function (e, t, n, r, i, o, a) { var s = e + (n ^ (t | ~r)) + (i >>> 0) + a; return (s << o | s >>> 32 - o) + t }, s._blocksize = 16, s._digestsize = 16, e.exports = function (e, t) { if (null == e) throw new Error("Illegal argument " + e); var n = r.wordsToBytes(s(e, t)); return t && t.asBytes ? n : t && t.asString ? a.bytesToString(n) : r.bytesToHex(n) } }, function (e, t) { var n, r; n = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/", r = { rotl: function (e, t) { return e << t | e >>> 32 - t }, rotr: function (e, t) { return e << 32 - t | e >>> t }, endian: function (e) { if (e.constructor == Number) return 16711935 & r.rotl(e, 8) | 4278255360 & r.rotl(e, 24); for (var t = 0; t < e.length; t++)e[t] = r.endian(e[t]); return e }, randomBytes: function (e) { for (var t = []; e > 0; e--)t.push(Math.floor(256 * Math.random())); return t }, bytesToWords: function (e) { for (var t = [], n = 0, r = 0; n < e.length; n++, r += 8)t[r >>> 5] |= e[n] << 24 - r % 32; return t }, wordsToBytes: function (e) { for (var t = [], n = 0; n < 32 * e.length; n += 8)t.push(e[n >>> 5] >>> 24 - n % 32 & 255); return t }, bytesToHex: function (e) { for (var t = [], n = 0; n < e.length; n++)t.push((e[n] >>> 4).toString(16)), t.push((15 & e[n]).toString(16)); return t.join("") }, hexToBytes: function (e) { for (var t = [], n = 0; n < e.length; n += 2)t.push(parseInt(e.substr(n, 2), 16)); return t }, bytesToBase64: function (e) { for (var t = [], r = 0; r < e.length; r += 3)for (var i = e[r] << 16 | e[r + 1] << 8 | e[r + 2], o = 0; o < 4; o++)8 * r + 6 * o <= 8 * e.length ? t.push(n.charAt(i >>> 6 * (3 - o) & 63)) : t.push("="); return t.join("") }, base64ToBytes: function (e) { e = e.replace(/[^A-Z0-9+\/]/gi, ""); for (var t = [], r = 0, i = 0; r < e.length; i = ++r % 4)0 != i && t.push((n.indexOf(e.charAt(r - 1)) & Math.pow(2, -2 * i + 8) - 1) << 2 * i | n.indexOf(e.charAt(r)) >>> 6 - 2 * i); return t } }, e.exports = r }, function (e, t) { + function n(e) { return !!e.constructor && "function" == typeof e.constructor.isBuffer && e.constructor.isBuffer(e) } + /*! + * Determine if an object is a Buffer + * + * @author Feross Aboukhadijeh + * @license MIT + */ + e.exports = function (e) { return null != e && (n(e) || function (e) { return "function" == typeof e.readFloatLE && "function" == typeof e.slice && n(e.slice(0, 0)) }(e) || !!e._isBuffer) } + }, function (e, t, n) { n(8).module("dbt").factory("locationService", ["$state", function (e) { var t = {}; return t.parseState = function (e) { return function (e) { return { selected: { include: e.g_i || "", exclude: e.g_e || "" }, show_graph: !!e.g_v } }(e) }, t.setState = function (t) { var n = function (e) { var t = { g_v: 1 }; return t.g_i = e.include, t.g_e = e.exclude, t }(t), r = e.current.name; e.go(r, n) }, t.clearState = function () { var t = e.current.name; e.go(t, { g_i: null, g_e: null, g_v: null }) }, t }]) }, function (e, t, n) { "use strict"; const r = n(8), i = n(202); r.module("dbt").controller("OverviewCtrl", ["$scope", "$state", "project", function (e, t, n) { e.overview_md = "(loading)", n.ready((function (n) { let r = t.params.project_name ? t.params.project_name : null; var o = n.docs["doc.dbt.__overview__"], a = i.filter(n.docs, { name: "__overview__" }); if (i.each(a, (function (e) { "dbt" != e.package_name && (o = e) })), null !== r) { o = n.docs[`doc.${r}.__${r}__`] || o; let e = i.filter(n.docs, { name: `__${r}__` }); i.each(e, e => { e.package_name !== r && (o = e) }) } e.overview_md = o.block_contents })) }]) }, function (e, t, n) { "use strict"; n(8).module("dbt").controller("SourceListCtrl", ["$scope", "$state", "project", function (e, t, n) { e.source = t.params.source, e.model = {}, e.extra_table_fields = [], e.has_more_info = function (e) { return (e.description || "").length }, e.toggle_source_expanded = function (t) { e.has_more_info(t) && (t.expanded = !t.expanded) }, n.ready((function (t) { var n = _.filter(t.nodes, (function (t) { return t.source_name == e.source })); if (0 != n.length) { n.sort((e, t) => e.name.localeCompare(t.name)); var r = n[0]; e.model = { name: e.source, source_description: r.source_description, sources: n }; var i = _.uniq(_.map(n, "metadata.owner")), o = _.uniq(_.map(n, "database")), a = _.uniq(_.map(n, "schema")); e.extra_table_fields = [{ name: "Loader", value: r.loader }, { name: 1 == i.length ? "Owner" : "Owners", value: i.join(", ") }, { name: 1 == o.length ? "Database" : "Databases", value: o.join(", ") }, { name: 1 == a.length ? "Schema" : "Schemas", value: a.join(", ") }, { name: "Tables", value: n.length }] } })) }]) }, function (e, t, n) { const r = n(8), i = { main: n(483), overview: n(484), graph: n(485), source: n(205), source_list: n(486), model: n(487), source: n(205), snapshot: n(488), seed: n(489), unit_test: n(490), test: n(491), analysis: n(492), macro: n(493), exposure: n(494), metric: n(495), semantic_model: n(496), saved_query: n(497), operation: n(498) }; r.module("dbt").config(["$stateProvider", "$urlRouterProvider", function (e, t) { var n = "g_v&g_i&g_e&g_p&g_n"; t.otherwise("/overview"), e.state("dbt", { url: "/", abstract: !0, controller: "MainController", templateUrl: i.main }).state("dbt.overview", { url: "overview?" + n, controller: "OverviewCtrl", templateUrl: i.overview }).state("dbt.project_overview", { url: "overview/:project_name?" + n, controller: "OverviewCtrl", templateUrl: i.overview, params: { project_name: { type: "string" } } }).state("dbt.graph", { url: "graph", controller: "GraphCtrl", templateUrl: i.graph }).state("dbt.model", { url: "model/:unique_id?section&" + n, controller: "ModelCtrl", templateUrl: i.model, params: { unique_id: { type: "string" } } }).state("dbt.seed", { url: "seed/:unique_id?section&" + n, controller: "SeedCtrl", templateUrl: i.seed, params: { unique_id: { type: "string" } } }).state("dbt.snapshot", { url: "snapshot/:unique_id?section&" + n, controller: "SnapshotCtrl", templateUrl: i.snapshot, params: { unique_id: { type: "string" } } }).state("dbt.unit_test", { url: "unit_test/:unique_id?section&" + n, controller: "TestCtrl", templateUrl: i.unit_test, params: { unique_id: { type: "string" } } }).state("dbt.test", { url: "test/:unique_id?section&" + n, controller: "TestCtrl", templateUrl: i.test, params: { unique_id: { type: "string" } } }).state("dbt.analysis", { url: "analysis/:unique_id?section&" + n, controller: "AnalysisCtrl", templateUrl: i.analysis, params: { unique_id: { type: "string" } } }).state("dbt.source", { url: "source/:unique_id?section&" + n, controller: "SourceCtrl", templateUrl: i.source, params: { unique_id: { type: "string" } } }).state("dbt.source_list", { url: "source_list/:source?section&" + n, controller: "SourceListCtrl", templateUrl: i.source_list, params: { source: { type: "string" } } }).state("dbt.macro", { url: "macro/:unique_id?section", controller: "MacroCtrl", templateUrl: i.macro, params: { unique_id: { type: "string" } } }).state("dbt.exposure", { url: "exposure/:unique_id?section&" + n, controller: "ExposureCtrl", templateUrl: i.exposure, params: { unique_id: { type: "string" } } }).state("dbt.metric", { url: "metric/:unique_id?section&" + n, controller: "MetricCtrl", templateUrl: i.metric, params: { unique_id: { type: "string" } } }).state("dbt.semantic_model", { url: "semantic_model/:unique_id?section&" + n, controller: "SemanticModelCtrl", templateUrl: i.semantic_model, params: { unique_id: { type: "string" } } }).state("dbt.saved_query", { url: "saved_query/:unique_id?section&" + n, controller: "SavedQueryCtrl", templateUrl: i.saved_query, params: { unique_id: { type: "string" } } }).state("dbt.operation", { url: "operation/:unique_id?section&" + n, controller: "OperationCtrl", templateUrl: i.operation, params: { unique_id: { type: "string" } } }) }]) }, function (e, t) { var n = "/main/main.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '\n\n
    \n \n\n
    \n
    \n
    \n
    \n
    \n
    \n \n
    \n
    \n
    \n
    \n
    \n \n
    \n
    \n
    \n
    \n
    \n
    \n \n
    \n
    \n \n
    \n
    \n
    \n
    \n \n
    \n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t) { var n = "/overview/overview.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '
    \n \n
    \n
    \n

    \n
    \n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t) { var n = "/graph/graph.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '
    \n
    \n
    \n
    \n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t) { var n = "/sources/source_list.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '\n\n
    \n \n
    \n
    \n
    \n
    \n \n
    \n\n
    \n
    \n
    \n
    Description
    \n
    \n
    \n
    \n
    This {{ model.resource_type }} is not currently documented
    \n
    \n
    \n
    \n
    \n\n\n
    \n
    \n
    \n
    Source Tables
    \n
    \n
    \n
    \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
    SourceTableDescriptionLinkMore?
    \n
    \n {{ source.source_name }}\n
    \n
    \n {{ source.name }}

    \n
    \n {{ source.description }}\n \n View docs\n \n \n \n \n \n \n \n \n \n
    \n
    \n
    \n
    Description
    \n \n
    \n
    \n
    \n
    \n
    \n
    \n
    \n
    \n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t) { var n = "/docs/model.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '\n\n
    \n \n
    \n
    \n
    \n
    \n \n
    \n\n
    \n
    \n
    \n
    Description
    \n
    \n
    \n
    \n
    This {{ model.resource_type }} is not currently documented
    \n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n
    Columns
    \n \n
    \n
    \n\n
    \n
    \n
    \n
    Referenced By
    \n \n
    \n
    \n\n
    \n
    \n
    \n
    Depends On
    \n \n
    \n
    \n\n
    \n
    \n
    \n \n
    \n
    \n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t) { var n = "/docs/snapshot.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '\n\n
    \n \n
    \n
    \n
    \n
    \n \n
    \n\n
    \n
    \n
    \n
    Description
    \n
    \n
    \n
    \n
    This {{ model.resource_type }} is not currently documented
    \n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n
    Columns
    \n \n
    \n
    \n\n
    \n
    \n
    \n
    Referenced By
    \n \n
    \n
    \n\n
    \n
    \n
    \n
    Depends On
    \n \n
    \n
    \n\n
    \n
    \n
    \n \n
    \n
    \n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t) { var n = "/docs/seed.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '\n\n
    \n \n
    \n
    \n
    \n
    \n \n
    \n\n
    \n
    \n
    \n
    Description
    \n
    \n
    \n
    \n
    This {{ model.resource_type }} is not currently documented
    \n
    \n
    \n
    \n
    \n
    \n
    \n
    \n
    Columns
    \n \n
    \n
    \n\n
    \n
    \n
    \n
    Referenced By
    \n \n
    \n
    \n\n
    \n
    \n
    \n
    Depends On
    \n \n
    \n
    \n\n
    \n
    \n
    \n \n
    \n
    \n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t) { var n = "/docs/unit_test.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '\n\n
    \n \n
    \n
    \n
    \n
    \n
    \n
    Description
    \n
    \n
    \n
    \n
    This {{ model.resource_type }} is not currently documented
    \n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n
    Depends On
    \n \n
    \n
    \n\n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t) { var n = "/docs/test.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '\n\n
    \n \n
    \n
    \n
    \n
    \n
    \n
    Description
    \n
    \n
    \n
    \n
    This {{ model.resource_type }} is not currently documented
    \n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n
    Depends On
    \n \n
    \n
    \n\n
    \n
    \n
    \n \n
    \n
    \n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t) { var n = "/docs/analysis.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '\n\n
    \n \n
    \n
    \n\n
    \n
    \n
    \n
    Description
    \n
    \n
    \n
    \n
    This {{ model.resource_type }} is not currently documented
    \n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n
    Depends On
    \n \n
    \n
    \n\n
    \n
    \n
    \n \n
    \n
    \n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t) { var n = "/docs/macro.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '\n\n
    \n \n
    \n
    \n
    \n
    \n
    \n
    Description
    \n
    \n
    \n
    \n
    This {{ macro.resource_type }} is not currently documented
    \n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n
    Arguments
    \n \n
    \n
    \n\n
    \n
    \n
    \n
    Referenced By
    \n \n
    \n
    \n\n
    \n
    \n
    \n
    Depends On
    \n \n
    \n
    \n\n
    \n
    \n
    \n \n
    \n
    \n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t) { var n = "/docs/exposure.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '\n\n
    \n \n
    \n
    \n\n
    \n
    \n \n
    \n\n
    \n
    \n
    \n
    Description
    \n
    \n
    \n
    \n
    This {{ exposure.resource_type }} is not currently documented
    \n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n
    Depends On
    \n \n
    \n
    \n\n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t) { var n = "/docs/metric.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '\n\n
    \n \n
    \n
    \n\n
    \n
    \n \n
    \n\n
    \n
    \n
    \n
    Description
    \n
    \n
    \n
    \n
    This {{ metric.resource_type }} is not currently documented
    \n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n
    Depends On
    \n \n
    \n
    \n\n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t) { var n = "/docs/semantic_model.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '\n\n
    \n \n
    \n
    \n\n
    \n
    \n \n
    \n\n
    \n
    \n
    \n
    Description
    \n
    \n
    \n
    \n
    This {{ semantic_model.resource_type }} is not currently documented
    \n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n
    Entities
    \n\n
    \n
    \n
    \n
    \n
    Name
    \n
    {{ entity.name }}
    \n
    None
    \n
    Type
    \n
    {{ entity.type }}
    \n
    None
    \n
    Expression
    \n
    {{ entity.expr }}
    \n
    None
    \n
    \n
    \n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n
    Depends On
    \n \n
    \n
    \n\n
    \n
    \n
    \n') }]), e.exports = n }, function (e, t) { var n = "/docs/saved_query.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '\n\n
    \n \n
    \n
    \n\n
    \n
    \n \n
    \n\n
    \n
    \n
    \n
    Description
    \n
    \n
    \n
    \n
    This {{ saved_query.resource_type }} is not currently documented
    \n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n
    Entities
    \n\n
    \n
    \n
    \n
    \n
    Name
    \n
    {{ entity.name }}
    \n
    None
    \n
    Type
    \n
    {{ entity.type }}
    \n
    None
    \n
    Expression
    \n
    {{ entity.expr }}
    \n
    None
    \n
    \n
    \n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n
    Depends On
    \n \n
    \n
    \n\n
    \n
    \n
    ') }]), e.exports = n }, function (e, t) { var n = "/docs/operation.html"; window.angular.module("ng").run(["$templateCache", function (e) { e.put(n, '\n\n
    \n \n
    \n
    \n
    \n
    \n
    \n
    Description
    \n
    \n
    \n
    \n
    This {{ model.resource_type }} is not currently documented
    \n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n
    Depends On
    \n \n
    \n
    \n\n
    \n
    \n
    \n \n
    \n
    \n
    \n
    \n
    \n') }]), e.exports = n }]); + //# sourceMappingURL=main.js.map + + + diff --git a/core/dbt/task/freshness.py b/core/dbt/task/freshness.py index e20671dc532..06e78b17c7b 100644 --- a/core/dbt/task/freshness.py +++ b/core/dbt/task/freshness.py @@ -1,9 +1,10 @@ import os import threading import time -from typing import AbstractSet, Dict, List, Optional +from typing import AbstractSet, Dict, List, Optional, Type from dbt import deprecations +from dbt.adapters.base import BaseAdapter from dbt.adapters.base.impl import FreshnessResponse from dbt.adapters.base.relation import BaseRelation from dbt.adapters.capability import Capability @@ -14,6 +15,7 @@ PartialSourceFreshnessResult, SourceFreshnessResult, ) +from dbt.contracts.graph.manifest import Manifest from dbt.contracts.graph.nodes import HookNode, SourceDefinition from dbt.contracts.results import RunStatus from dbt.events.types import FreshnessCheckComplete, LogFreshnessResult, LogStartLine @@ -44,7 +46,7 @@ def set_metadata_freshness_cache( def on_skip(self): raise DbtRuntimeError("Freshness: nodes cannot be skipped!") - def before_execute(self): + def before_execute(self) -> None: description = "freshness of {0.source_name}.{0.name}".format(self.node) fire_event( LogStartLine( @@ -55,7 +57,7 @@ def before_execute(self): ) ) - def after_execute(self, result): + def after_execute(self, result) -> None: if hasattr(result, "node"): source_name = result.node.source_name table_name = result.node.name @@ -162,7 +164,7 @@ def execute(self, compiled_node, manifest): **freshness, ) - def compile(self, manifest): + def compile(self, manifest: Manifest): if self.node.resource_type != NodeType.Source: # should be unreachable... raise DbtRuntimeError("freshness runner: got a non-Source") @@ -184,13 +186,13 @@ def __init__(self, args, config, manifest) -> None: super().__init__(args, config, manifest) self._metadata_freshness_cache: Dict[BaseRelation, FreshnessResult] = {} - def result_path(self): + def result_path(self) -> str: if self.args.output: return os.path.realpath(self.args.output) else: return os.path.join(self.config.project_target_path, RESULT_FILE_NAME) - def raise_on_first_error(self): + def raise_on_first_error(self) -> bool: return False def get_node_selector(self): @@ -203,10 +205,25 @@ def get_node_selector(self): resource_types=[NodeType.Source], ) - def before_run(self, adapter, selected_uids: AbstractSet[str]) -> None: - super().before_run(adapter, selected_uids) - if adapter.supports(Capability.TableLastModifiedMetadataBatch): - self.populate_metadata_freshness_cache(adapter, selected_uids) + def before_run(self, adapter: BaseAdapter, selected_uids: AbstractSet[str]) -> RunStatus: + populate_metadata_freshness_cache_status = RunStatus.Success + + before_run_status = super().before_run(adapter, selected_uids) + + if before_run_status == RunStatus.Success and adapter.supports( + Capability.TableLastModifiedMetadataBatch + ): + populate_metadata_freshness_cache_status = self.populate_metadata_freshness_cache( + adapter, selected_uids + ) + + if ( + before_run_status == RunStatus.Success + and populate_metadata_freshness_cache_status == RunStatus.Success + ): + return RunStatus.Success + else: + return RunStatus.Error def get_runner(self, node) -> BaseRunner: freshness_runner = super().get_runner(node) @@ -214,7 +231,7 @@ def get_runner(self, node) -> BaseRunner: freshness_runner.set_metadata_freshness_cache(self._metadata_freshness_cache) return freshness_runner - def get_runner_type(self, _): + def get_runner_type(self, _) -> Optional[Type[BaseRunner]]: return FreshnessRunner def get_result(self, results, elapsed_time, generated_at): @@ -222,7 +239,7 @@ def get_result(self, results, elapsed_time, generated_at): elapsed_time=elapsed_time, generated_at=generated_at, results=results ) - def task_end_messages(self, results): + def task_end_messages(self, results) -> None: for result in results: if result.status in ( FreshnessStatus.Error, @@ -242,7 +259,9 @@ def get_hooks_by_type(self, hook_type: RunHookType) -> List[HookNode]: deprecations.warn("source-freshness-project-hooks") return [] - def populate_metadata_freshness_cache(self, adapter, selected_uids: AbstractSet[str]) -> None: + def populate_metadata_freshness_cache( + self, adapter, selected_uids: AbstractSet[str] + ) -> RunStatus: if self.manifest is None: raise DbtInternalError("Manifest must be set to populate metadata freshness cache") @@ -265,6 +284,7 @@ def populate_metadata_freshness_cache(self, adapter, selected_uids: AbstractSet[ batch_metadata_sources ) self._metadata_freshness_cache.update(metadata_freshness_results) + return RunStatus.Success except Exception as e: # This error handling is intentionally very coarse. # If anything goes wrong during batch metadata calculation, we can safely @@ -275,6 +295,7 @@ def populate_metadata_freshness_cache(self, adapter, selected_uids: AbstractSet[ Note(msg=f"Metadata freshness could not be computed in batch: {e}"), EventLevel.WARN, ) + return RunStatus.Error def get_freshness_metadata_cache(self) -> Dict[BaseRelation, FreshnessResult]: return self._metadata_freshness_cache diff --git a/core/dbt/task/group_lookup.py b/core/dbt/task/group_lookup.py new file mode 100644 index 00000000000..c02ed26107c --- /dev/null +++ b/core/dbt/task/group_lookup.py @@ -0,0 +1,40 @@ +from typing import AbstractSet, Dict, Optional, Union + +from dbt.contracts.graph.manifest import Manifest +from dbt.contracts.graph.nodes import Group + +_node_id_to_group_name_map: Dict[str, str] = {} +_group_name_to_group_map: Dict[str, Group] = {} + + +def init(manifest: Optional[Manifest], selected_ids: AbstractSet[str]) -> None: + if not manifest: + return + + _every_group_name_to_group_map = {v.name: v for v in manifest.groups.values()} + + for group_name, node_ids in manifest.group_map.items(): + for node_id in node_ids: + # only add node to lookup if it's selected + if node_id in selected_ids: + _node_id_to_group_name_map[node_id] = group_name + + # only add group to lookup if it's not already there and if node is selected + if group_name not in _group_name_to_group_map: + _group_name_to_group_map[group_name] = _every_group_name_to_group_map[ + group_name + ] + + +def get(node_id: str) -> Optional[Dict[str, Union[str, Dict[str, str]]]]: + group_name = _node_id_to_group_name_map.get(node_id) + + if group_name is None: + return None + + group = _group_name_to_group_map.get(group_name) + + if group is None: + return None + + return group.to_logging_dict() diff --git a/core/dbt/task/list.py b/core/dbt/task/list.py index 09358df4ffe..2638920976a 100644 --- a/core/dbt/task/list.py +++ b/core/dbt/task/list.py @@ -1,4 +1,5 @@ import json +from typing import Iterator, List from dbt.cli.flags import Flags from dbt.config.runtime import RuntimeConfig @@ -16,7 +17,6 @@ from dbt.node_types import NodeType from dbt.task.base import resource_types_from_args from dbt.task.runnable import GraphRunnableTask -from dbt.task.test import TestSelector from dbt_common.events.contextvars import task_contextvars from dbt_common.events.functions import fire_event, warn_or_error from dbt_common.events.types import PrintEvent @@ -145,7 +145,7 @@ def generate_json(self): } ) - def generate_paths(self): + def generate_paths(self) -> Iterator[str]: for node in self._iterate_selected_nodes(): yield node.original_file_path @@ -177,7 +177,7 @@ def output_results(self, results): return self.node_results @property - def resource_types(self): + def resource_types(self) -> List[NodeType]: if self.args.models: return [NodeType.Model] @@ -196,23 +196,16 @@ def selection_arg(self): else: return self.args.select - def get_node_selector(self): + def get_node_selector(self) -> ResourceTypeSelector: if self.manifest is None or self.graph is None: raise DbtInternalError("manifest and graph must be set to get perform node selection") - if self.resource_types == [NodeType.Test]: - return TestSelector( - graph=self.graph, - manifest=self.manifest, - previous_state=self.previous_state, - ) - else: - return ResourceTypeSelector( - graph=self.graph, - manifest=self.manifest, - previous_state=self.previous_state, - resource_types=self.resource_types, - include_empty_nodes=True, - ) + return ResourceTypeSelector( + graph=self.graph, + manifest=self.manifest, + previous_state=self.previous_state, + resource_types=self.resource_types, + include_empty_nodes=True, + ) def interpret_results(self, results): # list command should always return 0 as exit code diff --git a/core/dbt/task/printer.py b/core/dbt/task/printer.py index 7bedbfaba93..58a39552450 100644 --- a/core/dbt/task/printer.py +++ b/core/dbt/task/printer.py @@ -1,4 +1,4 @@ -from typing import Dict +from typing import Dict, Optional, Union from dbt.artifacts.schemas.results import NodeStatus from dbt.events.types import ( @@ -13,6 +13,7 @@ StatsLine, ) from dbt.node_types import NodeType +from dbt.task import group_lookup from dbt_common.events.base_types import EventLevel from dbt_common.events.format import pluralize from dbt_common.events.functions import fire_event @@ -32,13 +33,14 @@ def get_counts(flat_nodes) -> str: counts[t] = counts.get(t, 0) + 1 - stat_line = ", ".join([pluralize(v, k).replace("_", " ") for k, v in counts.items()]) + sorted_items = sorted(counts.items(), key=lambda x: x[0]) + stat_line = ", ".join([pluralize(v, k).replace("_", " ") for k, v in sorted_items]) return stat_line def interpret_run_result(result) -> str: - if result.status in (NodeStatus.Error, NodeStatus.Fail): + if result.status in (NodeStatus.Error, NodeStatus.Fail, NodeStatus.PartialSuccess): return "error" elif result.status == NodeStatus.Skipped: return "skip" @@ -68,7 +70,12 @@ def print_run_status_line(results) -> None: fire_event(StatsLine(stats=stats)) -def print_run_result_error(result, newline: bool = True, is_warning: bool = False) -> None: +def print_run_result_error( + result, + newline: bool = True, + is_warning: bool = False, + group: Optional[Dict[str, Union[str, Dict[str, str]]]] = None, +) -> None: # set node_info for logging events node_info = None if hasattr(result, "node") and result.node: @@ -83,6 +90,7 @@ def print_run_result_error(result, newline: bool = True, is_warning: bool = Fals node_name=result.node.name, path=result.node.original_file_path, node_info=node_info, + group=group, ) ) else: @@ -92,6 +100,7 @@ def print_run_result_error(result, newline: bool = True, is_warning: bool = Fals node_name=result.node.name, path=result.node.original_file_path, node_info=node_info, + group=group, ) ) @@ -99,7 +108,7 @@ def print_run_result_error(result, newline: bool = True, is_warning: bool = Fals if is_warning: fire_event(RunResultWarningMessage(msg=result.message, node_info=node_info)) else: - fire_event(RunResultError(msg=result.message, node_info=node_info)) + fire_event(RunResultError(msg=result.message, node_info=node_info, group=group)) else: fire_event(RunResultErrorNoMessage(status=result.status, node_info=node_info)) @@ -119,11 +128,11 @@ def print_run_result_error(result, newline: bool = True, is_warning: bool = Fals elif result.message is not None: if newline: fire_event(Formatting("")) - fire_event(RunResultError(msg=result.message, node_info=node_info)) + fire_event(RunResultError(msg=result.message, node_info=node_info, group=group)) def print_run_end_messages(results, keyboard_interrupt: bool = False) -> None: - errors, warnings = [], [] + errors, warnings, partial_successes = [], [], [] for r in results: if r.status in (NodeStatus.RuntimeErr, NodeStatus.Error, NodeStatus.Fail): errors.append(r) @@ -133,20 +142,25 @@ def print_run_end_messages(results, keyboard_interrupt: bool = False) -> None: errors.append(r) elif r.status == NodeStatus.Warn: warnings.append(r) + elif r.status == NodeStatus.PartialSuccess: + partial_successes.append(r) fire_event(Formatting("")) fire_event( EndOfRunSummary( num_errors=len(errors), num_warnings=len(warnings), + num_partial_success=len(partial_successes), keyboard_interrupt=keyboard_interrupt, ) ) for error in errors: - print_run_result_error(error, is_warning=False) + group = group_lookup.get(error.node.unique_id) if hasattr(error, "node") else None + print_run_result_error(error, is_warning=False, group=group) for warning in warnings: - print_run_result_error(warning, is_warning=True) + group = group_lookup.get(warning.node.unique_id) if hasattr(warning, "node") else None + print_run_result_error(warning, is_warning=True, group=group) print_run_status_line(results) diff --git a/core/dbt/task/retry.py b/core/dbt/task/retry.py index fd943b1151f..9b3c3874718 100644 --- a/core/dbt/task/retry.py +++ b/core/dbt/task/retry.py @@ -23,7 +23,13 @@ from dbt.task.test import TestTask from dbt_common.exceptions import DbtRuntimeError -RETRYABLE_STATUSES = {NodeStatus.Error, NodeStatus.Fail, NodeStatus.Skipped, NodeStatus.RuntimeErr} +RETRYABLE_STATUSES = { + NodeStatus.Error, + NodeStatus.Fail, + NodeStatus.Skipped, + NodeStatus.RuntimeErr, + NodeStatus.PartialSuccess, +} IGNORE_PARENT_FLAGS = { "log_path", "output_path", @@ -115,13 +121,33 @@ def __init__(self, args: Flags, config: RuntimeConfig) -> None: self.task_class = TASK_DICT.get(self.previous_command_name) # type: ignore def run(self): - unique_ids = set( - [ - result.unique_id - for result in self.previous_results.results - if result.status in RETRYABLE_STATUSES - ] - ) + unique_ids = { + result.unique_id + for result in self.previous_results.results + if result.status in RETRYABLE_STATUSES + and not ( + self.previous_command_name != "run-operation" + and result.unique_id.startswith("operation.") + ) + } + + # We need this so that re-running of a microbatch model will only rerun + # batches that previously failed. Note _explicitly_ do no pass the + # batch info if there were _no_ successful batches previously. This is + # because passing the batch info _forces_ the microbatch process into + # _incremental_ model, and it may be that we need to be in full refresh + # mode which is only handled if batch_info _isn't_ passed for a node + batch_map = { + result.unique_id: result.batch_results + for result in self.previous_results.results + if result.batch_results is not None + and len(result.batch_results.successful) != 0 + and len(result.batch_results.failed) > 0 + and not ( + self.previous_command_name != "run-operation" + and result.unique_id.startswith("operation.") + ) + } class TaskWrapper(self.task_class): def get_graph_queue(self): @@ -138,6 +164,9 @@ def get_graph_queue(self): self.manifest, ) + if self.task_class == RunTask: + task.batch_map = batch_map + return_value = task.run() return return_value diff --git a/core/dbt/task/run.py b/core/dbt/task/run.py index 6263ee66b46..0b6fae4fc4c 100644 --- a/core/dbt/task/run.py +++ b/core/dbt/task/run.py @@ -1,41 +1,49 @@ import functools +import os import threading import time +from copy import deepcopy +from dataclasses import asdict from datetime import datetime -from typing import AbstractSet, Any, Dict, Iterable, List, Optional, Set, Tuple +from typing import AbstractSet, Any, Dict, Iterable, List, Optional, Set, Tuple, Type from dbt import tracking, utils -from dbt.adapters.base import BaseRelation -from dbt.adapters.events.types import ( - DatabaseErrorRunningHook, - FinishedRunningStats, - HooksRunning, -) +from dbt.adapters.base import BaseAdapter, BaseRelation +from dbt.adapters.events.types import FinishedRunningStats from dbt.adapters.exceptions import MissingMaterializationError from dbt.artifacts.resources import Hook +from dbt.artifacts.schemas.batch_results import BatchResults, BatchType from dbt.artifacts.schemas.results import ( - BaseResult, NodeStatus, RunningStatus, RunStatus, + TimingInfo, + collect_timing_info, ) from dbt.artifacts.schemas.run import RunResult from dbt.cli.flags import Flags from dbt.clients.jinja import MacroGenerator -from dbt.config.runtime import RuntimeConfig +from dbt.config import RuntimeConfig from dbt.context.providers import generate_runtime_model_context from dbt.contracts.graph.manifest import Manifest -from dbt.contracts.graph.nodes import HookNode, ResultNode +from dbt.contracts.graph.nodes import HookNode, ModelNode, ResultNode from dbt.events.types import ( LogHookEndLine, LogHookStartLine, LogModelResult, LogStartLine, + RunningOperationCaughtError, ) from dbt.exceptions import CompilationError, DbtInternalError, DbtRuntimeError from dbt.graph import ResourceTypeSelector from dbt.hooks import get_hook_dict +from dbt.materializations.incremental.microbatch import MicrobatchBuilder from dbt.node_types import NodeType, RunHookType +from dbt.task import group_lookup +from dbt.task.base import BaseRunner +from dbt.task.compile import CompileRunner, CompileTask +from dbt.task.printer import get_counts, print_run_end_messages +from dbt_common.clients.jinja import MacroProtocol from dbt_common.dataclass_schema import dbtClassMixin from dbt_common.events.base_types import EventLevel from dbt_common.events.contextvars import log_contextvars @@ -43,28 +51,6 @@ from dbt_common.events.types import Formatting from dbt_common.exceptions import DbtValidationError -from .compile import CompileRunner, CompileTask -from .printer import get_counts, print_run_end_messages - - -class Timer: - def __init__(self) -> None: - self.start = None - self.end = None - - @property - def elapsed(self): - if self.start is None or self.end is None: - return None - return self.end - self.start - - def __enter__(self): - self.start = time.time() - return self - - def __exit__(self, exc_type, exc_value, exc_tracebck): - self.end = time.time() - @functools.total_ordering class BiggestName(str): @@ -100,7 +86,34 @@ def get_hook(source, index): return Hook.from_dict(hook_dict) -def track_model_run(index, num_nodes, run_model_result): +def get_execution_status(sql: str, adapter: BaseAdapter) -> Tuple[RunStatus, str]: + if not sql.strip(): + return RunStatus.Success, "OK" + + try: + response, _ = adapter.execute(sql, auto_begin=False, fetch=False) + status = RunStatus.Success + message = response._message + except (KeyboardInterrupt, SystemExit): + raise + except DbtRuntimeError as exc: + status = RunStatus.Error + message = exc.msg + except Exception as exc: + status = RunStatus.Error + message = str(exc) + + return (status, message) + + +def _get_adapter_info(adapter, run_model_result) -> Dict[str, Any]: + """Each adapter returns a dataclass with a flexible dictionary for + adapter-specific fields. Only the non-'model_adapter_details' fields + are guaranteed cross adapter.""" + return asdict(adapter.get_adapter_run_info(run_model_result.node.config)) if adapter else {} + + +def track_model_run(index, num_nodes, run_model_result, adapter=None): if tracking.active_user is None: raise DbtInternalError("cannot track model run with no active user") invocation_id = get_invocation_id() @@ -110,10 +123,13 @@ def track_model_run(index, num_nodes, run_model_result): access = node.access.value if node.access is not None else None contract_enforced = node.contract.enforced versioned = True if node.version else False + incremental_strategy = node.config.incremental_strategy else: access = None contract_enforced = False versioned = False + incremental_strategy = None + tracking.track_model_run( { "invocation_id": invocation_id, @@ -124,6 +140,7 @@ def track_model_run(index, num_nodes, run_model_result): "run_skipped": run_model_result.status == NodeStatus.Skipped, "run_error": run_model_result.status == NodeStatus.Error, "model_materialization": node.get_materialization(), + "model_incremental_strategy": incremental_strategy, "model_id": utils.get_hash(node), "hashed_contents": utils.get_hashed_contents(node), "timing": [t.to_dict(omit_none=True) for t in run_model_result.timing], @@ -132,6 +149,7 @@ def track_model_run(index, num_nodes, run_model_result): "contract_enforced": contract_enforced, "access": access, "versioned": versioned, + "adapter_info": _get_adapter_info(adapter, run_model_result), } ) @@ -179,9 +197,23 @@ def get_node_representation(self): relation = relation.include(database=False) return str(relation) - def describe_node(self): + def describe_node(self) -> str: # TODO CL 'language' will be moved to node level when we change representation - return f"{self.node.language} {self.node.get_materialization()} model {self.get_node_representation()}" + materialization_strategy = self.node.config.get("incremental_strategy") + materialization = ( + "microbatch" + if materialization_strategy == "microbatch" + else self.node.get_materialization() + ) + return f"{self.node.language} {materialization} model {self.get_node_representation()}" + + def describe_batch(self, batch_start: Optional[datetime]) -> str: + # Only visualize date if batch_start year/month/day + formatted_batch_start = MicrobatchBuilder.format_batch_start( + batch_start, self.node.config.batch_size + ) + + return f"batch {formatted_batch_start} of {self.get_node_representation()}" def print_start_line(self): fire_event( @@ -195,6 +227,7 @@ def print_start_line(self): def print_result_line(self, result): description = self.describe_node() + group = group_lookup.get(self.node.unique_id) if result.status == NodeStatus.Error: status = result.status level = EventLevel.ERROR @@ -209,18 +242,66 @@ def print_result_line(self, result): total=self.num_nodes, execution_time=result.execution_time, node_info=self.node.node_info, + group=group, ), level=level, ) - def before_execute(self): + def print_batch_result_line( + self, + result: RunResult, + batch_start: Optional[datetime], + batch_idx: int, + batch_total: int, + exception: Optional[Exception], + ): + description = self.describe_batch(batch_start) + group = group_lookup.get(self.node.unique_id) + if result.status == NodeStatus.Error: + status = result.status + level = EventLevel.ERROR + else: + status = result.message + level = EventLevel.INFO + fire_event( + LogModelResult( + description=description, + status=status, + index=batch_idx, + total=batch_total, + execution_time=result.execution_time, + node_info=self.node.node_info, + group=group, + ), + level=level, + ) + if exception: + fire_event(RunningOperationCaughtError(exc=str(exception))) + + def print_batch_start_line( + self, batch_start: Optional[datetime], batch_idx: int, batch_total: int + ) -> None: + if batch_start is None: + return + + batch_description = self.describe_batch(batch_start) + fire_event( + LogStartLine( + description=batch_description, + index=batch_idx, + total=batch_total, + node_info=self.node.node_info, + ) + ) + + def before_execute(self) -> None: self.print_start_line() - def after_execute(self, result): - track_model_run(self.node_index, self.num_nodes, result) + def after_execute(self, result) -> None: + track_model_run(self.node_index, self.num_nodes, result, adapter=self.adapter) self.print_result_line(result) - def _build_run_model_result(self, model, context): + def _build_run_model_result(self, model, context, elapsed_time: float = 0.0): result = context["load_result"]("main") if not result: raise DbtRuntimeError("main is not being called during running model") @@ -232,10 +313,86 @@ def _build_run_model_result(self, model, context): status=RunStatus.Success, timing=[], thread_id=threading.current_thread().name, - execution_time=0, + execution_time=elapsed_time, message=str(result.response), adapter_response=adapter_response, failures=result.get("failures"), + batch_results=None, + ) + + def _build_run_microbatch_model_result( + self, model: ModelNode, batch_run_results: List[RunResult] + ) -> RunResult: + batch_results = BatchResults() + for result in batch_run_results: + if result.batch_results is not None: + batch_results += result.batch_results + else: + raise DbtInternalError( + "Got a run result without batch results for a batch run, this should be impossible" + ) + + num_successes = len(batch_results.successful) + num_failures = len(batch_results.failed) + + if num_failures == 0: + status = RunStatus.Success + msg = "SUCCESS" + elif num_successes == 0: + status = RunStatus.Error + msg = "ERROR" + else: + status = RunStatus.PartialSuccess + msg = f"PARTIAL SUCCESS ({num_successes}/{num_successes + num_failures})" + + if model.batch_info is not None: + new_batch_results = deepcopy(model.batch_info) + new_batch_results.failed = [] + new_batch_results = new_batch_results + batch_results + else: + new_batch_results = batch_results + + return RunResult( + node=model, + status=status, + timing=[], + thread_id=threading.current_thread().name, + # The execution_time here doesn't get propagated to logs because + # `safe_run_hooks` handles the elapsed time at the node level + execution_time=0, + message=msg, + adapter_response={}, + failures=num_failures, + batch_results=new_batch_results, + ) + + def _build_succesful_run_batch_result( + self, + model: ModelNode, + context: Dict[str, Any], + batch: BatchType, + elapsed_time: float = 0.0, + ) -> RunResult: + run_result = self._build_run_model_result(model, context, elapsed_time) + run_result.batch_results = BatchResults(successful=[batch]) + return run_result + + def _build_failed_run_batch_result( + self, + model: ModelNode, + batch: BatchType, + elapsed_time: float = 0.0, + ) -> RunResult: + return RunResult( + node=model, + status=RunStatus.Error, + timing=[], + thread_id=threading.current_thread().name, + execution_time=elapsed_time, + message="ERROR", + adapter_response={}, + failures=1, + batch_results=BatchResults(failed=[batch]), ) def _materialization_relations(self, result: Any, model) -> List[BaseRelation]: @@ -255,6 +412,48 @@ def _materialization_relations(self, result: Any, model) -> List[BaseRelation]: ) raise CompilationError(msg, node=model) + def _execute_model( + self, + hook_ctx: Any, + context_config: Any, + model: ModelNode, + context: Dict[str, Any], + materialization_macro: MacroProtocol, + ) -> RunResult: + try: + result = MacroGenerator( + materialization_macro, context, stack=context["context_macro_stack"] + )() + finally: + self.adapter.post_model_hook(context_config, hook_ctx) + + for relation in self._materialization_relations(result, model): + self.adapter.cache_added(relation.incorporate(dbt_created=True)) + + return self._build_run_model_result(model, context) + + def _execute_microbatch_model( + self, + hook_ctx: Any, + context_config: Any, + model: ModelNode, + manifest: Manifest, + context: Dict[str, Any], + materialization_macro: MacroProtocol, + ) -> RunResult: + batch_results = None + try: + batch_results = self._execute_microbatch_materialization( + model, manifest, context, materialization_macro + ) + finally: + self.adapter.post_model_hook(context_config, hook_ctx) + + if batch_results is not None: + return self._build_run_microbatch_model_result(model, batch_results) + else: + return self._build_run_model_result(model, context) + def execute(self, model, manifest): context = generate_runtime_model_context(model, self.config, manifest) @@ -283,27 +482,145 @@ def execute(self, model, manifest): ) hook_ctx = self.adapter.pre_model_hook(context_config) - try: - result = MacroGenerator( - materialization_macro, context, stack=context["context_macro_stack"] - )() - finally: - self.adapter.post_model_hook(context_config, hook_ctx) - for relation in self._materialization_relations(result, model): - self.adapter.cache_added(relation.incorporate(dbt_created=True)) + if ( + os.environ.get("DBT_EXPERIMENTAL_MICROBATCH") + and model.config.materialized == "incremental" + and model.config.incremental_strategy == "microbatch" + ): + return self._execute_microbatch_model( + hook_ctx, context_config, model, manifest, context, materialization_macro + ) + else: + return self._execute_model( + hook_ctx, context_config, model, context, materialization_macro + ) - return self._build_run_model_result(model, context) + def _execute_microbatch_materialization( + self, + model: ModelNode, + manifest: Manifest, + context: Dict[str, Any], + materialization_macro: MacroProtocol, + ) -> List[RunResult]: + batch_results: List[RunResult] = [] + microbatch_builder = MicrobatchBuilder( + model=model, + is_incremental=self._is_incremental(model), + event_time_start=getattr(self.config.args, "EVENT_TIME_START", None), + event_time_end=getattr(self.config.args, "EVENT_TIME_END", None), + default_end_time=self.config.invoked_at, + ) + # Indicates whether current batch should be run incrementally + incremental_batch = False + + # Note currently (9/30/2024) model.batch_info is only ever _not_ `None` + # IFF `dbt retry` is being run and the microbatch model had batches which + # failed on the run of the model (which is being retried) + if model.batch_info is None: + end = microbatch_builder.build_end_time() + start = microbatch_builder.build_start_time(end) + batches = microbatch_builder.build_batches(start, end) + else: + batches = model.batch_info.failed + # If there is batch info, then don't run as full_refresh and do force is_incremental + # not doing this risks blowing away the work that has already been done + if self._has_relation(model=model): + incremental_batch = True + + # iterate over each batch, calling materialization_macro to get a batch-level run result + for batch_idx, batch in enumerate(batches): + self.print_batch_start_line(batch[0], batch_idx + 1, len(batches)) + + exception = None + start_time = time.perf_counter() + try: + # Set start/end in context prior to re-compiling + model.config["__dbt_internal_microbatch_event_time_start"] = batch[0] + model.config["__dbt_internal_microbatch_event_time_end"] = batch[1] + + # Recompile node to re-resolve refs with event time filters rendered, update context + self.compiler.compile_node( + model, + manifest, + {}, + split_suffix=MicrobatchBuilder.format_batch_start( + batch[0], model.config.batch_size + ), + ) + # Update jinja context with batch context members + batch_context = microbatch_builder.build_batch_context( + incremental_batch=incremental_batch + ) + context.update(batch_context) + + # Materialize batch and cache any materialized relations + result = MacroGenerator( + materialization_macro, context, stack=context["context_macro_stack"] + )() + for relation in self._materialization_relations(result, model): + self.adapter.cache_added(relation.incorporate(dbt_created=True)) + + # Build result of executed batch + batch_run_result = self._build_succesful_run_batch_result( + model, context, batch, time.perf_counter() - start_time + ) + # At least one batch has been inserted successfully! + incremental_batch = True + + except (KeyboardInterrupt, SystemExit): + # reraise it for GraphRunnableTask.execute_nodes to handle + raise + except Exception as e: + exception = e + batch_run_result = self._build_failed_run_batch_result( + model, batch, time.perf_counter() - start_time + ) + + self.print_batch_result_line( + batch_run_result, batch[0], batch_idx + 1, len(batches), exception + ) + batch_results.append(batch_run_result) + + return batch_results + + def _has_relation(self, model) -> bool: + relation_info = self.adapter.Relation.create_from(self.config, model) + relation = self.adapter.get_relation( + relation_info.database, relation_info.schema, relation_info.name + ) + return relation is not None + + def _is_incremental(self, model) -> bool: + # TODO: Remove. This is a temporary method. We're working with adapters on + # a strategy to ensure we can access the `is_incremental` logic without drift + relation_info = self.adapter.Relation.create_from(self.config, model) + relation = self.adapter.get_relation( + relation_info.database, relation_info.schema, relation_info.name + ) + if ( + relation is not None + and relation.type == "table" + and model.config.materialized == "incremental" + ): + if model.config.full_refresh is not None: + return not model.config.full_refresh + else: + return not getattr(self.config.args, "FULL_REFRESH", False) + else: + return False class RunTask(CompileTask): - def __init__(self, args: Flags, config: RuntimeConfig, manifest: Manifest) -> None: + def __init__( + self, + args: Flags, + config: RuntimeConfig, + manifest: Manifest, + batch_map: Optional[Dict[str, BatchResults]] = None, + ) -> None: super().__init__(args, config, manifest) - self.ran_hooks: List[HookNode] = [] - self._total_executed = 0 - - def index_offset(self, value: int) -> int: - return self._total_executed + value + self.batch_map = batch_map def raise_on_first_error(self) -> bool: return False @@ -335,88 +652,101 @@ def get_hooks_by_type(self, hook_type: RunHookType) -> List[HookNode]: hooks.sort(key=self._hook_keyfunc) return hooks - def run_hooks(self, adapter, hook_type: RunHookType, extra_context) -> None: + def safe_run_hooks( + self, adapter: BaseAdapter, hook_type: RunHookType, extra_context: Dict[str, Any] + ) -> RunStatus: ordered_hooks = self.get_hooks_by_type(hook_type) - # on-run-* hooks should run outside of a transaction. This happens - # b/c psycopg2 automatically begins a transaction when a connection - # is created. + if hook_type == RunHookType.End and ordered_hooks: + fire_event(Formatting("")) + + # on-run-* hooks should run outside a transaction. This happens because psycopg2 automatically begins a transaction when a connection is created. adapter.clear_transaction() if not ordered_hooks: - return - num_hooks = len(ordered_hooks) + return RunStatus.Success - fire_event(Formatting("")) - fire_event(HooksRunning(num_hooks=num_hooks, hook_type=hook_type)) + status = RunStatus.Success + failed = False + num_hooks = len(ordered_hooks) - for idx, hook in enumerate(ordered_hooks, start=1): - # We want to include node_info in the appropriate log files, so use - # log_contextvars + for idx, hook in enumerate(ordered_hooks, 1): with log_contextvars(node_info=hook.node_info): - hook.update_event_status( - started_at=datetime.utcnow().isoformat(), node_status=RunningStatus.Started - ) - sql = self.get_hook_sql(adapter, hook, idx, num_hooks, extra_context) + hook.index = idx + hook_name = f"{hook.package_name}.{hook_type}.{hook.index - 1}" + execution_time = 0.0 + timing: List[TimingInfo] = [] + failures = 1 + + if not failed: + with collect_timing_info("compile", timing.append): + sql = self.get_hook_sql( + adapter, hook, hook.index, num_hooks, extra_context + ) + + started_at = timing[0].started_at or datetime.utcnow() + hook.update_event_status( + started_at=started_at.isoformat(), node_status=RunningStatus.Started + ) - hook_text = "{}.{}.{}".format(hook.package_name, hook_type, hook.index) - fire_event( - LogHookStartLine( - statement=hook_text, - index=idx, - total=num_hooks, - node_info=hook.node_info, + fire_event( + LogHookStartLine( + statement=hook_name, + index=hook.index, + total=num_hooks, + node_info=hook.node_info, + ) ) - ) - with Timer() as timer: - if len(sql.strip()) > 0: - response, _ = adapter.execute(sql, auto_begin=False, fetch=False) - status = response._message + with collect_timing_info("execute", timing.append): + status, message = get_execution_status(sql, adapter) + + finished_at = timing[1].completed_at or datetime.utcnow() + hook.update_event_status(finished_at=finished_at.isoformat()) + execution_time = (finished_at - started_at).total_seconds() + failures = 0 if status == RunStatus.Success else 1 + + if status == RunStatus.Success: + message = f"{hook_name} passed" else: - status = "OK" + message = f"{hook_name} failed, error:\n {message}" + failed = True + else: + status = RunStatus.Skipped + message = f"{hook_name} skipped" + + hook.update_event_status(node_status=status) + + self.node_results.append( + RunResult( + status=status, + thread_id="main", + timing=timing, + message=message, + adapter_response={}, + execution_time=execution_time, + failures=failures, + node=hook, + ) + ) - self.ran_hooks.append(hook) - hook.update_event_status(finished_at=datetime.utcnow().isoformat()) - hook.update_event_status(node_status=RunStatus.Success) fire_event( LogHookEndLine( - statement=hook_text, + statement=hook_name, status=status, - index=idx, + index=hook.index, total=num_hooks, - execution_time=timer.elapsed, + execution_time=execution_time, node_info=hook.node_info, ) ) - # `_event_status` dict is only used for logging. Make sure - # it gets deleted when we're done with it - hook.clear_event_status() - self._total_executed += len(ordered_hooks) + if hook_type == RunHookType.Start and ordered_hooks: + fire_event(Formatting("")) - fire_event(Formatting("")) - - def safe_run_hooks( - self, adapter, hook_type: RunHookType, extra_context: Dict[str, Any] - ) -> None: - try: - self.run_hooks(adapter, hook_type, extra_context) - except DbtRuntimeError as exc: - fire_event(DatabaseErrorRunningHook(hook_type=hook_type.value)) - self.node_results.append( - BaseResult( - status=RunStatus.Error, - thread_id="main", - timing=[], - message=f"{hook_type.value} failed, error:\n {exc.msg}", - adapter_response={}, - execution_time=0, - failures=1, - ) - ) + return status def print_results_line(self, results, execution_time) -> None: - nodes = [r.node for r in results if hasattr(r, "node")] + self.ran_hooks + nodes = [r.node for r in results if hasattr(r, "node")] stat_line = get_counts(nodes) execution = "" @@ -431,13 +761,24 @@ def print_results_line(self, results, execution_time) -> None: ) ) - def before_run(self, adapter, selected_uids: AbstractSet[str]) -> None: + def populate_microbatch_batches(self, selected_uids: AbstractSet[str]): + if self.batch_map is not None and self.manifest is not None: + for uid in selected_uids: + if uid in self.batch_map: + node = self.manifest.ref_lookup.perform_lookup(uid, self.manifest) + if isinstance(node, ModelNode): + node.batch_info = self.batch_map[uid] + + def before_run(self, adapter: BaseAdapter, selected_uids: AbstractSet[str]) -> RunStatus: with adapter.connection_named("master"): self.defer_to_manifest() required_schemas = self.get_model_schemas(adapter, selected_uids) self.create_schemas(adapter, required_schemas) self.populate_adapter_cache(adapter, required_schemas) - self.safe_run_hooks(adapter, RunHookType.Start, {}) + self.populate_microbatch_batches(selected_uids) + group_lookup.init(self.manifest, selected_uids) + run_hooks_status = self.safe_run_hooks(adapter, RunHookType.Start, {}) + return run_hooks_status def after_run(self, adapter, results) -> None: # in on-run-end hooks, provide the value 'database_schemas', which is a @@ -452,15 +793,30 @@ def after_run(self, adapter, results) -> None: and r.status not in (NodeStatus.Error, NodeStatus.Fail, NodeStatus.Skipped) } - self._total_executed += len(results) - extras = { "schemas": list({s for _, s in database_schema_set}), - "results": results, + "results": [ + r for r in results if r.thread_id != "main" or r.status == RunStatus.Error + ], # exclude that didn't fail to preserve backwards compatibility "database_schemas": list(database_schema_set), } - with adapter.connection_named("master"): - self.safe_run_hooks(adapter, RunHookType.End, extras) + + try: + with adapter.connection_named("master"): + self.safe_run_hooks(adapter, RunHookType.End, extras) + except (KeyboardInterrupt, SystemExit): + run_result = self.get_result( + results=self.node_results, + elapsed_time=time.time() - self.started_at, + generated_at=datetime.utcnow(), + ) + + if self.args.write_json and hasattr(run_result, "write"): + run_result.write(self.result_path()) + + print_run_end_messages(self.node_results, keyboard_interrupt=True) + + raise def get_node_selector(self) -> ResourceTypeSelector: if self.manifest is None or self.graph is None: @@ -472,7 +828,7 @@ def get_node_selector(self) -> ResourceTypeSelector: resource_types=[NodeType.Model], ) - def get_runner_type(self, _): + def get_runner_type(self, _) -> Optional[Type[BaseRunner]]: return ModelRunner def task_end_messages(self, results) -> None: diff --git a/core/dbt/task/run_operation.py b/core/dbt/task/run_operation.py index 6f7cd7b64c0..f87ac63e04e 100644 --- a/core/dbt/task/run_operation.py +++ b/core/dbt/task/run_operation.py @@ -2,15 +2,16 @@ import threading import traceback from datetime import datetime -from typing import TYPE_CHECKING +from typing import TYPE_CHECKING, List import dbt_common.exceptions from dbt.adapters.factory import get_adapter -from dbt.artifacts.schemas.results import RunStatus, TimingInfo +from dbt.artifacts.schemas.results import RunStatus, TimingInfo, collect_timing_info from dbt.artifacts.schemas.run import RunResult, RunResultsArtifact from dbt.contracts.files import FileHash from dbt.contracts.graph.nodes import HookNode from dbt.events.types import ( + ArtifactWritten, LogDebugStackTrace, RunningOperationCaughtError, RunningOperationUncaughtError, @@ -51,25 +52,29 @@ def _run_unsafe(self, package_name, macro_name) -> "agate.Table": return res def run(self) -> RunResultsArtifact: - start = datetime.utcnow() - self.compile_manifest() + timing: List[TimingInfo] = [] - success = True + with collect_timing_info("compile", timing.append): + self.compile_manifest() + + start = timing[0].started_at + success = True package_name, macro_name = self._get_macro_parts() - try: - self._run_unsafe(package_name, macro_name) - except dbt_common.exceptions.DbtBaseException as exc: - fire_event(RunningOperationCaughtError(exc=str(exc))) - fire_event(LogDebugStackTrace(exc_info=traceback.format_exc())) - success = False - except Exception as exc: - fire_event(RunningOperationUncaughtError(exc=str(exc))) - fire_event(LogDebugStackTrace(exc_info=traceback.format_exc())) - success = False + with collect_timing_info("execute", timing.append): + try: + self._run_unsafe(package_name, macro_name) + except dbt_common.exceptions.DbtBaseException as exc: + fire_event(RunningOperationCaughtError(exc=str(exc))) + fire_event(LogDebugStackTrace(exc_info=traceback.format_exc())) + success = False + except Exception as exc: + fire_event(RunningOperationUncaughtError(exc=str(exc))) + fire_event(LogDebugStackTrace(exc_info=traceback.format_exc())) + success = False - end = datetime.utcnow() + end = timing[1].completed_at macro = ( self.manifest.find_macro_by_name(macro_name, self.config.project_name, package_name) @@ -85,10 +90,12 @@ def run(self) -> RunResultsArtifact: f"dbt could not find a macro with the name '{macro_name}' in any package" ) + execution_time = (end - start).total_seconds() if start and end else 0.0 + run_result = RunResult( adapter_response={}, status=RunStatus.Success if success else RunStatus.Error, - execution_time=(end - start).total_seconds(), + execution_time=execution_time, failures=0 if success else 1, message=None, node=HookNode( @@ -105,12 +112,13 @@ def run(self) -> RunResultsArtifact: original_file_path="", ), thread_id=threading.current_thread().name, - timing=[TimingInfo(name=macro_name, started_at=start, completed_at=end)], + timing=timing, + batch_results=None, ) results = RunResultsArtifact.from_execution_results( - generated_at=end, - elapsed_time=(end - start).total_seconds(), + generated_at=end or datetime.utcnow(), + elapsed_time=execution_time, args={ k: v for k, v in self.args.__dict__.items() @@ -123,6 +131,11 @@ def run(self) -> RunResultsArtifact: if self.args.write_json: results.write(result_path) + fire_event( + ArtifactWritten( + artifact_type=results.__class__.__name__, artifact_path=result_path + ) + ) return results diff --git a/core/dbt/task/runnable.py b/core/dbt/task/runnable.py index c2e3b0e18ee..492fafff84d 100644 --- a/core/dbt/task/runnable.py +++ b/core/dbt/task/runnable.py @@ -5,13 +5,13 @@ from datetime import datetime from multiprocessing.dummy import Pool as ThreadPool from pathlib import Path -from typing import AbstractSet, Dict, Iterable, List, Optional, Set, Tuple, Union +from typing import AbstractSet, Dict, Iterable, List, Optional, Set, Tuple, Type, Union import dbt.exceptions import dbt.tracking import dbt.utils import dbt_common.utils.formatting -from dbt.adapters.base import BaseRelation +from dbt.adapters.base import BaseAdapter, BaseRelation from dbt.adapters.factory import get_adapter from dbt.artifacts.schemas.results import ( BaseResult, @@ -26,6 +26,7 @@ from dbt.contracts.graph.nodes import ResultNode from dbt.contracts.state import PreviousState from dbt.events.types import ( + ArtifactWritten, ConcurrencyLine, DefaultSelector, EndRunResult, @@ -36,6 +37,7 @@ NodeStart, NothingToDo, QueryCancelationUnsupported, + SkippingDetails, ) from dbt.exceptions import DbtInternalError, DbtRuntimeError, FailFastError from dbt.flags import get_flags @@ -65,8 +67,16 @@ class GraphRunnableMode(StrEnum): Independent = "independent" +def mark_node_as_skipped( + node: ResultNode, executed_node_ids: Set[str], message: Optional[str] +) -> Optional[RunResult]: + if node.unique_id not in executed_node_ids: + return RunResult.from_node(node, RunStatus.Skipped, message) + return None + + class GraphRunnableTask(ConfiguredTask): - MARK_DEPENDENT_ERRORS_STATUSES = [NodeStatus.Error] + MARK_DEPENDENT_ERRORS_STATUSES = [NodeStatus.Error, NodeStatus.PartialSuccess] def __init__(self, args: Flags, config: RuntimeConfig, manifest: Manifest) -> None: super().__init__(args, config, manifest) @@ -181,13 +191,13 @@ def _runtime_initialize(self): self.num_nodes = len([n for n in self._flattened_nodes if not n.is_ephemeral_model]) - def raise_on_first_error(self): + def raise_on_first_error(self) -> bool: return False - def get_runner_type(self, node): + def get_runner_type(self, node) -> Optional[Type[BaseRunner]]: raise NotImplementedError("Not Implemented") - def result_path(self): + def result_path(self) -> str: return os.path.join(self.config.project_target_path, RESULT_FILE_NAME) def get_runner(self, node) -> BaseRunner: @@ -204,6 +214,10 @@ def get_runner(self, node) -> BaseRunner: num_nodes = self.num_nodes cls = self.get_runner_type(node) + + if cls is None: + raise DbtInternalError("Could not find runner type for node.") + return cls(self.config, adapter, node, run_count, num_nodes) def call_runner(self, runner: BaseRunner) -> RunResult: @@ -247,6 +261,7 @@ def call_runner(self, runner: BaseRunner) -> RunResult: adapter_response={}, message=msg, failures=None, + batch_results=None, node=runner.node, ) @@ -256,7 +271,10 @@ def call_runner(self, runner: BaseRunner) -> RunResult: fail_fast = get_flags().FAIL_FAST - if result.status in (NodeStatus.Error, NodeStatus.Fail) and fail_fast: + if ( + result.status in (NodeStatus.Error, NodeStatus.Fail, NodeStatus.PartialSuccess) + and fail_fast + ): self._raise_next_tick = FailFastError( msg="Failing early due to test failure or runtime error", result=result, @@ -334,7 +352,7 @@ def handle_job_queue(self, pool, callback): args = [runner] self._submit(pool, args, callback) - def _handle_result(self, result: RunResult): + def _handle_result(self, result: RunResult) -> None: """Mark the result as completed, insert the `CompileResultNode` into the manifest, and mark any descendants (potentially with a 'cause' if the result was an ephemeral model) as skipped. @@ -382,14 +400,6 @@ def _cancel_connections(self, pool): def execute_nodes(self): num_threads = self.config.threads - target_name = self.config.target_name - - fire_event( - ConcurrencyLine( - num_threads=num_threads, target_name=target_name, node_count=self.num_nodes - ) - ) - fire_event(Formatting("")) pool = ThreadPool(num_threads, self._pool_thread_initializer, [get_invocation_context()]) try: @@ -397,12 +407,13 @@ def execute_nodes(self): except FailFastError as failure: self._cancel_connections(pool) - executed_node_ids = [r.node.unique_id for r in self.node_results] + executed_node_ids = {r.node.unique_id for r in self.node_results} + message = "Skipping due to fail_fast" - for r in self._flattened_nodes: - if r.unique_id not in executed_node_ids: + for node in self._flattened_nodes: + if node.unique_id not in executed_node_ids: self.node_results.append( - RunResult.from_node(r, RunStatus.Skipped, "Skipping due to fail_fast") + mark_node_as_skipped(node, executed_node_ids, message) ) print_run_result_error(failure.result) @@ -417,6 +428,12 @@ def execute_nodes(self): if self.args.write_json and hasattr(run_result, "write"): run_result.write(self.result_path()) + fire_event( + ArtifactWritten( + artifact_type=run_result.__class__.__name__, + artifact_path=self.result_path(), + ) + ) self._cancel_connections(pool) print_run_end_messages(self.node_results, keyboard_interrupt=True) @@ -474,12 +491,13 @@ def populate_adapter_cache( {"adapter_cache_construction_elapsed": cache_populate_time} ) - def before_run(self, adapter, selected_uids: AbstractSet[str]): + def before_run(self, adapter: BaseAdapter, selected_uids: AbstractSet[str]) -> RunStatus: with adapter.connection_named("master"): self.defer_to_manifest() self.populate_adapter_cache(adapter) + return RunStatus.Success - def after_run(self, adapter, results): + def after_run(self, adapter, results) -> None: pass def print_results_line(self, node_results, elapsed): @@ -487,10 +505,47 @@ def print_results_line(self, node_results, elapsed): def execute_with_hooks(self, selected_uids: AbstractSet[str]): adapter = get_adapter(self.config) + + fire_event(Formatting("")) + fire_event( + ConcurrencyLine( + num_threads=self.config.threads, + target_name=self.config.target_name, + node_count=self.num_nodes, + ) + ) + fire_event(Formatting("")) + self.started_at = time.time() try: - self.before_run(adapter, selected_uids) - res = self.execute_nodes() + before_run_status = self.before_run(adapter, selected_uids) + if before_run_status == RunStatus.Success or ( + not get_flags().skip_nodes_if_on_run_start_fails + ): + res = self.execute_nodes() + else: + executed_node_ids = { + r.node.unique_id for r in self.node_results if hasattr(r, "node") + } + + res = [] + + for index, node in enumerate(self._flattened_nodes or []): + if node.unique_id not in executed_node_ids: + fire_event( + SkippingDetails( + resource_type=node.resource_type, + schema=node.schema, + node_name=node.name, + index=index + 1, + total=self.num_nodes, + node_info=node.node_info, + ) + ) + skipped_node_result = mark_node_as_skipped(node, executed_node_ids, None) + if skipped_node_result: + self.node_results.append(skipped_node_result) + self.after_run(adapter, res) finally: adapter.cleanup_connections() @@ -517,7 +572,6 @@ def run(self): ) if len(self._flattened_nodes) == 0: - fire_event(Formatting("")) warn_or_error(NothingToDo()) result = self.get_result( results=[], @@ -525,7 +579,6 @@ def run(self): elapsed_time=0.0, ) else: - fire_event(Formatting("")) selected_uids = frozenset(n.unique_id for n in self._flattened_nodes) result = self.execute_with_hooks(selected_uids) @@ -545,6 +598,11 @@ def run(self): write_manifest(self.manifest, self.config.project_target_path) if hasattr(result, "write"): result.write(self.result_path()) + fire_event( + ArtifactWritten( + artifact_type=result.__class__.__name__, artifact_path=self.result_path() + ) + ) self.task_end_messages(result.results) return result @@ -659,7 +717,7 @@ def get_result(self, results, elapsed_time, generated_at): args=dbt.utils.args_to_dict(self.args), ) - def task_end_messages(self, results): + def task_end_messages(self, results) -> None: print_run_end_messages(results) def _get_previous_state(self) -> Optional[Manifest]: diff --git a/core/dbt/task/seed.py b/core/dbt/task/seed.py index 031bcdf7feb..961a55c131b 100644 --- a/core/dbt/task/seed.py +++ b/core/dbt/task/seed.py @@ -1,9 +1,12 @@ import random +from typing import Optional, Type from dbt.artifacts.schemas.results import NodeStatus, RunStatus +from dbt.contracts.graph.manifest import Manifest from dbt.events.types import LogSeedResult, LogStartLine, SeedHeader from dbt.graph import ResourceTypeSelector from dbt.node_types import NodeType +from dbt.task.base import BaseRunner from dbt_common.events.base_types import EventLevel from dbt_common.events.functions import fire_event from dbt_common.events.types import Formatting @@ -14,10 +17,10 @@ class SeedRunner(ModelRunner): - def describe_node(self): + def describe_node(self) -> str: return "seed file {}".format(self.get_node_representation()) - def before_execute(self): + def before_execute(self) -> None: fire_event( LogStartLine( description=self.describe_node(), @@ -33,7 +36,7 @@ def _build_run_model_result(self, model, context): result.agate_table = agate_result.table return result - def compile(self, manifest): + def compile(self, manifest: Manifest): return self.node def print_result_line(self, result): @@ -55,7 +58,7 @@ def print_result_line(self, result): class SeedTask(RunTask): - def raise_on_first_error(self): + def raise_on_first_error(self) -> bool: return False def get_node_selector(self): @@ -68,10 +71,10 @@ def get_node_selector(self): resource_types=[NodeType.Seed], ) - def get_runner_type(self, _): + def get_runner_type(self, _) -> Optional[Type[BaseRunner]]: return SeedRunner - def task_end_messages(self, results): + def task_end_messages(self, results) -> None: if self.args.show: self.show_tables(results) diff --git a/core/dbt/task/show.py b/core/dbt/task/show.py index e986b94814b..d9ba25efe32 100644 --- a/core/dbt/task/show.py +++ b/core/dbt/task/show.py @@ -2,11 +2,13 @@ import threading import time +from dbt.adapters.factory import get_adapter from dbt.artifacts.schemas.run import RunResult, RunStatus from dbt.context.providers import generate_runtime_model_context from dbt.contracts.graph.nodes import SeedNode from dbt.events.types import ShowNode from dbt.flags import get_flags +from dbt.task.base import ConfiguredTask from dbt.task.compile import CompileRunner, CompileTask from dbt.task.seed import SeedRunner from dbt_common.events.base_types import EventLevel @@ -53,6 +55,7 @@ def execute(self, compiled_node, manifest): adapter_response=adapter_response.to_dict(), agate_table=execute_result, failures=None, + batch_results=None, ) @@ -68,7 +71,7 @@ def get_runner_type(self, node): else: return ShowRunner - def task_end_messages(self, results): + def task_end_messages(self, results) -> None: is_inline = bool(getattr(self.args, "inline", None)) if is_inline: @@ -111,7 +114,7 @@ def task_end_messages(self, results): # No formatting, still get to stdout when --quiet is used fire_event(PrintEvent(msg=show_node_event.message())) - def _handle_result(self, result): + def _handle_result(self, result) -> None: super()._handle_result(result) if ( @@ -120,3 +123,28 @@ def _handle_result(self, result): and (self.args.select or getattr(self.args, "inline", None)) ): self.node_results.append(result) + + +class ShowTaskDirect(ConfiguredTask): + def run(self): + adapter = get_adapter(self.config) + with adapter.connection_named("show", should_release_connection=False): + response, table = adapter.execute( + self.args.inline_direct, fetch=True, limit=self.args.limit + ) + + output = io.StringIO() + if self.args.output == "json": + table.to_json(path=output) + else: + table.print_table(output=output, max_rows=None) + + fire_event( + ShowNode( + node_name="direct-query", + preview=output.getvalue(), + is_inline=True, + output_format=self.args.output, + unique_id="direct-query", + ) + ) diff --git a/core/dbt/task/snapshot.py b/core/dbt/task/snapshot.py index 5604d9cc546..3d8873f21fc 100644 --- a/core/dbt/task/snapshot.py +++ b/core/dbt/task/snapshot.py @@ -1,7 +1,10 @@ +from typing import Optional, Type + from dbt.artifacts.schemas.results import NodeStatus from dbt.events.types import LogSnapshotResult from dbt.graph import ResourceTypeSelector from dbt.node_types import NodeType +from dbt.task.base import BaseRunner from dbt_common.events.base_types import EventLevel from dbt_common.events.functions import fire_event from dbt_common.exceptions import DbtInternalError @@ -11,7 +14,7 @@ class SnapshotRunner(ModelRunner): - def describe_node(self): + def describe_node(self) -> str: return "snapshot {}".format(self.get_node_representation()) def print_result_line(self, result): @@ -34,7 +37,7 @@ def print_result_line(self, result): class SnapshotTask(RunTask): - def raise_on_first_error(self): + def raise_on_first_error(self) -> bool: return False def get_node_selector(self): @@ -47,5 +50,5 @@ def get_node_selector(self): resource_types=[NodeType.Snapshot], ) - def get_runner_type(self, _): + def get_runner_type(self, _) -> Optional[Type[BaseRunner]]: return SnapshotRunner diff --git a/core/dbt/task/sql.py b/core/dbt/task/sql.py index 2fa6f8ccb8c..ab8c89d6b98 100644 --- a/core/dbt/task/sql.py +++ b/core/dbt/task/sql.py @@ -5,6 +5,7 @@ import dbt.exceptions import dbt_common.exceptions.base +from dbt.contracts.graph.manifest import Manifest from dbt.contracts.sql import ( RemoteCompileResult, RemoteCompileResultMixin, @@ -28,18 +29,19 @@ def handle_exception(self, e, ctx): exc=str(e), exc_info=traceback.format_exc(), node_info=self.node.node_info ) ) + # REVIEW: This code is invalid and will always throw. if isinstance(e, dbt.exceptions.Exception): if isinstance(e, dbt_common.exceptions.DbtRuntimeError): e.add_node(ctx.node) return e - def before_execute(self): + def before_execute(self) -> None: pass - def after_execute(self, result): + def after_execute(self, result) -> None: pass - def compile(self, manifest): + def compile(self, manifest: Manifest): return self.compiler.compile_node(self.node, manifest, {}, write=False) @abstractmethod diff --git a/core/dbt/task/test.py b/core/dbt/task/test.py index 546bd43a943..6580b85879b 100644 --- a/core/dbt/task/test.py +++ b/core/dbt/task/test.py @@ -3,7 +3,17 @@ import re import threading from dataclasses import dataclass -from typing import TYPE_CHECKING, Any, Dict, List, Optional, Tuple, Union +from typing import ( + TYPE_CHECKING, + Any, + Collection, + Dict, + List, + Optional, + Tuple, + Type, + Union, +) import daff @@ -25,8 +35,9 @@ from dbt.exceptions import BooleanError, DbtInternalError from dbt.flags import get_flags from dbt.graph import ResourceTypeSelector -from dbt.node_types import NodeType +from dbt.node_types import TEST_NODE_TYPES, NodeType from dbt.parser.unit_tests import UnitTestManifestLoader +from dbt.task.base import BaseRunner, resource_types_from_args from dbt.utils import _coerce_decimal, strtobool from dbt_common.dataclass_schema import dbtClassMixin from dbt_common.events.format import pluralize @@ -34,6 +45,7 @@ from dbt_common.exceptions import DbtBaseException, DbtRuntimeError from dbt_common.ui import green, red +from . import group_lookup from .compile import CompileRunner from .run import RunTask @@ -82,23 +94,26 @@ class UnitTestResultData(dbtClassMixin): class TestRunner(CompileRunner): _ANSI_ESCAPE = re.compile(r"\x1B(?:[@-Z\\-_]|\[[0-?]*[ -/]*[@-~])") - _LOG_TEST_RESULT_EVENTS = LogTestResult - def describe_node_name(self): + def describe_node_name(self) -> str: if self.node.resource_type == NodeType.Unit: name = f"{self.node.model}::{self.node.versioned_name}" return name else: return self.node.name - def describe_node(self): + def describe_node(self) -> str: return f"{self.node.resource_type} {self.describe_node_name()}" def print_result_line(self, result): model = result.node + group = group_lookup.get(model.unique_id) + attached_node = ( + result.node.attached_node if isinstance(result.node, GenericTestNode) else None + ) fire_event( - self._LOG_TEST_RESULT_EVENTS( + LogTestResult( name=self.describe_node_name(), status=str(result.status), index=self.node_index, @@ -106,6 +121,8 @@ def print_result_line(self, result): execution_time=result.execution_time, node_info=model.node_info, num_failures=result.failures, + group=group, + attached_node=attached_node, ), level=LogTestResult.status_to_level(str(result.status)), ) @@ -120,13 +137,13 @@ def print_start_line(self): ) ) - def before_execute(self): + def before_execute(self) -> None: self.print_start_line() def execute_data_test(self, data_test: TestNode, manifest: Manifest) -> TestResultData: context = generate_runtime_model_context(data_test, self.config, manifest) - hook_ctx = self.adapter.pre_model_hook(context) + hook_ctx = self.adapter.pre_model_hook(context["config"]) materialization_macro = manifest.find_materialization_macro_by_name( self.config.project_name, data_test.get_materialization(), self.adapter.type() @@ -204,7 +221,7 @@ def execute_unit_test( # materialization, not compile the node.compiled_code context = generate_runtime_model_context(unit_test_node, self.config, unit_test_manifest) - hook_ctx = self.adapter.pre_model_hook(context) + hook_ctx = self.adapter.pre_model_hook(context["config"]) materialization_macro = unit_test_manifest.find_materialization_macro_by_name( self.config.project_name, unit_test_node.get_materialization(), self.adapter.type() @@ -287,7 +304,7 @@ def build_test_run_result(self, test: TestNode, result: TestResultData) -> RunRe failures = result.failures elif result.should_warn: if get_flags().WARN_ERROR or get_flags().WARN_ERROR_OPTIONS.includes( - self._LOG_TEST_RESULT_EVENTS.__name__ + LogTestResult.__name__ ): status = TestStatus.Fail message = f"Got {num_errors}, configured to fail if {test.config.warn_if}" @@ -307,6 +324,7 @@ def build_test_run_result(self, test: TestNode, result: TestResultData) -> RunRe message=message, adapter_response=result.adapter_response, failures=failures, + batch_results=None, ) return run_result @@ -332,9 +350,10 @@ def build_unit_test_run_result( message=message, adapter_response=result.adapter_response, failures=failures, + batch_results=None, ) - def after_execute(self, result): + def after_execute(self, result) -> None: self.print_result_line(result) def _get_unit_test_agate_table(self, result_table, actual_or_expected: str): @@ -374,16 +393,6 @@ def _render_daff_diff(self, daff_diff: daff.TableDiff) -> str: return rendered -class TestSelector(ResourceTypeSelector): - def __init__(self, graph, manifest, previous_state) -> None: - super().__init__( - graph=graph, - manifest=manifest, - previous_state=previous_state, - resource_types=[NodeType.Test, NodeType.Unit], - ) - - class TestTask(RunTask): """ Testing: @@ -393,19 +402,30 @@ class TestTask(RunTask): __test__ = False - def raise_on_first_error(self): + def raise_on_first_error(self) -> bool: return False - def get_node_selector(self) -> TestSelector: + @property + def resource_types(self) -> List[NodeType]: + resource_types: Collection[NodeType] = resource_types_from_args( + self.args, set(TEST_NODE_TYPES), set(TEST_NODE_TYPES) + ) + + # filter out any non-test node types + resource_types = [rt for rt in resource_types if rt in TEST_NODE_TYPES] + return list(resource_types) + + def get_node_selector(self) -> ResourceTypeSelector: if self.manifest is None or self.graph is None: raise DbtInternalError("manifest and graph must be set to get perform node selection") - return TestSelector( + return ResourceTypeSelector( graph=self.graph, manifest=self.manifest, previous_state=self.previous_state, + resource_types=self.resource_types, ) - def get_runner_type(self, _): + def get_runner_type(self, _) -> Optional[Type[BaseRunner]]: return TestRunner diff --git a/core/dbt/tests/fixtures/project.py b/core/dbt/tests/fixtures/project.py index a12638b16a2..2f458c5b95e 100644 --- a/core/dbt/tests/fixtures/project.py +++ b/core/dbt/tests/fixtures/project.py @@ -428,6 +428,8 @@ def test_config(): # from the pytest fixtures that may be needed in the test functions, including # a 'run_sql' method. class TestProjInfo: + __test__ = False + def __init__( self, project_root, diff --git a/core/dbt/tests/util.py b/core/dbt/tests/util.py index a01ee9b67e2..d745560d2ed 100644 --- a/core/dbt/tests/util.py +++ b/core/dbt/tests/util.py @@ -5,16 +5,19 @@ from contextvars import ContextVar, copy_context from datetime import datetime from io import StringIO -from typing import Any, Dict, List, Optional +from typing import Any, Callable, Dict, List, Optional +from unittest import mock +import pytz import yaml from dbt.adapters.base.relation import BaseRelation from dbt.adapters.factory import Adapter from dbt.cli.main import dbtRunner from dbt.contracts.graph.manifest import Manifest +from dbt.materializations.incremental.microbatch import MicrobatchBuilder from dbt_common.context import _INVOCATION_CONTEXT_VAR, InvocationContext -from dbt_common.events.base_types import EventLevel +from dbt_common.events.base_types import EventLevel, EventMsg from dbt_common.events.functions import ( capture_stdout_logs, fire_event, @@ -73,6 +76,7 @@ def run_dbt( args: Optional[List[str]] = None, expect_pass: bool = True, + callbacks: Optional[List[Callable[[EventMsg], None]]] = None, ): # reset global vars reset_metadata_vars() @@ -90,7 +94,8 @@ def run_dbt( args.extend(["--project-dir", project_dir]) if profiles_dir and "--profiles-dir" not in args: args.extend(["--profiles-dir", profiles_dir]) - dbt = dbtRunner() + dbt = dbtRunner(callbacks=callbacks) + res = dbt.invoke(args) # the exception is immediately raised to be caught in tests @@ -148,7 +153,7 @@ def get_manifest(project_root) -> Optional[Manifest]: if os.path.exists(path): with open(path, "rb") as fp: manifest_mp = fp.read() - manifest: Manifest = Manifest.from_msgpack(manifest_mp) + manifest: Manifest = Manifest.from_msgpack(manifest_mp) # type: ignore[attr-defined] return manifest else: return None @@ -639,3 +644,8 @@ def safe_set_invocation_context(): if invocation_var is None: invocation_var = _INVOCATION_CONTEXT_VAR invocation_var.set(InvocationContext(os.environ)) + + +def patch_microbatch_end_time(dt_str: str): + dt = datetime.strptime(dt_str, "%Y-%m-%d %H:%M:%S").replace(tzinfo=pytz.UTC) + return mock.patch.object(MicrobatchBuilder, "build_end_time", return_value=dt) diff --git a/core/dbt/tracking.py b/core/dbt/tracking.py index 880243e4d6e..7fb62c4bb2d 100644 --- a/core/dbt/tracking.py +++ b/core/dbt/tracking.py @@ -12,6 +12,7 @@ from snowplow_tracker import Emitter, SelfDescribingJson, Subject, Tracker from snowplow_tracker import __version__ as snowplow_version # type: ignore from snowplow_tracker import logger as sp_logger +from snowplow_tracker.events import StructuredEvent from dbt import version as dbt_version from dbt.adapters.exceptions import FailedToConnectError @@ -25,7 +26,8 @@ SendingEvent, TrackingInitializeFailure, ) -from dbt_common.events.functions import fire_event, get_invocation_id +from dbt_common.events.base_types import EventMsg +from dbt_common.events.functions import fire_event, get_invocation_id, msg_to_dict from dbt_common.exceptions import NotImplementedError sp_logger.setLevel(100) @@ -36,6 +38,7 @@ ADAPTER_INFO_SPEC = "iglu:com.dbt/adapter_info/jsonschema/1-0-1" DEPRECATION_WARN_SPEC = "iglu:com.dbt/deprecation_warn/jsonschema/1-0-0" +BEHAVIOR_CHANGE_WARN_SPEC = "iglu:com.dbt/behavior_change_warn/jsonschema/1-0-0" EXPERIMENTAL_PARSER = "iglu:com.dbt/experimental_parser/jsonschema/1-0-0" INVOCATION_ENV_SPEC = "iglu:com.dbt/invocation_env/jsonschema/1-0-0" INVOCATION_SPEC = "iglu:com.dbt/invocation/jsonschema/1-0-2" @@ -47,7 +50,7 @@ RESOURCE_COUNTS = "iglu:com.dbt/resource_counts/jsonschema/1-0-1" RPC_REQUEST_SPEC = "iglu:com.dbt/rpc_request/jsonschema/1-0-1" RUNNABLE_TIMING = "iglu:com.dbt/runnable/jsonschema/1-0-0" -RUN_MODEL_SPEC = "iglu:com.dbt/run_model/jsonschema/1-0-3" +RUN_MODEL_SPEC = "iglu:com.dbt/run_model/jsonschema/1-1-0" PLUGIN_GET_NODES = "iglu:com.dbt/plugin_get_nodes/jsonschema/1-0-0" SNOWPLOW_TRACKER_VERSION = Version(snowplow_version) @@ -215,12 +218,12 @@ def get_dbt_env_context(): def track(user, *args, **kwargs): if user.do_not_track: return - else: - fire_event(SendingEvent(kwargs=str(kwargs))) - try: - tracker.track_struct_event(*args, **kwargs) - except Exception: - fire_event(SendEventFailure()) + + fire_event(SendingEvent(kwargs=str(kwargs))) + try: + tracker.track(StructuredEvent(*args, **kwargs)) + except Exception: + fire_event(SendEventFailure()) def track_project_id(options): @@ -364,6 +367,20 @@ def track_deprecation_warn(options): ) +def track_behavior_change_warn(msg: EventMsg) -> None: + if msg.info.name != "BehaviorChangeEvent" or active_user is None: + return + + context = [SelfDescribingJson(BEHAVIOR_CHANGE_WARN_SPEC, msg_to_dict(msg))] + track( + active_user, + category="dbt", + action=msg.info.name, + label=get_invocation_id(), + context=context, + ) + + def track_invocation_end(invocation_context, result_type=None): data = {"progress": "end", "result_type": result_type, "result": None} data.update(invocation_context) diff --git a/core/dbt/version.py b/core/dbt/version.py index a4a219e9529..841f0e6fca2 100644 --- a/core/dbt/version.py +++ b/core/dbt/version.py @@ -49,7 +49,10 @@ def get_latest_version( return semver.VersionSpecifier.from_version_string(version_string) -def _get_core_msg_lines(installed, latest) -> Tuple[List[List[str]], str]: +def _get_core_msg_lines( + installed: semver.VersionSpecifier, + latest: Optional[semver.VersionSpecifier], +) -> Tuple[List[List[str]], str]: installed_s = installed.to_version_string(skip_matcher=True) installed_line = ["installed", installed_s, ""] update_info = "" @@ -208,7 +211,7 @@ def _get_dbt_plugins_info() -> Iterator[Tuple[str, str]]: except ImportError: # not an adapter continue - yield plugin_name, mod.version # type: ignore + yield plugin_name, mod.version def _get_adapter_plugin_names() -> Iterator[str]: @@ -228,5 +231,5 @@ def _get_adapter_plugin_names() -> Iterator[str]: yield plugin_name -__version__ = "1.9.0a1" +__version__ = "1.9.0b3" installed = get_installed_version() diff --git a/core/setup.py b/core/setup.py index 1491204ae3c..273934d5be8 100644 --- a/core/setup.py +++ b/core/setup.py @@ -2,9 +2,9 @@ import os import sys -if sys.version_info < (3, 8): +if sys.version_info < (3, 9): print("Error: dbt does not support this version of Python.") - print("Please upgrade to Python 3.8 or higher.") + print("Please upgrade to Python 3.9 or higher.") sys.exit(1) @@ -25,7 +25,7 @@ package_name = "dbt-core" -package_version = "1.9.0a1" +package_version = "1.9.0b3" description = """With dbt, data analysts and engineers can build analytics \ the way engineers build applications.""" @@ -59,6 +59,7 @@ "networkx>=2.3,<4.0", "protobuf>=4.0.0,<5", "requests<3.0.0", # should match dbt-common + "snowplow-tracker>=1.0.2,<2.0", # ---- # These packages are major-version-0. Keep upper bounds on upcoming minor versions (which could have breaking changes) # and check compatibility / bump in each new minor version of dbt-core. @@ -68,11 +69,10 @@ # These are major-version-0 packages also maintained by dbt-labs. # Accept patches but avoid automatically updating past a set minor version range. "dbt-extractor>=0.5.0,<=0.6", - "minimal-snowplow-tracker>=0.0.2,<0.1", - "dbt-semantic-interfaces>=0.6.8,<0.7", + "dbt-semantic-interfaces>=0.7.4,<0.8", # Minor versions for these are expected to be backwards-compatible - "dbt-common>=1.3.0,<2.0", - "dbt-adapters>=1.1.1,<2.0", + "dbt-common>=1.11.0,<2.0", + "dbt-adapters>=1.8.0,<2.0", # ---- # Expect compatibility with all new versions of these packages, so lower bounds only. "packaging>20.9", @@ -89,11 +89,10 @@ "Operating System :: Microsoft :: Windows", "Operating System :: MacOS :: MacOS X", "Operating System :: POSIX :: Linux", - "Programming Language :: Python :: 3.8", "Programming Language :: Python :: 3.9", "Programming Language :: Python :: 3.10", "Programming Language :: Python :: 3.11", "Programming Language :: Python :: 3.12", ], - python_requires=">=3.8", + python_requires=">=3.9", ) diff --git a/docker/Dockerfile b/docker/Dockerfile index 5b07514d76b..0cfe9ace811 100644 --- a/docker/Dockerfile +++ b/docker/Dockerfile @@ -8,7 +8,7 @@ RUN apt-get update \ build-essential=12.9 \ ca-certificates=20210119 \ git=1:2.30.2-1+deb11u2 \ - libpq-dev=13.14-0+deb11u1 \ + libpq-dev=13.16-0+deb11u1 \ make=4.3-4.1 \ openssh-client=1:8.4p1-5+deb11u3 \ software-properties-common=0.96.20.2-2.1 \ diff --git a/schemas/dbt/catalog/v1.json b/schemas/dbt/catalog/v1.json index 25c2b25b2b3..f104c5b977f 100644 --- a/schemas/dbt/catalog/v1.json +++ b/schemas/dbt/catalog/v1.json @@ -12,7 +12,7 @@ }, "dbt_version": { "type": "string", - "default": "1.9.0a1" + "default": "1.9.0b2" }, "generated_at": { "type": "string" diff --git a/schemas/dbt/manifest/v10.json b/schemas/dbt/manifest/v10.json index 39a495af9cb..dff60014a78 100644 --- a/schemas/dbt/manifest/v10.json +++ b/schemas/dbt/manifest/v10.json @@ -5689,4 +5689,4 @@ }, "$schema": "http://json-schema.org/draft-07/schema#", "$id": "https://schemas.getdbt.com/dbt/manifest/v10.json" -} +} \ No newline at end of file diff --git a/schemas/dbt/manifest/v11.json b/schemas/dbt/manifest/v11.json index b8dc9e70f8a..43cb42cb157 100644 --- a/schemas/dbt/manifest/v11.json +++ b/schemas/dbt/manifest/v11.json @@ -7060,4 +7060,4 @@ } }, "$id": "https://schemas.getdbt.com/dbt/manifest/v11.json" -} +} \ No newline at end of file diff --git a/schemas/dbt/manifest/v12.json b/schemas/dbt/manifest/v12.json index 0ec6daae1f1..0a8322611e4 100644 --- a/schemas/dbt/manifest/v12.json +++ b/schemas/dbt/manifest/v12.json @@ -13,7 +13,7 @@ }, "dbt_version": { "type": "string", - "default": "1.9.0a1" + "default": "1.9.0b2" }, "generated_at": { "type": "string" @@ -259,6 +259,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -434,6 +443,9 @@ }, "additionalProperties": false }, + "event_time": { + "default": null + }, "delimiter": { "type": "string", "default": "," @@ -537,6 +549,23 @@ "warn_unsupported": { "type": "boolean", "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -567,6 +596,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -654,6 +706,12 @@ "type": "string" } }, + "unrendered_config_call_dict": { + "type": "object", + "propertyNames": { + "type": "string" + } + }, "relation_name": { "anyOf": [ { @@ -869,6 +927,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -1043,6 +1110,9 @@ } }, "additionalProperties": false + }, + "event_time": { + "default": null } }, "additionalProperties": true @@ -1243,6 +1313,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -1417,6 +1496,9 @@ } }, "additionalProperties": false + }, + "event_time": { + "default": null } }, "additionalProperties": true @@ -1506,6 +1588,23 @@ "warn_unsupported": { "type": "boolean", "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -1536,6 +1635,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -1623,6 +1745,12 @@ "type": "string" } }, + "unrendered_config_call_dict": { + "type": "object", + "propertyNames": { + "type": "string" + } + }, "relation_name": { "anyOf": [ { @@ -2114,6 +2242,23 @@ "warn_unsupported": { "type": "boolean", "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -2144,6 +2289,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -2231,6 +2399,12 @@ "type": "string" } }, + "unrendered_config_call_dict": { + "type": "object", + "propertyNames": { + "type": "string" + } + }, "relation_name": { "anyOf": [ { @@ -2584,6 +2758,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -2758,6 +2941,9 @@ } }, "additionalProperties": false + }, + "event_time": { + "default": null } }, "additionalProperties": true @@ -2847,6 +3033,23 @@ "warn_unsupported": { "type": "boolean", "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -2877,6 +3080,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -2964,6 +3190,12 @@ "type": "string" } }, + "unrendered_config_call_dict": { + "type": "object", + "propertyNames": { + "type": "string" + } + }, "relation_name": { "anyOf": [ { @@ -3328,6 +3560,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -3503,6 +3744,9 @@ }, "additionalProperties": false }, + "event_time": { + "default": null + }, "access": { "enum": [ "private", @@ -3599,6 +3843,23 @@ "warn_unsupported": { "type": "boolean", "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -3629,6 +3890,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -3716,6 +4000,12 @@ "type": "string" } }, + "unrendered_config_call_dict": { + "type": "object", + "propertyNames": { + "type": "string" + } + }, "relation_name": { "anyOf": [ { @@ -3954,7 +4244,24 @@ "type": "boolean", "default": true }, - "columns": { + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } + }, + "columns": { "type": "array", "items": { "type": "string" @@ -4182,6 +4489,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -4356,6 +4672,9 @@ } }, "additionalProperties": false + }, + "event_time": { + "default": null } }, "additionalProperties": true @@ -4392,6 +4711,54 @@ "items": { "type": "string" } + }, + "time_spine": { + "anyOf": [ + { + "type": "object", + "title": "TimeSpine", + "properties": { + "standard_granularity_column": { + "type": "string" + }, + "custom_granularities": { + "type": "array", + "items": { + "type": "object", + "title": "CustomGranularity", + "properties": { + "name": { + "type": "string" + }, + "column_name": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + } + }, + "additionalProperties": false, + "required": [ + "name" + ] + } + } + }, + "additionalProperties": false, + "required": [ + "standard_granularity_column" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": false, @@ -4562,6 +4929,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -4736,6 +5112,9 @@ } }, "additionalProperties": false + }, + "event_time": { + "default": null } }, "additionalProperties": true @@ -4825,6 +5204,23 @@ "warn_unsupported": { "type": "boolean", "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -4855,6 +5251,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -4942,6 +5361,12 @@ "type": "string" } }, + "unrendered_config_call_dict": { + "type": "object", + "propertyNames": { + "type": "string" + } + }, "relation_name": { "anyOf": [ { @@ -5433,6 +5858,23 @@ "warn_unsupported": { "type": "boolean", "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -5463,6 +5905,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -5550,6 +6015,12 @@ "type": "string" } }, + "unrendered_config_call_dict": { + "type": "object", + "propertyNames": { + "type": "string" + } + }, "relation_name": { "anyOf": [ { @@ -5964,6 +6435,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -6060,6 +6540,12 @@ { "type": "string" }, + { + "type": "array", + "items": { + "type": "string" + } + }, { "type": "null" } @@ -6133,6 +6619,9 @@ }, "additionalProperties": false }, + "event_time": { + "default": null + }, "strategy": { "anyOf": [ { @@ -6193,6 +6682,68 @@ } ], "default": null + }, + "snapshot_meta_column_names": { + "type": "object", + "title": "SnapshotMetaColumnNames", + "properties": { + "dbt_valid_to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "dbt_valid_from": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "dbt_scd_id": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "dbt_updated_at": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + } + }, + "additionalProperties": false + }, + "dbt_valid_to_current": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true @@ -6282,6 +6833,23 @@ "warn_unsupported": { "type": "boolean", "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -6312,6 +6880,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -6399,6 +6990,12 @@ "type": "string" } }, + "unrendered_config_call_dict": { + "type": "object", + "propertyNames": { + "type": "string" + } + }, "relation_name": { "anyOf": [ { @@ -6759,6 +7356,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -6933,6 +7539,9 @@ } }, "additionalProperties": false + }, + "event_time": { + "default": null } }, "additionalProperties": true @@ -7403,6 +8012,23 @@ "warn_unsupported": { "type": "boolean", "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -7433,6 +8059,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -7475,6 +8124,9 @@ "enabled": { "type": "boolean", "default": true + }, + "event_time": { + "default": null } }, "additionalProperties": true @@ -7509,6 +8161,28 @@ }, "created_at": { "type": "number" + }, + "unrendered_database": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "unrendered_schema": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": false, @@ -9476,6 +10150,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -9651,6 +10334,9 @@ }, "additionalProperties": false }, + "event_time": { + "default": null + }, "delimiter": { "type": "string", "default": "," @@ -9754,6 +10440,23 @@ "warn_unsupported": { "type": "boolean", "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -9784,6 +10487,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -9871,6 +10597,12 @@ "type": "string" } }, + "unrendered_config_call_dict": { + "type": "object", + "propertyNames": { + "type": "string" + } + }, "relation_name": { "anyOf": [ { @@ -10086,6 +10818,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -10260,6 +11001,9 @@ } }, "additionalProperties": false + }, + "event_time": { + "default": null } }, "additionalProperties": true @@ -10460,6 +11204,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -10634,6 +11387,9 @@ } }, "additionalProperties": false + }, + "event_time": { + "default": null } }, "additionalProperties": true @@ -10723,6 +11479,23 @@ "warn_unsupported": { "type": "boolean", "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -10753,6 +11526,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -10840,6 +11636,12 @@ "type": "string" } }, + "unrendered_config_call_dict": { + "type": "object", + "propertyNames": { + "type": "string" + } + }, "relation_name": { "anyOf": [ { @@ -11331,6 +12133,23 @@ "warn_unsupported": { "type": "boolean", "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -11361,6 +12180,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -11448,6 +12290,12 @@ "type": "string" } }, + "unrendered_config_call_dict": { + "type": "object", + "propertyNames": { + "type": "string" + } + }, "relation_name": { "anyOf": [ { @@ -11801,6 +12649,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -11975,6 +12832,9 @@ } }, "additionalProperties": false + }, + "event_time": { + "default": null } }, "additionalProperties": true @@ -12064,6 +12924,23 @@ "warn_unsupported": { "type": "boolean", "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -12094,6 +12971,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -12181,6 +13081,12 @@ "type": "string" } }, + "unrendered_config_call_dict": { + "type": "object", + "propertyNames": { + "type": "string" + } + }, "relation_name": { "anyOf": [ { @@ -12545,6 +13451,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -12720,6 +13635,9 @@ }, "additionalProperties": false }, + "event_time": { + "default": null + }, "access": { "enum": [ "private", @@ -12816,6 +13734,23 @@ "warn_unsupported": { "type": "boolean", "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -12846,6 +13781,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -12933,6 +13891,12 @@ "type": "string" } }, + "unrendered_config_call_dict": { + "type": "object", + "propertyNames": { + "type": "string" + } + }, "relation_name": { "anyOf": [ { @@ -13171,6 +14135,23 @@ "type": "boolean", "default": true }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } + }, "columns": { "type": "array", "items": { @@ -13399,6 +14380,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -13573,6 +14563,9 @@ } }, "additionalProperties": false + }, + "event_time": { + "default": null } }, "additionalProperties": true @@ -13609,6 +14602,54 @@ "items": { "type": "string" } + }, + "time_spine": { + "anyOf": [ + { + "type": "object", + "title": "TimeSpine", + "properties": { + "standard_granularity_column": { + "type": "string" + }, + "custom_granularities": { + "type": "array", + "items": { + "type": "object", + "title": "CustomGranularity", + "properties": { + "name": { + "type": "string" + }, + "column_name": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + } + }, + "additionalProperties": false, + "required": [ + "name" + ] + } + } + }, + "additionalProperties": false, + "required": [ + "standard_granularity_column" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": false, @@ -13779,6 +14820,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -13953,6 +15003,9 @@ } }, "additionalProperties": false + }, + "event_time": { + "default": null } }, "additionalProperties": true @@ -14042,6 +15095,23 @@ "warn_unsupported": { "type": "boolean", "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -14072,6 +15142,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -14159,6 +15252,12 @@ "type": "string" } }, + "unrendered_config_call_dict": { + "type": "object", + "propertyNames": { + "type": "string" + } + }, "relation_name": { "anyOf": [ { @@ -14647,9 +15746,26 @@ "type": "boolean", "default": true }, - "warn_unsupported": { - "type": "boolean", - "default": true + "warn_unsupported": { + "type": "boolean", + "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -14680,6 +15796,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -14767,6 +15906,12 @@ "type": "string" } }, + "unrendered_config_call_dict": { + "type": "object", + "propertyNames": { + "type": "string" + } + }, "relation_name": { "anyOf": [ { @@ -15181,6 +16326,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -15277,6 +16431,12 @@ { "type": "string" }, + { + "type": "array", + "items": { + "type": "string" + } + }, { "type": "null" } @@ -15350,6 +16510,9 @@ }, "additionalProperties": false }, + "event_time": { + "default": null + }, "strategy": { "anyOf": [ { @@ -15410,6 +16573,68 @@ } ], "default": null + }, + "snapshot_meta_column_names": { + "type": "object", + "title": "SnapshotMetaColumnNames", + "properties": { + "dbt_valid_to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "dbt_valid_from": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "dbt_scd_id": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "dbt_updated_at": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + } + }, + "additionalProperties": false + }, + "dbt_valid_to_current": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true @@ -15499,6 +16724,23 @@ "warn_unsupported": { "type": "boolean", "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -15529,6 +16771,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -15616,6 +16881,12 @@ "type": "string" } }, + "unrendered_config_call_dict": { + "type": "object", + "propertyNames": { + "type": "string" + } + }, "relation_name": { "anyOf": [ { @@ -15976,6 +17247,15 @@ ], "default": null }, + "batch_size": { + "default": null + }, + "lookback": { + "default": 1 + }, + "begin": { + "default": null + }, "persist_docs": { "type": "object", "propertyNames": { @@ -16150,6 +17430,9 @@ } }, "additionalProperties": false + }, + "event_time": { + "default": null } }, "additionalProperties": true @@ -16611,6 +17894,23 @@ "warn_unsupported": { "type": "boolean", "default": true + }, + "to": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "to_columns": { + "type": "array", + "items": { + "type": "string" + } } }, "additionalProperties": false, @@ -16641,6 +17941,29 @@ "propertyNames": { "type": "string" } + }, + "granularity": { + "anyOf": [ + { + "enum": [ + "nanosecond", + "microsecond", + "millisecond", + "second", + "minute", + "hour", + "day", + "week", + "month", + "quarter", + "year" + ] + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": true, @@ -16683,6 +18006,9 @@ "enabled": { "type": "boolean", "default": true + }, + "event_time": { + "default": null } }, "additionalProperties": true @@ -16717,6 +18043,28 @@ }, "created_at": { "type": "number" + }, + "unrendered_database": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null + }, + "unrendered_schema": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": false, @@ -18302,6 +19650,23 @@ "type": "null" } ] + }, + "order_by": { + "type": "array", + "items": { + "type": "string" + } + }, + "limit": { + "anyOf": [ + { + "type": "integer" + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": false, @@ -18368,6 +19733,15 @@ "required": [ "export_as" ] + }, + "unrendered_config": { + "type": "object", + "additionalProperties": { + "type": "string" + }, + "propertyNames": { + "type": "string" + } } }, "additionalProperties": false, @@ -19827,6 +21201,23 @@ "type": "null" } ] + }, + "order_by": { + "type": "array", + "items": { + "type": "string" + } + }, + "limit": { + "anyOf": [ + { + "type": "integer" + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": false, @@ -19893,6 +21284,15 @@ "required": [ "export_as" ] + }, + "unrendered_config": { + "type": "object", + "additionalProperties": { + "type": "string" + }, + "propertyNames": { + "type": "string" + } } }, "additionalProperties": false, diff --git a/schemas/dbt/manifest/v5.json b/schemas/dbt/manifest/v5.json index 6c73ab8876b..6b048d7c231 100644 --- a/schemas/dbt/manifest/v5.json +++ b/schemas/dbt/manifest/v5.json @@ -5981,4 +5981,4 @@ }, "$schema": "http://json-schema.org/draft-07/schema#", "$id": "https://schemas.getdbt.com/dbt/manifest/v5.json" -} +} \ No newline at end of file diff --git a/schemas/dbt/manifest/v6.json b/schemas/dbt/manifest/v6.json index 79b627332cf..d835acbab92 100644 --- a/schemas/dbt/manifest/v6.json +++ b/schemas/dbt/manifest/v6.json @@ -6206,4 +6206,4 @@ }, "$schema": "http://json-schema.org/draft-07/schema#", "$id": "https://schemas.getdbt.com/dbt/manifest/v6.json" -} +} \ No newline at end of file diff --git a/schemas/dbt/manifest/v7.json b/schemas/dbt/manifest/v7.json index efb3ff99c89..03884400405 100644 --- a/schemas/dbt/manifest/v7.json +++ b/schemas/dbt/manifest/v7.json @@ -6572,4 +6572,4 @@ }, "$schema": "http://json-schema.org/draft-07/schema#", "$id": "https://schemas.getdbt.com/dbt/manifest/v7.json" -} +} \ No newline at end of file diff --git a/schemas/dbt/manifest/v8.json b/schemas/dbt/manifest/v8.json index d7192ab84c0..ff71790e5f0 100644 --- a/schemas/dbt/manifest/v8.json +++ b/schemas/dbt/manifest/v8.json @@ -4431,4 +4431,4 @@ }, "$schema": "http://json-schema.org/draft-07/schema#", "$id": "https://schemas.getdbt.com/dbt/manifest/v8.json" -} +} \ No newline at end of file diff --git a/schemas/dbt/manifest/v9.json b/schemas/dbt/manifest/v9.json index 28a477367a7..48ede936b47 100644 --- a/schemas/dbt/manifest/v9.json +++ b/schemas/dbt/manifest/v9.json @@ -4962,4 +4962,4 @@ }, "$schema": "http://json-schema.org/draft-07/schema#", "$id": "https://schemas.getdbt.com/dbt/manifest/v9.json" -} +} \ No newline at end of file diff --git a/schemas/dbt/run-results/v6.json b/schemas/dbt/run-results/v6.json index 86b79e1206c..1bf1cf75e83 100644 --- a/schemas/dbt/run-results/v6.json +++ b/schemas/dbt/run-results/v6.json @@ -12,7 +12,7 @@ }, "dbt_version": { "type": "string", - "default": "1.9.0a1" + "default": "1.9.0b2" }, "generated_at": { "type": "string" @@ -54,7 +54,8 @@ "enum": [ "success", "error", - "skipped" + "skipped", + "partial success" ] }, { @@ -178,6 +179,53 @@ "type": "null" } ] + }, + "batch_results": { + "anyOf": [ + { + "type": "object", + "title": "BatchResults", + "properties": { + "successful": { + "type": "array", + "items": { + "type": "array", + "prefixItems": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "maxItems": 2, + "minItems": 2 + } + }, + "failed": { + "type": "array", + "items": { + "type": "array", + "prefixItems": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "maxItems": 2, + "minItems": 2 + } + } + }, + "additionalProperties": false + }, + { + "type": "null" + } + ], + "default": null } }, "additionalProperties": false, diff --git a/schemas/dbt/sources/v3.json b/schemas/dbt/sources/v3.json index 5ade4a90be0..df2784f1a81 100644 --- a/schemas/dbt/sources/v3.json +++ b/schemas/dbt/sources/v3.json @@ -12,7 +12,7 @@ }, "dbt_version": { "type": "string", - "default": "1.9.0a1" + "default": "1.9.0b2" }, "generated_at": { "type": "string" diff --git a/test/setup_db.sh b/test/setup_db.sh index de59bf0fac6..2d877a9e8df 100755 --- a/test/setup_db.sh +++ b/test/setup_db.sh @@ -39,6 +39,7 @@ for i in {1..10}; do done; createdb dbt +psql -c "SELECT version();" psql -c "CREATE ROLE root WITH PASSWORD 'password';" psql -c "ALTER ROLE root WITH LOGIN;" psql -c "GRANT CREATE, CONNECT ON DATABASE dbt TO root WITH GRANT OPTION;" diff --git a/tests/functional/adapter/basic/files.py b/tests/functional/adapter/basic/files.py index 751b01a0b8a..67176babab1 100644 --- a/tests/functional/adapter/basic/files.py +++ b/tests/functional/adapter/basic/files.py @@ -65,8 +65,15 @@ generic_test_view_yml = """ version: 2 + +groups: + - name: my_group + owner: + name: group_owner + models: - name: view_model + group: my_group columns: - name: id data_tests: diff --git a/tests/functional/adapter/basic/test_generic_tests.py b/tests/functional/adapter/basic/test_generic_tests.py index 12c564609ab..d36bc3a0cad 100644 --- a/tests/functional/adapter/basic/test_generic_tests.py +++ b/tests/functional/adapter/basic/test_generic_tests.py @@ -1,6 +1,6 @@ import pytest -from dbt.tests.util import run_dbt +from dbt.tests.util import run_dbt, run_dbt_and_capture from tests.functional.adapter.basic.files import ( base_table_sql, base_view_sql, @@ -58,9 +58,17 @@ def test_generic_tests(self, project): assert len(results) == 2 # test command, all tests - results = run_dbt(["test"]) + results, log_output = run_dbt_and_capture(["test", "--log-format", "json"]) assert len(results) == 3 + result_log_lines = [ + line for line in log_output.split("\n") if "LogTestResult" in line and "group" in line + ] + assert len(result_log_lines) == 1 + assert "my_group" in result_log_lines[0] + assert "group_owner" in result_log_lines[0] + assert "model.generic_tests.view_model" in result_log_lines[0] + class TestGenericTests(BaseGenericTests): pass diff --git a/tests/functional/adapter/hooks/fixtures.py b/tests/functional/adapter/hooks/fixtures.py index c9f1e7dadf8..a7c99a02bc3 100644 --- a/tests/functional/adapter/hooks/fixtures.py +++ b/tests/functional/adapter/hooks/fixtures.py @@ -27,9 +27,9 @@ """ macros__before_and_after = """ -{% macro custom_run_hook(state, target, run_started_at, invocation_id) %} +{% macro custom_run_hook(state, target, run_started_at, invocation_id, table_name="on_run_hook") %} - insert into {{ target.schema }}.on_run_hook ( + insert into {{ target.schema }}.{{ table_name }} ( test_state, target_dbname, target_host, @@ -355,6 +355,26 @@ - not_null """ +properties__model_hooks = """ +version: 2 +models: + - name: hooks + config: + pre_hook: "{{ custom_run_hook('start', target, run_started_at, invocation_id, table_name='on_model_hook') }}" + post_hook: "{{ custom_run_hook('end', target, run_started_at, invocation_id, table_name='on_model_hook') }}" +""" + +properties__model_hooks_list = """ +version: 2 +models: + - name: hooks + config: + pre_hook: + - "{{ custom_run_hook('start', target, run_started_at, invocation_id, table_name='on_model_hook') }}" + post_hook: + - "{{ custom_run_hook('end', target, run_started_at, invocation_id, table_name='on_model_hook') }}" +""" + seeds__example_seed_csv = """a,b,c 1,2,3 4,5,6 diff --git a/tests/functional/adapter/hooks/test_model_hooks.py b/tests/functional/adapter/hooks/test_model_hooks.py index 1a4c01cb506..ff397993f3e 100644 --- a/tests/functional/adapter/hooks/test_model_hooks.py +++ b/tests/functional/adapter/hooks/test_model_hooks.py @@ -6,6 +6,7 @@ from dbt.tests.util import run_dbt, write_file from dbt_common.exceptions import CompilationError from tests.functional.adapter.hooks.fixtures import ( + macros__before_and_after, models__hooked, models__hooks, models__hooks_configured, @@ -13,6 +14,8 @@ models__hooks_kwargs, models__post, models__pre, + properties__model_hooks, + properties__model_hooks_list, properties__seed_models, properties__test_snapshot_models, seeds__example_seed_csv, @@ -257,6 +260,27 @@ def test_hooks_on_seeds(self, project): assert len(res) == 1, "Expected exactly one item" +class TestPrePostModelHooksWithMacros(BaseTestPrePost): + @pytest.fixture(scope="class") + def macros(self): + return {"before-and-after.sql": macros__before_and_after} + + @pytest.fixture(scope="class") + def models(self): + return {"schema.yml": properties__model_hooks, "hooks.sql": models__hooks} + + def test_pre_and_post_run_hooks(self, project, dbt_profile_target): + run_dbt() + self.check_hooks("start", project, dbt_profile_target.get("host", None)) + self.check_hooks("end", project, dbt_profile_target.get("host", None)) + + +class TestPrePostModelHooksListWithMacros(TestPrePostModelHooksWithMacros): + @pytest.fixture(scope="class") + def models(self): + return {"schema.yml": properties__model_hooks_list, "hooks.sql": models__hooks} + + class TestHooksRefsOnSeeds: """ This should not succeed, and raise an explicit error diff --git a/tests/functional/adapter/hooks/test_on_run_hooks.py b/tests/functional/adapter/hooks/test_on_run_hooks.py new file mode 100644 index 00000000000..b85784be3cf --- /dev/null +++ b/tests/functional/adapter/hooks/test_on_run_hooks.py @@ -0,0 +1,244 @@ +import pytest + +from dbt.artifacts.schemas.results import RunStatus +from dbt.contracts.graph.nodes import HookNode +from dbt.tests.util import get_artifact, run_dbt_and_capture + + +class Test__StartHookFail__FlagIsNone__ModelFail: + @pytest.fixture(scope="class") + def flags(self): + return {} + + @pytest.fixture(scope="class") + def project_config_update(self, flags): + return { + "on-run-start": [ + "create table {{ target.schema }}.my_hook_table ( id int )", # success + "drop table {{ target.schema }}.my_hook_table", # success + "insert into {{ target.schema }}.my_hook_table (id) values (1, 2, 3)", # fail + "create table {{ target.schema }}.my_hook_table ( id int )", # skip + ], + "flags": flags, + } + + @pytest.fixture(scope="class") + def models(self): + return { + "my_model.sql": "select * from {{ target.schema }}.my_hook_table" + " union all " + "select * from {{ target.schema }}.my_end_table" + } + + @pytest.fixture(scope="class") + def log_counts(self): + return "PASS=2 WARN=0 ERROR=2 SKIP=1 TOTAL=5" + + @pytest.fixture(scope="class") + def my_model_run_status(self): + return RunStatus.Error + + def test_results(self, project, log_counts, my_model_run_status): + results, log_output = run_dbt_and_capture(["run"], expect_pass=False) + + expected_results = [ + ("operation.test.test-on-run-start-0", RunStatus.Success), + ("operation.test.test-on-run-start-1", RunStatus.Success), + ("operation.test.test-on-run-start-2", RunStatus.Error), + ("operation.test.test-on-run-start-3", RunStatus.Skipped), + ("model.test.my_model", my_model_run_status), + ] + + assert [(result.node.unique_id, result.status) for result in results] == expected_results + assert [ + (result.node.unique_id, result.node.node_info["node_status"]) + for result in results + if isinstance(result.node, HookNode) + ] == [(id, str(status)) for id, status in expected_results if id.startswith("operation")] + + for result in results: + if result.status == RunStatus.Skipped: + continue + + timing_keys = [timing.name for timing in result.timing] + assert timing_keys == ["compile", "execute"] + + assert log_counts in log_output + assert "4 project hooks, 1 view model" in log_output + + run_results = get_artifact(project.project_root, "target", "run_results.json") + assert [ + (result["unique_id"], result["status"]) for result in run_results["results"] + ] == expected_results + assert ( + f'relation "{project.test_schema}.my_hook_table" does not exist' + in run_results["results"][2]["message"] + ) + + +class Test__StartHookFail__FlagIsFalse__ModelFail(Test__StartHookFail__FlagIsNone__ModelFail): + @pytest.fixture(scope="class") + def flags(self): + return {"skip_nodes_if_on_run_start_fails": False} + + +class Test__StartHookFail__FlagIsTrue__ModelSkipped(Test__StartHookFail__FlagIsNone__ModelFail): + @pytest.fixture(scope="class") + def flags(self): + return {"skip_nodes_if_on_run_start_fails": True} + + @pytest.fixture(scope="class") + def log_counts(self): + return "PASS=2 WARN=0 ERROR=1 SKIP=2 TOTAL=5" + + @pytest.fixture(scope="class") + def my_model_run_status(self): + return RunStatus.Skipped + + +class Test__ModelPass__EndHookFail: + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "on-run-end": [ + "create table {{ target.schema }}.my_hook_table ( id int )", # success + "drop table {{ target.schema }}.my_hook_table", # success + "insert into {{ target.schema }}.my_hook_table (id) values (1, 2, 3)", # fail + "create table {{ target.schema }}.my_hook_table ( id int )", # skip + ], + } + + @pytest.fixture(scope="class") + def models(self): + return {"my_model.sql": "select 1"} + + def test_results(self, project): + results, log_output = run_dbt_and_capture(["run"], expect_pass=False) + + expected_results = [ + ("model.test.my_model", RunStatus.Success), + ("operation.test.test-on-run-end-0", RunStatus.Success), + ("operation.test.test-on-run-end-1", RunStatus.Success), + ("operation.test.test-on-run-end-2", RunStatus.Error), + ("operation.test.test-on-run-end-3", RunStatus.Skipped), + ] + + assert [(result.node.unique_id, result.status) for result in results] == expected_results + assert "PASS=3 WARN=0 ERROR=1 SKIP=1 TOTAL=5" in log_output + assert "4 project hooks, 1 view model" in log_output + + run_results = get_artifact(project.project_root, "target", "run_results.json") + assert [ + (result["unique_id"], result["status"]) for result in run_results["results"] + ] == expected_results + assert ( + f'relation "{project.test_schema}.my_hook_table" does not exist' + in run_results["results"][3]["message"] + ) + + +class Test__SelectorEmpty__NoHooksRan: + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "on-run-start": [ + "create table {{ target.schema }}.my_hook_table ( id int )", # success + "drop table {{ target.schema }}.my_hook_table", # success + ], + "on-run-end": [ + "create table {{ target.schema }}.my_hook_table ( id int )", # success + "drop table {{ target.schema }}.my_hook_table", # success + ], + } + + @pytest.fixture(scope="class") + def models(self): + return {"my_model.sql": "select 1"} + + def test_results(self, project): + results, log_output = run_dbt_and_capture( + ["--debug", "run", "--select", "tag:no_such_tag", "--log-format", "json"] + ) + + assert results.results == [] + assert ( + "The selection criterion 'tag:no_such_tag' does not match any enabled nodes" + in log_output + ) + + run_results = get_artifact(project.project_root, "target", "run_results.json") + assert run_results["results"] == [] + + +class Test__HookContext__HookSuccess: + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "on-run-start": [ + "select 1 as id", # success + "select 1 as id", # success + ], + "on-run-end": [ + '{{ log("Num Results in context: " ~ results|length)}}' + "{{ output_thread_ids(results) }}", + ], + } + + @pytest.fixture(scope="class") + def macros(self): + return { + "log.sql": """ +{% macro output_thread_ids(results) %} + {% for result in results %} + {{ log("Thread ID: " ~ result.thread_id) }} + {% endfor %} +{% endmacro %} +""" + } + + @pytest.fixture(scope="class") + def models(self): + return {"my_model.sql": "select 1"} + + def test_results_in_context_success(self, project): + results, log_output = run_dbt_and_capture(["--debug", "run"]) + assert "Thread ID: " in log_output + assert "Thread ID: main" not in log_output + assert results[0].thread_id == "main" # hook still exists in run results + assert "Num Results in context: 1" in log_output # only model given hook was successful + + +class Test__HookContext__HookFail: + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "on-run-start": [ + "select a as id", # fail + ], + "on-run-end": [ + '{{ log("Num Results in context: " ~ results|length)}}' + "{{ output_thread_ids(results) }}", + ], + } + + @pytest.fixture(scope="class") + def macros(self): + return { + "log.sql": """ +{% macro output_thread_ids(results) %} + {% for result in results %} + {{ log("Thread ID: " ~ result.thread_id) }} + {% endfor %} +{% endmacro %} +""" + } + + @pytest.fixture(scope="class") + def models(self): + return {"my_model.sql": "select 1"} + + def test_results_in_context_hook_fail(self, project): + results, log_output = run_dbt_and_capture(["--debug", "run"], expect_pass=False) + assert "Thread ID: main" in log_output + assert results[0].thread_id == "main" + assert "Num Results in context: 2" in log_output # failed hook and model diff --git a/tests/functional/adapter/hooks/test_run_hooks.py b/tests/functional/adapter/hooks/test_run_hooks.py deleted file mode 100644 index f8bec5c6aeb..00000000000 --- a/tests/functional/adapter/hooks/test_run_hooks.py +++ /dev/null @@ -1,161 +0,0 @@ -import os -from pathlib import Path - -import pytest - -from dbt.tests.util import check_table_does_not_exist, run_dbt -from tests.functional.adapter.hooks.fixtures import ( - macros__before_and_after, - macros__hook, - macros_missing_column, - models__hooks, - models__missing_column, - seeds__example_seed_csv, -) - - -class TestPrePostRunHooks(object): - @pytest.fixture(scope="function") - def setUp(self, project): - project.run_sql_file(project.test_data_dir / Path("seed_run.sql")) - project.run_sql(f"drop table if exists { project.test_schema }.schemas") - project.run_sql(f"drop table if exists { project.test_schema }.db_schemas") - os.environ["TERM_TEST"] = "TESTING" - - @pytest.fixture(scope="class") - def macros(self): - return {"hook.sql": macros__hook, "before-and-after.sql": macros__before_and_after} - - @pytest.fixture(scope="class") - def models(self): - return {"hooks.sql": models__hooks} - - @pytest.fixture(scope="class") - def seeds(self): - return {"example_seed.csv": seeds__example_seed_csv} - - @pytest.fixture(scope="class") - def project_config_update(self): - return { - # The create and drop table statements here validate that these hooks run - # in the same order that they are defined. Drop before create is an error. - # Also check that the table does not exist below. - "on-run-start": [ - "{{ custom_run_hook('start', target, run_started_at, invocation_id) }}", - "create table {{ target.schema }}.start_hook_order_test ( id int )", - "drop table {{ target.schema }}.start_hook_order_test", - "{{ log(env_var('TERM_TEST'), info=True) }}", - ], - "on-run-end": [ - "{{ custom_run_hook('end', target, run_started_at, invocation_id) }}", - "create table {{ target.schema }}.end_hook_order_test ( id int )", - "drop table {{ target.schema }}.end_hook_order_test", - "create table {{ target.schema }}.schemas ( schema text )", - "insert into {{ target.schema }}.schemas (schema) values {% for schema in schemas %}( '{{ schema }}' ){% if not loop.last %},{% endif %}{% endfor %}", - "create table {{ target.schema }}.db_schemas ( db text, schema text )", - "insert into {{ target.schema }}.db_schemas (db, schema) values {% for db, schema in database_schemas %}('{{ db }}', '{{ schema }}' ){% if not loop.last %},{% endif %}{% endfor %}", - ], - "seeds": { - "quote_columns": False, - }, - } - - def get_ctx_vars(self, state, project): - fields = [ - "test_state", - "target_dbname", - "target_host", - "target_name", - "target_schema", - "target_threads", - "target_type", - "target_user", - "target_pass", - "run_started_at", - "invocation_id", - "thread_id", - ] - field_list = ", ".join(['"{}"'.format(f) for f in fields]) - query = f"select {field_list} from {project.test_schema}.on_run_hook where test_state = '{state}'" - - vals = project.run_sql(query, fetch="all") - assert len(vals) != 0, "nothing inserted into on_run_hook table" - assert len(vals) == 1, "too many rows in hooks table" - ctx = dict([(k, v) for (k, v) in zip(fields, vals[0])]) - - return ctx - - def assert_used_schemas(self, project): - schemas_query = "select * from {}.schemas".format(project.test_schema) - results = project.run_sql(schemas_query, fetch="all") - assert len(results) == 1 - assert results[0][0] == project.test_schema - - db_schemas_query = "select * from {}.db_schemas".format(project.test_schema) - results = project.run_sql(db_schemas_query, fetch="all") - assert len(results) == 1 - assert results[0][0] == project.database - assert results[0][1] == project.test_schema - - def check_hooks(self, state, project, host): - ctx = self.get_ctx_vars(state, project) - - assert ctx["test_state"] == state - assert ctx["target_dbname"] == "dbt" - assert ctx["target_host"] == host - assert ctx["target_name"] == "default" - assert ctx["target_schema"] == project.test_schema - assert ctx["target_threads"] == 4 - assert ctx["target_type"] == "postgres" - assert ctx["target_user"] == "root" - assert ctx["target_pass"] == "" - - assert ( - ctx["run_started_at"] is not None and len(ctx["run_started_at"]) > 0 - ), "run_started_at was not set" - assert ( - ctx["invocation_id"] is not None and len(ctx["invocation_id"]) > 0 - ), "invocation_id was not set" - assert ctx["thread_id"].startswith("Thread-") or ctx["thread_id"] == "MainThread" - - def test_pre_and_post_run_hooks(self, setUp, project, dbt_profile_target): - run_dbt(["run"]) - - self.check_hooks("start", project, dbt_profile_target.get("host", None)) - self.check_hooks("end", project, dbt_profile_target.get("host", None)) - - check_table_does_not_exist(project.adapter, "start_hook_order_test") - check_table_does_not_exist(project.adapter, "end_hook_order_test") - self.assert_used_schemas(project) - - def test_pre_and_post_seed_hooks(self, setUp, project, dbt_profile_target): - run_dbt(["seed"]) - - self.check_hooks("start", project, dbt_profile_target.get("host", None)) - self.check_hooks("end", project, dbt_profile_target.get("host", None)) - - check_table_does_not_exist(project.adapter, "start_hook_order_test") - check_table_does_not_exist(project.adapter, "end_hook_order_test") - self.assert_used_schemas(project) - - -class TestAfterRunHooks(object): - @pytest.fixture(scope="class") - def macros(self): - return {"temp_macro.sql": macros_missing_column} - - @pytest.fixture(scope="class") - def models(self): - return {"test_column.sql": models__missing_column} - - @pytest.fixture(scope="class") - def project_config_update(self): - return { - # The create and drop table statements here validate that these hooks run - # in the same order that they are defined. Drop before create is an error. - # Also check that the table does not exist below. - "on-run-start": "- {{ export_table_check() }}" - } - - def test_missing_column_pre_hook(self, project): - run_dbt(["run"], expect_pass=False) diff --git a/tests/functional/adapter/incremental/test_incremental_merge_exclude_columns.py b/tests/functional/adapter/incremental/test_incremental_merge_exclude_columns.py index 7597865154c..a56634e0906 100644 --- a/tests/functional/adapter/incremental/test_incremental_merge_exclude_columns.py +++ b/tests/functional/adapter/incremental/test_incremental_merge_exclude_columns.py @@ -115,3 +115,7 @@ def test__merge_exclude_columns(self, project): update_sql_file=None, ) self.check_scenario_correctness(expected_fields, test_case_fields, project) + + +class TestMergeExcludeColumns(BaseMergeExcludeColumns): + pass diff --git a/tests/functional/adapter/simple_snapshot/fixtures.py b/tests/functional/adapter/simple_snapshot/fixtures.py new file mode 100644 index 00000000000..cec28a7d64d --- /dev/null +++ b/tests/functional/adapter/simple_snapshot/fixtures.py @@ -0,0 +1,430 @@ +create_seed_sql = """ +create table {schema}.seed ( + id INTEGER, + first_name VARCHAR(50), + last_name VARCHAR(50), + email VARCHAR(50), + gender VARCHAR(50), + ip_address VARCHAR(20), + updated_at TIMESTAMP +); +""" + +create_snapshot_expected_sql = """ +create table {schema}.snapshot_expected ( + id INTEGER, + first_name VARCHAR(50), + last_name VARCHAR(50), + email VARCHAR(50), + gender VARCHAR(50), + ip_address VARCHAR(20), + + -- snapshotting fields + updated_at TIMESTAMP, + test_valid_from TIMESTAMP, + test_valid_to TIMESTAMP, + test_scd_id TEXT, + test_updated_at TIMESTAMP +); +""" + + +seed_insert_sql = """ +-- seed inserts +-- use the same email for two users to verify that duplicated check_cols values +-- are handled appropriately +insert into {schema}.seed (id, first_name, last_name, email, gender, ip_address, updated_at) values +(1, 'Judith', 'Kennedy', '(not provided)', 'Female', '54.60.24.128', '2015-12-24 12:19:28'), +(2, 'Arthur', 'Kelly', '(not provided)', 'Male', '62.56.24.215', '2015-10-28 16:22:15'), +(3, 'Rachel', 'Moreno', 'rmoreno2@msu.edu', 'Female', '31.222.249.23', '2016-04-05 02:05:30'), +(4, 'Ralph', 'Turner', 'rturner3@hp.com', 'Male', '157.83.76.114', '2016-08-08 00:06:51'), +(5, 'Laura', 'Gonzales', 'lgonzales4@howstuffworks.com', 'Female', '30.54.105.168', '2016-09-01 08:25:38'), +(6, 'Katherine', 'Lopez', 'klopez5@yahoo.co.jp', 'Female', '169.138.46.89', '2016-08-30 18:52:11'), +(7, 'Jeremy', 'Hamilton', 'jhamilton6@mozilla.org', 'Male', '231.189.13.133', '2016-07-17 02:09:46'), +(8, 'Heather', 'Rose', 'hrose7@goodreads.com', 'Female', '87.165.201.65', '2015-12-29 22:03:56'), +(9, 'Gregory', 'Kelly', 'gkelly8@trellian.com', 'Male', '154.209.99.7', '2016-03-24 21:18:16'), +(10, 'Rachel', 'Lopez', 'rlopez9@themeforest.net', 'Female', '237.165.82.71', '2016-08-20 15:44:49'), +(11, 'Donna', 'Welch', 'dwelcha@shutterfly.com', 'Female', '103.33.110.138', '2016-02-27 01:41:48'), +(12, 'Russell', 'Lawrence', 'rlawrenceb@qq.com', 'Male', '189.115.73.4', '2016-06-11 03:07:09'), +(13, 'Michelle', 'Montgomery', 'mmontgomeryc@scientificamerican.com', 'Female', '243.220.95.82', '2016-06-18 16:27:19'), +(14, 'Walter', 'Castillo', 'wcastillod@pagesperso-orange.fr', 'Male', '71.159.238.196', '2016-10-06 01:55:44'), +(15, 'Robin', 'Mills', 'rmillse@vkontakte.ru', 'Female', '172.190.5.50', '2016-10-31 11:41:21'), +(16, 'Raymond', 'Holmes', 'rholmesf@usgs.gov', 'Male', '148.153.166.95', '2016-10-03 08:16:38'), +(17, 'Gary', 'Bishop', 'gbishopg@plala.or.jp', 'Male', '161.108.182.13', '2016-08-29 19:35:20'), +(18, 'Anna', 'Riley', 'arileyh@nasa.gov', 'Female', '253.31.108.22', '2015-12-11 04:34:27'), +(19, 'Sarah', 'Knight', 'sknighti@foxnews.com', 'Female', '222.220.3.177', '2016-09-26 00:49:06'), +(20, 'Phyllis', 'Fox', null, 'Female', '163.191.232.95', '2016-08-21 10:35:19'); +""" + + +populate_snapshot_expected_sql = """ +-- populate snapshot table +insert into {schema}.snapshot_expected ( + id, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + test_valid_from, + test_valid_to, + test_updated_at, + test_scd_id +) + +select + id, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + -- fields added by snapshotting + updated_at as test_valid_from, + null::timestamp as test_valid_to, + updated_at as test_updated_at, + md5(id || '-' || first_name || '|' || updated_at::text) as test_scd_id +from {schema}.seed; +""" + +populate_snapshot_expected_valid_to_current_sql = """ +-- populate snapshot table +insert into {schema}.snapshot_expected ( + id, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + test_valid_from, + test_valid_to, + test_updated_at, + test_scd_id +) + +select + id, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + -- fields added by snapshotting + updated_at as test_valid_from, + date('2099-12-31') as test_valid_to, + updated_at as test_updated_at, + md5(id || '-' || first_name || '|' || updated_at::text) as test_scd_id +from {schema}.seed; +""" + +snapshot_actual_sql = """ +{% snapshot snapshot_actual %} + + {{ + config( + unique_key='id || ' ~ "'-'" ~ ' || first_name', + ) + }} + + select * from {{target.database}}.{{target.schema}}.seed + +{% endsnapshot %} +""" + +snapshots_yml = """ +snapshots: + - name: snapshot_actual + config: + strategy: timestamp + updated_at: updated_at + snapshot_meta_column_names: + dbt_valid_to: test_valid_to + dbt_valid_from: test_valid_from + dbt_scd_id: test_scd_id + dbt_updated_at: test_updated_at +""" + +snapshots_no_column_names_yml = """ +snapshots: + - name: snapshot_actual + config: + strategy: timestamp + updated_at: updated_at +""" + +ref_snapshot_sql = """ +select * from {{ ref('snapshot_actual') }} +""" + + +invalidate_sql = """ +-- update records 11 - 21. Change email and updated_at field +update {schema}.seed set + updated_at = updated_at + interval '1 hour', + email = case when id = 20 then 'pfoxj@creativecommons.org' else 'new_' || email end +where id >= 10 and id <= 20; + + +-- invalidate records 11 - 21 +update {schema}.snapshot_expected set + test_valid_to = updated_at + interval '1 hour' +where id >= 10 and id <= 20; + +""" + +update_sql = """ +-- insert v2 of the 11 - 21 records + +insert into {schema}.snapshot_expected ( + id, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + test_valid_from, + test_valid_to, + test_updated_at, + test_scd_id +) + +select + id, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + -- fields added by snapshotting + updated_at as test_valid_from, + null::timestamp as test_valid_to, + updated_at as test_updated_at, + md5(id || '-' || first_name || '|' || updated_at::text) as test_scd_id +from {schema}.seed +where id >= 10 and id <= 20; +""" + +# valid_to_current fixtures + +snapshots_valid_to_current_yml = """ +snapshots: + - name: snapshot_actual + config: + strategy: timestamp + updated_at: updated_at + dbt_valid_to_current: "date('2099-12-31')" + snapshot_meta_column_names: + dbt_valid_to: test_valid_to + dbt_valid_from: test_valid_from + dbt_scd_id: test_scd_id + dbt_updated_at: test_updated_at +""" + +update_with_current_sql = """ +-- insert v2 of the 11 - 21 records + +insert into {schema}.snapshot_expected ( + id, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + test_valid_from, + test_valid_to, + test_updated_at, + test_scd_id +) + +select + id, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + -- fields added by snapshotting + updated_at as test_valid_from, + date('2099-12-31') as test_valid_to, + updated_at as test_updated_at, + md5(id || '-' || first_name || '|' || updated_at::text) as test_scd_id +from {schema}.seed +where id >= 10 and id <= 20; +""" + + +# multi-key snapshot fixtures + +create_multi_key_seed_sql = """ +create table {schema}.seed ( + id1 INTEGER, + id2 INTEGER, + first_name VARCHAR(50), + last_name VARCHAR(50), + email VARCHAR(50), + gender VARCHAR(50), + ip_address VARCHAR(20), + updated_at TIMESTAMP +); +""" + + +create_multi_key_snapshot_expected_sql = """ +create table {schema}.snapshot_expected ( + id1 INTEGER, + id2 INTEGER, + first_name VARCHAR(50), + last_name VARCHAR(50), + email VARCHAR(50), + gender VARCHAR(50), + ip_address VARCHAR(20), + + -- snapshotting fields + updated_at TIMESTAMP, + test_valid_from TIMESTAMP, + test_valid_to TIMESTAMP, + test_scd_id TEXT, + test_updated_at TIMESTAMP +); +""" + +seed_multi_key_insert_sql = """ +-- seed inserts +-- use the same email for two users to verify that duplicated check_cols values +-- are handled appropriately +insert into {schema}.seed (id1, id2, first_name, last_name, email, gender, ip_address, updated_at) values +(1, 100, 'Judith', 'Kennedy', '(not provided)', 'Female', '54.60.24.128', '2015-12-24 12:19:28'), +(2, 200, 'Arthur', 'Kelly', '(not provided)', 'Male', '62.56.24.215', '2015-10-28 16:22:15'), +(3, 300, 'Rachel', 'Moreno', 'rmoreno2@msu.edu', 'Female', '31.222.249.23', '2016-04-05 02:05:30'), +(4, 400, 'Ralph', 'Turner', 'rturner3@hp.com', 'Male', '157.83.76.114', '2016-08-08 00:06:51'), +(5, 500, 'Laura', 'Gonzales', 'lgonzales4@howstuffworks.com', 'Female', '30.54.105.168', '2016-09-01 08:25:38'), +(6, 600, 'Katherine', 'Lopez', 'klopez5@yahoo.co.jp', 'Female', '169.138.46.89', '2016-08-30 18:52:11'), +(7, 700, 'Jeremy', 'Hamilton', 'jhamilton6@mozilla.org', 'Male', '231.189.13.133', '2016-07-17 02:09:46'), +(8, 800, 'Heather', 'Rose', 'hrose7@goodreads.com', 'Female', '87.165.201.65', '2015-12-29 22:03:56'), +(9, 900, 'Gregory', 'Kelly', 'gkelly8@trellian.com', 'Male', '154.209.99.7', '2016-03-24 21:18:16'), +(10, 1000, 'Rachel', 'Lopez', 'rlopez9@themeforest.net', 'Female', '237.165.82.71', '2016-08-20 15:44:49'), +(11, 1100, 'Donna', 'Welch', 'dwelcha@shutterfly.com', 'Female', '103.33.110.138', '2016-02-27 01:41:48'), +(12, 1200, 'Russell', 'Lawrence', 'rlawrenceb@qq.com', 'Male', '189.115.73.4', '2016-06-11 03:07:09'), +(13, 1300, 'Michelle', 'Montgomery', 'mmontgomeryc@scientificamerican.com', 'Female', '243.220.95.82', '2016-06-18 16:27:19'), +(14, 1400, 'Walter', 'Castillo', 'wcastillod@pagesperso-orange.fr', 'Male', '71.159.238.196', '2016-10-06 01:55:44'), +(15, 1500, 'Robin', 'Mills', 'rmillse@vkontakte.ru', 'Female', '172.190.5.50', '2016-10-31 11:41:21'), +(16, 1600, 'Raymond', 'Holmes', 'rholmesf@usgs.gov', 'Male', '148.153.166.95', '2016-10-03 08:16:38'), +(17, 1700, 'Gary', 'Bishop', 'gbishopg@plala.or.jp', 'Male', '161.108.182.13', '2016-08-29 19:35:20'), +(18, 1800, 'Anna', 'Riley', 'arileyh@nasa.gov', 'Female', '253.31.108.22', '2015-12-11 04:34:27'), +(19, 1900, 'Sarah', 'Knight', 'sknighti@foxnews.com', 'Female', '222.220.3.177', '2016-09-26 00:49:06'), +(20, 2000, 'Phyllis', 'Fox', null, 'Female', '163.191.232.95', '2016-08-21 10:35:19'); +""" + +populate_multi_key_snapshot_expected_sql = """ +-- populate snapshot table +insert into {schema}.snapshot_expected ( + id1, + id2, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + test_valid_from, + test_valid_to, + test_updated_at, + test_scd_id +) + +select + id1, + id2, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + -- fields added by snapshotting + updated_at as test_valid_from, + null::timestamp as test_valid_to, + updated_at as test_updated_at, + md5(id1::text || '|' || id2::text || '|' || updated_at::text) as test_scd_id +from {schema}.seed; +""" + +model_seed_sql = """ +select * from {{target.database}}.{{target.schema}}.seed +""" + +snapshots_multi_key_yml = """ +snapshots: + - name: snapshot_actual + relation: "ref('seed')" + config: + strategy: timestamp + updated_at: updated_at + unique_key: + - id1 + - id2 + snapshot_meta_column_names: + dbt_valid_to: test_valid_to + dbt_valid_from: test_valid_from + dbt_scd_id: test_scd_id + dbt_updated_at: test_updated_at +""" + +invalidate_multi_key_sql = """ +-- update records 11 - 21. Change email and updated_at field +update {schema}.seed set + updated_at = updated_at + interval '1 hour', + email = case when id1 = 20 then 'pfoxj@creativecommons.org' else 'new_' || email end +where id1 >= 10 and id1 <= 20; + + +-- invalidate records 11 - 21 +update {schema}.snapshot_expected set + test_valid_to = updated_at + interval '1 hour' +where id1 >= 10 and id1 <= 20; + +""" + +update_multi_key_sql = """ +-- insert v2 of the 11 - 21 records + +insert into {schema}.snapshot_expected ( + id1, + id2, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + test_valid_from, + test_valid_to, + test_updated_at, + test_scd_id +) + +select + id1, + id2, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + -- fields added by snapshotting + updated_at as test_valid_from, + null::timestamp as test_valid_to, + updated_at as test_updated_at, + md5(id1::text || '|' || id2::text || '|' || updated_at::text) as test_scd_id +from {schema}.seed +where id1 >= 10 and id1 <= 20; +""" diff --git a/tests/functional/adapter/simple_snapshot/test_various_configs.py b/tests/functional/adapter/simple_snapshot/test_various_configs.py new file mode 100644 index 00000000000..d288d2934df --- /dev/null +++ b/tests/functional/adapter/simple_snapshot/test_various_configs.py @@ -0,0 +1,277 @@ +import datetime + +import pytest + +from dbt.tests.util import ( + check_relations_equal, + get_manifest, + run_dbt, + run_dbt_and_capture, + run_sql_with_adapter, + update_config_file, +) +from tests.functional.adapter.simple_snapshot.fixtures import ( + create_multi_key_seed_sql, + create_multi_key_snapshot_expected_sql, + create_seed_sql, + create_snapshot_expected_sql, + invalidate_multi_key_sql, + invalidate_sql, + model_seed_sql, + populate_multi_key_snapshot_expected_sql, + populate_snapshot_expected_sql, + populate_snapshot_expected_valid_to_current_sql, + ref_snapshot_sql, + seed_insert_sql, + seed_multi_key_insert_sql, + snapshot_actual_sql, + snapshots_multi_key_yml, + snapshots_no_column_names_yml, + snapshots_valid_to_current_yml, + snapshots_yml, + update_multi_key_sql, + update_sql, + update_with_current_sql, +) + + +class BaseSnapshotColumnNames: + @pytest.fixture(scope="class") + def snapshots(self): + return {"snapshot.sql": snapshot_actual_sql} + + @pytest.fixture(scope="class") + def models(self): + return { + "snapshots.yml": snapshots_yml, + "ref_snapshot.sql": ref_snapshot_sql, + } + + def test_snapshot_column_names(self, project): + project.run_sql(create_seed_sql) + project.run_sql(create_snapshot_expected_sql) + project.run_sql(seed_insert_sql) + project.run_sql(populate_snapshot_expected_sql) + + results = run_dbt(["snapshot"]) + assert len(results) == 1 + + project.run_sql(invalidate_sql) + project.run_sql(update_sql) + + results = run_dbt(["snapshot"]) + assert len(results) == 1 + + # run_dbt(["test"]) + check_relations_equal(project.adapter, ["snapshot_actual", "snapshot_expected"]) + + +class TestSnapshotColumnNames(BaseSnapshotColumnNames): + pass + + +class BaseSnapshotColumnNamesFromDbtProject: + @pytest.fixture(scope="class") + def snapshots(self): + return {"snapshot.sql": snapshot_actual_sql} + + @pytest.fixture(scope="class") + def models(self): + return { + "snapshots.yml": snapshots_no_column_names_yml, + "ref_snapshot.sql": ref_snapshot_sql, + } + + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "snapshots": { + "test": { + "+snapshot_meta_column_names": { + "dbt_valid_to": "test_valid_to", + "dbt_valid_from": "test_valid_from", + "dbt_scd_id": "test_scd_id", + "dbt_updated_at": "test_updated_at", + } + } + } + } + + def test_snapshot_column_names_from_project(self, project): + project.run_sql(create_seed_sql) + project.run_sql(create_snapshot_expected_sql) + project.run_sql(seed_insert_sql) + project.run_sql(populate_snapshot_expected_sql) + + results = run_dbt(["snapshot"]) + assert len(results) == 1 + + project.run_sql(invalidate_sql) + project.run_sql(update_sql) + + results = run_dbt(["snapshot"]) + assert len(results) == 1 + + # run_dbt(["test"]) + check_relations_equal(project.adapter, ["snapshot_actual", "snapshot_expected"]) + + +class TestSnapshotColumnNamesFromDbtProject(BaseSnapshotColumnNamesFromDbtProject): + pass + + +class BaseSnapshotInvalidColumnNames: + @pytest.fixture(scope="class") + def snapshots(self): + return {"snapshot.sql": snapshot_actual_sql} + + @pytest.fixture(scope="class") + def models(self): + return { + "snapshots.yml": snapshots_no_column_names_yml, + "ref_snapshot.sql": ref_snapshot_sql, + } + + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "snapshots": { + "test": { + "+snapshot_meta_column_names": { + "dbt_valid_to": "test_valid_to", + "dbt_valid_from": "test_valid_from", + "dbt_scd_id": "test_scd_id", + "dbt_updated_at": "test_updated_at", + } + } + } + } + + def test_snapshot_invalid_column_names(self, project): + project.run_sql(create_seed_sql) + project.run_sql(create_snapshot_expected_sql) + project.run_sql(seed_insert_sql) + project.run_sql(populate_snapshot_expected_sql) + + results = run_dbt(["snapshot"]) + assert len(results) == 1 + manifest = get_manifest(project.project_root) + snapshot_node = manifest.nodes["snapshot.test.snapshot_actual"] + snapshot_node.config.snapshot_meta_column_names == { + "dbt_valid_to": "test_valid_to", + "dbt_valid_from": "test_valid_from", + "dbt_scd_id": "test_scd_id", + "dbt_updated_at": "test_updated_at", + } + + project.run_sql(invalidate_sql) + project.run_sql(update_sql) + + # Change snapshot_meta_columns and look for an error + different_columns = { + "snapshots": { + "test": { + "+snapshot_meta_column_names": { + "dbt_valid_to": "test_valid_to", + "dbt_updated_at": "test_updated_at", + } + } + } + } + update_config_file(different_columns, "dbt_project.yml") + + results, log_output = run_dbt_and_capture(["snapshot"], expect_pass=False) + assert len(results) == 1 + assert "Compilation Error in snapshot snapshot_actual" in log_output + assert "Snapshot target is missing configured columns" in log_output + + +class TestSnapshotInvalidColumnNames(BaseSnapshotInvalidColumnNames): + pass + + +class BaseSnapshotDbtValidToCurrent: + @pytest.fixture(scope="class") + def snapshots(self): + return {"snapshot.sql": snapshot_actual_sql} + + @pytest.fixture(scope="class") + def models(self): + return { + "snapshots.yml": snapshots_valid_to_current_yml, + "ref_snapshot.sql": ref_snapshot_sql, + } + + def test_valid_to_current(self, project): + project.run_sql(create_seed_sql) + project.run_sql(create_snapshot_expected_sql) + project.run_sql(seed_insert_sql) + project.run_sql(populate_snapshot_expected_valid_to_current_sql) + + results = run_dbt(["snapshot"]) + assert len(results) == 1 + + original_snapshot = run_sql_with_adapter( + project.adapter, + "select id, test_scd_id, test_valid_to from {schema}.snapshot_actual", + "all", + ) + assert original_snapshot[0][2] == datetime.datetime(2099, 12, 31, 0, 0) + assert original_snapshot[9][2] == datetime.datetime(2099, 12, 31, 0, 0) + + project.run_sql(invalidate_sql) + project.run_sql(update_with_current_sql) + + results = run_dbt(["snapshot"]) + assert len(results) == 1 + + updated_snapshot = run_sql_with_adapter( + project.adapter, + "select id, test_scd_id, test_valid_to from {schema}.snapshot_actual", + "all", + ) + assert updated_snapshot[0][2] == datetime.datetime(2099, 12, 31, 0, 0) + # Original row that was updated now has a non-current (2099/12/31) date + assert updated_snapshot[9][2] == datetime.datetime(2016, 8, 20, 16, 44, 49) + # Updated row has a current date + assert updated_snapshot[20][2] == datetime.datetime(2099, 12, 31, 0, 0) + + check_relations_equal(project.adapter, ["snapshot_actual", "snapshot_expected"]) + + +class TestSnapshotDbtValidToCurrent(BaseSnapshotDbtValidToCurrent): + pass + + +# This uses snapshot_meta_column_names, yaml-only snapshot def, +# and multiple keys +class BaseSnapshotMultiUniqueKey: + @pytest.fixture(scope="class") + def models(self): + return { + "seed.sql": model_seed_sql, + "snapshots.yml": snapshots_multi_key_yml, + "ref_snapshot.sql": ref_snapshot_sql, + } + + def test_multi_column_unique_key(self, project): + project.run_sql(create_multi_key_seed_sql) + project.run_sql(create_multi_key_snapshot_expected_sql) + project.run_sql(seed_multi_key_insert_sql) + project.run_sql(populate_multi_key_snapshot_expected_sql) + + results = run_dbt(["snapshot"]) + assert len(results) == 1 + + project.run_sql(invalidate_multi_key_sql) + project.run_sql(update_multi_key_sql) + + results = run_dbt(["snapshot"]) + assert len(results) == 1 + + # run_dbt(["test"]) + check_relations_equal(project.adapter, ["snapshot_actual", "snapshot_expected"]) + + +class TestSnapshotMultiUniqueKey(BaseSnapshotMultiUniqueKey): + pass diff --git a/tests/functional/artifacts/data/results/v6/run_results.json b/tests/functional/artifacts/data/results/v6/run_results.json index f78176c930c..60ffaef6d9b 100644 --- a/tests/functional/artifacts/data/results/v6/run_results.json +++ b/tests/functional/artifacts/data/results/v6/run_results.json @@ -1 +1 @@ -{"metadata": {"dbt_schema_version": "https://schemas.getdbt.com/dbt/run-results/v6.json", "dbt_version": "1.8.0a1", "generated_at": "2023-12-06T18:53:19.641690Z", "invocation_id": "ad4ef714-e6c6-425e-b7c8-c1c4369df4ea", "env": {}}, "results": [{"status": "success", "timing": [{"name": "compile", "started_at": "2023-12-06T18:53:19.554953Z", "completed_at": "2023-12-06T18:53:19.559711Z"}, {"name": "execute", "started_at": "2023-12-06T18:53:19.564874Z", "completed_at": "2023-12-06T18:53:19.620151Z"}], "thread_id": "Thread-8", "execution_time": 0.06995701789855957, "adapter_response": {"_message": "CREATE VIEW", "code": "CREATE VIEW", "rows_affected": -1}, "message": "CREATE VIEW", "failures": null, "unique_id": "model.test.metricflow_time_spine", "compiled": true, "compiled_code": "SELECT to_date('02/20/2023', 'mm/dd/yyyy') as date_day", "relation_name": "\"dbt\".\"test17018887966812726006_test_previous_version_state\".\"metricflow_time_spine\""}, {"status": "success", "timing": [{"name": "compile", "started_at": "2023-12-06T18:53:19.557019Z", "completed_at": "2023-12-06T18:53:19.559247Z"}, {"name": "execute", "started_at": "2023-12-06T18:53:19.561000Z", "completed_at": "2023-12-06T18:53:19.622080Z"}], "thread_id": "Thread-9", "execution_time": 0.07100677490234375, "adapter_response": {"_message": "CREATE VIEW", "code": "CREATE VIEW", "rows_affected": -1}, "message": "CREATE VIEW", "failures": null, "unique_id": "model.test.my_model", "compiled": true, "compiled_code": "select 1 as id", "relation_name": "\"dbt\".\"test17018887966812726006_test_previous_version_state\".\"my_model\""}], "elapsed_time": 0.13903093338012695, "args": {"print": true, "log_level_file": "debug", "quiet": false, "warn_error_options": {"include": [], "exclude": []}, "write_json": true, "invocation_command": "dbt --cov=core --cov-append --cov-report=xml tests/functional/artifacts/test_previous_version_state.py", "log_level": "info", "select": [], "project_dir": "/private/var/folders/67/r0f0jlj54h95zl3fhmb217jh0000gp/T/pytest-of-william/pytest-68/project0", "static_parser": true, "log_file_max_bytes": 10485760, "empty": false, "introspect": true, "log_format_file": "debug", "vars": {}, "strict_mode": false, "indirect_selection": "eager", "show_resource_report": false, "favor_state": false, "version_check": true, "cache_selected_only": false, "enable_legacy_logger": false, "partial_parse": true, "profiles_dir": "/private/var/folders/67/r0f0jlj54h95zl3fhmb217jh0000gp/T/pytest-of-william/pytest-68/profile0", "defer": false, "printer_width": 80, "send_anonymous_usage_stats": false, "use_colors": true, "log_path": "/Users/william/git/dbt-core/logs/test17018887966812726006", "partial_parse_file_diff": true, "populate_cache": true, "macro_debugging": false, "use_colors_file": true, "log_format": "default", "which": "run", "exclude": []}} +{"metadata": {"dbt_schema_version": "https://schemas.getdbt.com/dbt/run-results/v6.json", "dbt_version": "1.9.0a1", "generated_at": "2024-09-24T15:24:36.246406Z", "invocation_id": "ca07d2ca-054f-4ea1-be63-4acccfe4a6a7", "env": {}}, "results": [{"status": "success", "timing": [{"name": "compile", "started_at": "2024-09-24T15:24:36.181448Z", "completed_at": "2024-09-24T15:24:36.183192Z"}, {"name": "execute", "started_at": "2024-09-24T15:24:36.185536Z", "completed_at": "2024-09-24T15:24:36.231442Z"}], "thread_id": "Thread-9 (worker)", "execution_time": 0.0535128116607666, "adapter_response": {"_message": "CREATE VIEW", "code": "CREATE VIEW", "rows_affected": -1}, "message": "CREATE VIEW", "failures": null, "unique_id": "model.test.my_model", "compiled": true, "compiled_code": "select 1 as id", "relation_name": "\"dbt\".\"test17271914717676665882_test_previous_version_state\".\"my_model\"", "batch_results": null}, {"status": "success", "timing": [{"name": "compile", "started_at": "2024-09-24T15:24:36.179261Z", "completed_at": "2024-09-24T15:24:36.182955Z"}, {"name": "execute", "started_at": "2024-09-24T15:24:36.183455Z", "completed_at": "2024-09-24T15:24:36.232800Z"}], "thread_id": "Thread-8 (worker)", "execution_time": 0.055058956146240234, "adapter_response": {"_message": "CREATE VIEW", "code": "CREATE VIEW", "rows_affected": -1}, "message": "CREATE VIEW", "failures": null, "unique_id": "model.test.metricflow_time_spine", "compiled": true, "compiled_code": "SELECT to_date('02/20/2023', 'mm/dd/yyyy') as date_day", "relation_name": "\"dbt\".\"test17271914717676665882_test_previous_version_state\".\"metricflow_time_spine\"", "batch_results": null}], "elapsed_time": 0.6437027454376221, "args": {"profiles_dir": "/private/var/folders/79/5290gpvn3lx5jdryk4844rm80000gn/T/pytest-of-quigleymalcolm/pytest-139/profile0", "invocation_command": "dbt tests/functional/artifacts/test_previous_version_state.py", "strict_mode": false, "partial_parse_file_diff": true, "favor_state": false, "select": [], "log_level_file": "debug", "log_format_file": "debug", "which": "run", "introspect": true, "cache_selected_only": false, "log_level": "info", "defer": false, "static_parser": true, "macro_debugging": false, "write_json": true, "partial_parse": true, "version_check": true, "exclude": [], "use_colors_file": true, "indirect_selection": "eager", "project_dir": "/private/var/folders/79/5290gpvn3lx5jdryk4844rm80000gn/T/pytest-of-quigleymalcolm/pytest-139/project0", "require_resource_names_without_spaces": false, "warn_error_options": {"include": [], "exclude": []}, "log_path": "/Users/quigleymalcolm/Developer/dbt-labs/dbt-core/logs/test17271914717676665882", "printer_width": 80, "use_colors": true, "require_explicit_package_overrides_for_builtin_materializations": true, "show_resource_report": false, "quiet": false, "log_format": "default", "populate_cache": true, "send_anonymous_usage_stats": false, "source_freshness_run_project_hooks": false, "log_file_max_bytes": 10485760, "print": true, "vars": {}, "empty": false}} diff --git a/tests/functional/artifacts/data/state/v12/manifest.json b/tests/functional/artifacts/data/state/v12/manifest.json index 25d83c3e796..0e5a34a06d2 100644 --- a/tests/functional/artifacts/data/state/v12/manifest.json +++ b/tests/functional/artifacts/data/state/v12/manifest.json @@ -1 +1 @@ -{"metadata": {"dbt_schema_version": "https://schemas.getdbt.com/dbt/manifest/v12.json", "dbt_version": "1.8.0b3", "generated_at": "2024-05-02T11:13:36.981553Z", "invocation_id": "05015bbc-b4d2-47f4-996f-acac2c7e1a85", "env": {}, "project_name": "test", "project_id": "098f6bcd4621d373cade4e832627b4f6", "user_id": null, "send_anonymous_usage_stats": false, "adapter_type": "postgres"}, "nodes": {"model.test.my_model": {"database": "dbt", "schema": "test17146484148326086409_test_previous_version_state", "name": "my_model", "resource_type": "model", "package_name": "test", "path": "my_model.sql", "original_file_path": "models/my_model.sql", "unique_id": "model.test.my_model", "fqn": ["test", "my_model"], "alias": "my_model", "checksum": {"name": "sha256", "checksum": "3ea0f972fa1b56aa2dc2f56ee784b6a5796312f9a813d59ae70fd8855f10d16d"}, "config": {"enabled": true, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "view", "incremental_strategy": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": null, "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}, "access": "protected"}, "tags": [], "description": "Example model", "columns": {"id": {"name": "id", "description": "", "meta": {}, "data_type": null, "constraints": [], "quote": null, "tags": []}}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": "test://models/schema.yml", "build_path": null, "unrendered_config": {}, "created_at": 1714648415.895201, "relation_name": "\"dbt\".\"test17146484148326086409_test_previous_version_state\".\"my_model\"", "raw_code": "select 1 as id", "language": "sql", "refs": [], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": []}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}, "access": "protected", "constraints": [], "version": null, "latest_version": null, "deprecation_date": null}, "model.test.metricflow_time_spine": {"database": "dbt", "schema": "test17146484148326086409_test_previous_version_state", "name": "metricflow_time_spine", "resource_type": "model", "package_name": "test", "path": "metricflow_time_spine.sql", "original_file_path": "models/metricflow_time_spine.sql", "unique_id": "model.test.metricflow_time_spine", "fqn": ["test", "metricflow_time_spine"], "alias": "metricflow_time_spine", "checksum": {"name": "sha256", "checksum": "954d9b349821edb5558a373119a7d91eeac9e620aaa96cd112c0d14bab729fdb"}, "config": {"enabled": true, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "view", "incremental_strategy": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": null, "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}, "access": "protected"}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {}, "created_at": 1714648415.362005, "relation_name": "\"dbt\".\"test17146484148326086409_test_previous_version_state\".\"metricflow_time_spine\"", "raw_code": "SELECT to_date('02/20/2023', 'mm/dd/yyyy') as date_day", "language": "sql", "refs": [], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": []}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}, "access": "protected", "constraints": [], "version": null, "latest_version": null, "deprecation_date": null}, "snapshot.test.snapshot_seed": {"database": "dbt", "schema": "test17146484148326086409_test_previous_version_state", "name": "snapshot_seed", "resource_type": "snapshot", "package_name": "test", "path": "snapshot_seed.sql", "original_file_path": "snapshots/snapshot_seed.sql", "unique_id": "snapshot.test.snapshot_seed", "fqn": ["test", "snapshot_seed", "snapshot_seed"], "alias": "snapshot_seed", "checksum": {"name": "sha256", "checksum": "5fc998f39655f8fe52443a919e749b6e23883ef90202b040412baac13c6bfe18"}, "config": {"enabled": true, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "snapshot", "incremental_strategy": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": "id", "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}, "strategy": "check", "target_schema": "test17146484148326086409_test_previous_version_state", "target_database": null, "updated_at": null, "check_cols": "all"}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {"unique_key": "id", "strategy": "check", "check_cols": "all", "target_schema": "test17146484148326086409_test_previous_version_state"}, "created_at": 1714648415.444036, "relation_name": "\"dbt\".\"test17146484148326086409_test_previous_version_state\".\"snapshot_seed\"", "raw_code": "\n{{\n config(\n unique_key='id',\n strategy='check',\n check_cols='all',\n target_schema=schema,\n )\n}}\nselect * from {{ ref('my_seed') }}\n", "language": "sql", "refs": [{"name": "my_seed", "package": null, "version": null}], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": ["seed.test.my_seed"]}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}}, "analysis.test.a": {"database": "dbt", "schema": "test17146484148326086409_test_previous_version_state", "name": "a", "resource_type": "analysis", "package_name": "test", "path": "analysis/a.sql", "original_file_path": "analyses/a.sql", "unique_id": "analysis.test.a", "fqn": ["test", "analysis", "a"], "alias": "a", "checksum": {"name": "sha256", "checksum": "a389c282f569f0bbdc2a8a4f174dea746c28582fdaf2048d31d9226af9feab23"}, "config": {"enabled": true, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "view", "incremental_strategy": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": null, "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {}, "created_at": 1714648415.595048, "relation_name": null, "raw_code": "select 4 as id", "language": "sql", "refs": [], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": []}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}}, "test.test.just_my": {"database": "dbt", "schema": "test17146484148326086409_test_previous_version_state_dbt_test__audit", "name": "just_my", "resource_type": "test", "package_name": "test", "path": "just_my.sql", "original_file_path": "tests/just_my.sql", "unique_id": "test.test.just_my", "fqn": ["test", "just_my"], "alias": "just_my", "checksum": {"name": "sha256", "checksum": "744889a2e2d9ce380619265e1217d7ccf6e6ca896c048d42ebe0f9cfb74d7156"}, "config": {"enabled": true, "alias": null, "schema": "dbt_test__audit", "database": null, "tags": ["data_test_tag"], "meta": {}, "group": null, "materialized": "test", "severity": "ERROR", "store_failures": null, "store_failures_as": null, "where": null, "limit": null, "fail_calc": "count(*)", "warn_if": "!= 0", "error_if": "!= 0"}, "tags": ["data_test_tag"], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {"tags": ["data_test_tag"]}, "created_at": 1714648415.67546, "relation_name": null, "raw_code": "{{ config(tags = ['data_test_tag']) }}\n\nselect * from {{ ref('my_model') }}\nwhere false", "language": "sql", "refs": [{"name": "my_model", "package": null, "version": null}], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": ["model.test.my_model"]}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}}, "seed.test.my_seed": {"database": "dbt", "schema": "test17146484148326086409_test_previous_version_state", "name": "my_seed", "resource_type": "seed", "package_name": "test", "path": "my_seed.csv", "original_file_path": "seeds/my_seed.csv", "unique_id": "seed.test.my_seed", "fqn": ["test", "my_seed"], "alias": "my_seed", "checksum": {"name": "sha256", "checksum": "f7ede83f36165ac6b7a047aa2c3f212dff385bfa9f35f395108cd06fc8e96943"}, "config": {"enabled": true, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "seed", "incremental_strategy": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": null, "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}, "delimiter": ",", "quote_columns": null}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {}, "created_at": 1714648415.774666, "relation_name": "\"dbt\".\"test17146484148326086409_test_previous_version_state\".\"my_seed\"", "raw_code": "", "root_path": "/private/var/folders/7h/hj5_fw9j291c58hwfdvy5xbm0000gp/T/pytest-of-jerco/pytest-50/project0", "depends_on": {"macros": []}}, "test.test.not_null_my_model_id.43e0e9183a": {"database": "dbt", "schema": "test17146484148326086409_test_previous_version_state_dbt_test__audit", "name": "not_null_my_model_id", "resource_type": "test", "package_name": "test", "path": "not_null_my_model_id.sql", "original_file_path": "models/schema.yml", "unique_id": "test.test.not_null_my_model_id.43e0e9183a", "fqn": ["test", "not_null_my_model_id"], "alias": "not_null_my_model_id", "checksum": {"name": "none", "checksum": ""}, "config": {"enabled": true, "alias": null, "schema": "dbt_test__audit", "database": null, "tags": [], "meta": {}, "group": null, "materialized": "test", "severity": "ERROR", "store_failures": null, "store_failures_as": null, "where": null, "limit": null, "fail_calc": "count(*)", "warn_if": "!= 0", "error_if": "!= 0"}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {}, "created_at": 1714648415.947873, "relation_name": null, "raw_code": "{{ test_not_null(**_dbt_generic_test_kwargs) }}", "language": "sql", "refs": [{"name": "my_model", "package": null, "version": null}], "sources": [], "metrics": [], "depends_on": {"macros": ["macro.dbt.test_not_null"], "nodes": ["model.test.my_model"]}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}, "column_name": "id", "file_key_name": "models.my_model", "attached_node": "model.test.my_model", "test_metadata": {"name": "not_null", "kwargs": {"column_name": "id", "model": "{{ get_where_subquery(ref('my_model')) }}"}, "namespace": null}}, "test.test.check_nothing_my_model_.d5a5e66110": {"database": "dbt", "schema": "test17146484148326086409_test_previous_version_state_dbt_test__audit", "name": "check_nothing_my_model_", "resource_type": "test", "package_name": "test", "path": "check_nothing_my_model_.sql", "original_file_path": "models/schema.yml", "unique_id": "test.test.check_nothing_my_model_.d5a5e66110", "fqn": ["test", "check_nothing_my_model_"], "alias": "check_nothing_my_model_", "checksum": {"name": "none", "checksum": ""}, "config": {"enabled": true, "alias": null, "schema": "dbt_test__audit", "database": null, "tags": [], "meta": {}, "group": null, "materialized": "test", "severity": "ERROR", "store_failures": null, "store_failures_as": null, "where": null, "limit": null, "fail_calc": "count(*)", "warn_if": "!= 0", "error_if": "!= 0"}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {}, "created_at": 1714648415.949169, "relation_name": null, "raw_code": "{{ test_check_nothing(**_dbt_generic_test_kwargs) }}", "language": "sql", "refs": [{"name": "my_model", "package": null, "version": null}], "sources": [], "metrics": [], "depends_on": {"macros": ["macro.test.test_check_nothing", "macro.dbt.get_where_subquery"], "nodes": ["model.test.my_model"]}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}, "column_name": null, "file_key_name": "models.my_model", "attached_node": "model.test.my_model", "test_metadata": {"name": "check_nothing", "kwargs": {"model": "{{ get_where_subquery(ref('my_model')) }}"}, "namespace": null}}}, "sources": {"source.test.my_source.my_table": {"database": "dbt", "schema": "my_source", "name": "my_table", "resource_type": "source", "package_name": "test", "path": "models/schema.yml", "original_file_path": "models/schema.yml", "unique_id": "source.test.my_source.my_table", "fqn": ["test", "my_source", "my_table"], "source_name": "my_source", "source_description": "My source", "loader": "a_loader", "identifier": "my_seed", "quoting": {"database": null, "schema": null, "identifier": null, "column": null}, "loaded_at_field": null, "freshness": {"warn_after": {"count": null, "period": null}, "error_after": {"count": null, "period": null}, "filter": null}, "external": null, "description": "My table", "columns": {}, "meta": {}, "source_meta": {}, "tags": [], "config": {"enabled": true}, "patch_path": null, "unrendered_config": {}, "relation_name": "\"dbt\".\"my_source\".\"my_seed\"", "created_at": 1714648416.1599889}}, "macros": {"macro.test.test_check_nothing": {"name": "test_check_nothing", "resource_type": "macro", "package_name": "test", "path": "macros/dummy_test.sql", "original_file_path": "macros/dummy_test.sql", "unique_id": "macro.test.test_check_nothing", "macro_sql": "{% test check_nothing(model) %}\n-- a silly test to make sure that table-level tests show up in the manifest\n-- without a column_name field\n\nselect 0\n\n{% endtest %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9534872, "supported_languages": null}, "macro.test.test_disabled_check_nothing": {"name": "test_disabled_check_nothing", "resource_type": "macro", "package_name": "test", "path": "macros/disabled_dummy_test.sql", "original_file_path": "macros/disabled_dummy_test.sql", "unique_id": "macro.test.test_disabled_check_nothing", "macro_sql": "{% test disabled_check_nothing(model) %}\n-- a silly test to make sure that table-level tests show up in the manifest\n-- without a column_name field\n\n{{ config(enabled=False) }}\nselect 0\n\n{% endtest %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.953828, "supported_languages": null}, "macro.test.do_nothing": {"name": "do_nothing", "resource_type": "macro", "package_name": "test", "path": "macros/do_nothing.sql", "original_file_path": "macros/do_nothing.sql", "unique_id": "macro.test.do_nothing", "macro_sql": "{% macro do_nothing(foo2, bar2) %}\n select\n '{{ foo2 }}' as foo2,\n '{{ bar2 }}' as bar2\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954112, "supported_languages": null}, "macro.dbt_postgres.postgres__current_timestamp": {"name": "postgres__current_timestamp", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/timestamps.sql", "original_file_path": "macros/timestamps.sql", "unique_id": "macro.dbt_postgres.postgres__current_timestamp", "macro_sql": "{% macro postgres__current_timestamp() -%}\n now()\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954139, "supported_languages": null}, "macro.dbt_postgres.postgres__snapshot_string_as_time": {"name": "postgres__snapshot_string_as_time", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/timestamps.sql", "original_file_path": "macros/timestamps.sql", "unique_id": "macro.dbt_postgres.postgres__snapshot_string_as_time", "macro_sql": "{% macro postgres__snapshot_string_as_time(timestamp) -%}\n {%- set result = \"'\" ~ timestamp ~ \"'::timestamp without time zone\" -%}\n {{ return(result) }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954147, "supported_languages": null}, "macro.dbt_postgres.postgres__snapshot_get_time": {"name": "postgres__snapshot_get_time", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/timestamps.sql", "original_file_path": "macros/timestamps.sql", "unique_id": "macro.dbt_postgres.postgres__snapshot_get_time", "macro_sql": "{% macro postgres__snapshot_get_time() -%}\n {{ current_timestamp() }}::timestamp without time zone\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.current_timestamp"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954156, "supported_languages": null}, "macro.dbt_postgres.postgres__current_timestamp_backcompat": {"name": "postgres__current_timestamp_backcompat", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/timestamps.sql", "original_file_path": "macros/timestamps.sql", "unique_id": "macro.dbt_postgres.postgres__current_timestamp_backcompat", "macro_sql": "{% macro postgres__current_timestamp_backcompat() %}\n current_timestamp::{{ type_timestamp() }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.type_timestamp"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954165, "supported_languages": null}, "macro.dbt_postgres.postgres__current_timestamp_in_utc_backcompat": {"name": "postgres__current_timestamp_in_utc_backcompat", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/timestamps.sql", "original_file_path": "macros/timestamps.sql", "unique_id": "macro.dbt_postgres.postgres__current_timestamp_in_utc_backcompat", "macro_sql": "{% macro postgres__current_timestamp_in_utc_backcompat() %}\n (current_timestamp at time zone 'utc')::{{ type_timestamp() }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.type_timestamp"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954171, "supported_languages": null}, "macro.dbt_postgres.postgres__get_catalog_relations": {"name": "postgres__get_catalog_relations", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/catalog.sql", "original_file_path": "macros/catalog.sql", "unique_id": "macro.dbt_postgres.postgres__get_catalog_relations", "macro_sql": "{% macro postgres__get_catalog_relations(information_schema, relations) -%}\n {%- call statement('catalog', fetch_result=True) -%}\n\n {#\n If the user has multiple databases set and the first one is wrong, this will fail.\n But we won't fail in the case where there are multiple quoting-difference-only dbs, which is better.\n #}\n {% set database = information_schema.database %}\n {{ adapter.verify_database(database) }}\n\n select\n '{{ database }}' as table_database,\n sch.nspname as table_schema,\n tbl.relname as table_name,\n case tbl.relkind\n when 'v' then 'VIEW'\n when 'm' then 'MATERIALIZED VIEW'\n else 'BASE TABLE'\n end as table_type,\n tbl_desc.description as table_comment,\n col.attname as column_name,\n col.attnum as column_index,\n pg_catalog.format_type(col.atttypid, col.atttypmod) as column_type,\n col_desc.description as column_comment,\n pg_get_userbyid(tbl.relowner) as table_owner\n\n from pg_catalog.pg_namespace sch\n join pg_catalog.pg_class tbl on tbl.relnamespace = sch.oid\n join pg_catalog.pg_attribute col on col.attrelid = tbl.oid\n left outer join pg_catalog.pg_description tbl_desc on (tbl_desc.objoid = tbl.oid and tbl_desc.objsubid = 0)\n left outer join pg_catalog.pg_description col_desc on (col_desc.objoid = tbl.oid and col_desc.objsubid = col.attnum)\n where (\n {%- for relation in relations -%}\n {%- if relation.identifier -%}\n (upper(sch.nspname) = upper('{{ relation.schema }}') and\n upper(tbl.relname) = upper('{{ relation.identifier }}'))\n {%- else-%}\n upper(sch.nspname) = upper('{{ relation.schema }}')\n {%- endif -%}\n {%- if not loop.last %} or {% endif -%}\n {%- endfor -%}\n )\n and not pg_is_other_temp_schema(sch.oid) -- not a temporary schema belonging to another session\n and tbl.relpersistence in ('p', 'u') -- [p]ermanent table or [u]nlogged table. Exclude [t]emporary tables\n and tbl.relkind in ('r', 'v', 'f', 'p', 'm') -- o[r]dinary table, [v]iew, [f]oreign table, [p]artitioned table, [m]aterialized view. Other values are [i]ndex, [S]equence, [c]omposite type, [t]OAST table\n and col.attnum > 0 -- negative numbers are used for system columns such as oid\n and not col.attisdropped -- column as not been dropped\n\n order by\n sch.nspname,\n tbl.relname,\n col.attnum\n\n {%- endcall -%}\n\n {{ return(load_result('catalog').table) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954188, "supported_languages": null}, "macro.dbt_postgres.postgres__get_catalog": {"name": "postgres__get_catalog", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/catalog.sql", "original_file_path": "macros/catalog.sql", "unique_id": "macro.dbt_postgres.postgres__get_catalog", "macro_sql": "{% macro postgres__get_catalog(information_schema, schemas) -%}\n {%- set relations = [] -%}\n {%- for schema in schemas -%}\n {%- set dummy = relations.append({'schema': schema}) -%}\n {%- endfor -%}\n {{ return(postgres__get_catalog_relations(information_schema, relations)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_catalog_relations"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954249, "supported_languages": null}, "macro.dbt_postgres.postgres__get_relations": {"name": "postgres__get_relations", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations.sql", "original_file_path": "macros/relations.sql", "unique_id": "macro.dbt_postgres.postgres__get_relations", "macro_sql": "{% macro postgres__get_relations() -%}\n\n {#\n -- in pg_depend, objid is the dependent, refobjid is the referenced object\n -- > a pg_depend entry indicates that the referenced object cannot be\n -- > dropped without also dropping the dependent object.\n #}\n\n {%- call statement('relations', fetch_result=True) -%}\n with relation as (\n select\n pg_rewrite.ev_class as class,\n pg_rewrite.oid as id\n from pg_rewrite\n ),\n class as (\n select\n oid as id,\n relname as name,\n relnamespace as schema,\n relkind as kind\n from pg_class\n ),\n dependency as (\n select distinct\n pg_depend.objid as id,\n pg_depend.refobjid as ref\n from pg_depend\n ),\n schema as (\n select\n pg_namespace.oid as id,\n pg_namespace.nspname as name\n from pg_namespace\n where nspname != 'information_schema' and nspname not like 'pg\\_%'\n ),\n referenced as (\n select\n relation.id AS id,\n referenced_class.name ,\n referenced_class.schema ,\n referenced_class.kind\n from relation\n join class as referenced_class on relation.class=referenced_class.id\n where referenced_class.kind in ('r', 'v', 'm')\n ),\n relationships as (\n select\n referenced.name as referenced_name,\n referenced.schema as referenced_schema_id,\n dependent_class.name as dependent_name,\n dependent_class.schema as dependent_schema_id,\n referenced.kind as kind\n from referenced\n join dependency on referenced.id=dependency.id\n join class as dependent_class on dependency.ref=dependent_class.id\n where\n (referenced.name != dependent_class.name or\n referenced.schema != dependent_class.schema)\n )\n\n select\n referenced_schema.name as referenced_schema,\n relationships.referenced_name as referenced_name,\n dependent_schema.name as dependent_schema,\n relationships.dependent_name as dependent_name\n from relationships\n join schema as dependent_schema on relationships.dependent_schema_id=dependent_schema.id\n join schema as referenced_schema on relationships.referenced_schema_id=referenced_schema.id\n group by referenced_schema, referenced_name, dependent_schema, dependent_name\n order by referenced_schema, referenced_name, dependent_schema, dependent_name;\n\n {%- endcall -%}\n\n {{ return(load_result('relations').table) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954271, "supported_languages": null}, "macro.dbt_postgres.postgres_get_relations": {"name": "postgres_get_relations", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations.sql", "original_file_path": "macros/relations.sql", "unique_id": "macro.dbt_postgres.postgres_get_relations", "macro_sql": "{% macro postgres_get_relations() %}\n {{ return(postgres__get_relations()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_relations"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9542809, "supported_languages": null}, "macro.dbt_postgres.postgres__create_table_as": {"name": "postgres__create_table_as", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__create_table_as", "macro_sql": "{% macro postgres__create_table_as(temporary, relation, sql) -%}\n {%- set unlogged = config.get('unlogged', default=false) -%}\n {%- set sql_header = config.get('sql_header', none) -%}\n\n {{ sql_header if sql_header is not none }}\n\n create {% if temporary -%}\n temporary\n {%- elif unlogged -%}\n unlogged\n {%- endif %} table {{ relation }}\n {% set contract_config = config.get('contract') %}\n {% if contract_config.enforced %}\n {{ get_assert_columns_equivalent(sql) }}\n {% endif -%}\n {% if contract_config.enforced and (not temporary) -%}\n {{ get_table_columns_and_constraints() }} ;\n insert into {{ relation }} (\n {{ adapter.dispatch('get_column_names', 'dbt')() }}\n )\n {%- set sql = get_select_subquery(sql) %}\n {% else %}\n as\n {% endif %}\n (\n {{ sql }}\n );\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.get_assert_columns_equivalent", "macro.dbt.get_table_columns_and_constraints", "macro.dbt.default__get_column_names", "macro.dbt.get_select_subquery"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9543018, "supported_languages": null}, "macro.dbt_postgres.postgres__get_create_index_sql": {"name": "postgres__get_create_index_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__get_create_index_sql", "macro_sql": "{% macro postgres__get_create_index_sql(relation, index_dict) -%}\n {%- set index_config = adapter.parse_index(index_dict) -%}\n {%- set comma_separated_columns = \", \".join(index_config.columns) -%}\n {%- set index_name = index_config.render(relation) -%}\n\n create {% if index_config.unique -%}\n unique\n {%- endif %} index if not exists\n \"{{ index_name }}\"\n on {{ relation }} {% if index_config.type -%}\n using {{ index_config.type }}\n {%- endif %}\n ({{ comma_separated_columns }})\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9543092, "supported_languages": null}, "macro.dbt_postgres.postgres__create_schema": {"name": "postgres__create_schema", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__create_schema", "macro_sql": "{% macro postgres__create_schema(relation) -%}\n {% if relation.database -%}\n {{ adapter.verify_database(relation.database) }}\n {%- endif -%}\n {%- call statement('create_schema') -%}\n create schema if not exists {{ relation.without_identifier().include(database=False) }}\n {%- endcall -%}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954317, "supported_languages": null}, "macro.dbt_postgres.postgres__drop_schema": {"name": "postgres__drop_schema", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__drop_schema", "macro_sql": "{% macro postgres__drop_schema(relation) -%}\n {% if relation.database -%}\n {{ adapter.verify_database(relation.database) }}\n {%- endif -%}\n {%- call statement('drop_schema') -%}\n drop schema if exists {{ relation.without_identifier().include(database=False) }} cascade\n {%- endcall -%}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9543219, "supported_languages": null}, "macro.dbt_postgres.postgres__get_columns_in_relation": {"name": "postgres__get_columns_in_relation", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__get_columns_in_relation", "macro_sql": "{% macro postgres__get_columns_in_relation(relation) -%}\n {% call statement('get_columns_in_relation', fetch_result=True) %}\n select\n column_name,\n data_type,\n character_maximum_length,\n numeric_precision,\n numeric_scale\n\n from {{ relation.information_schema('columns') }}\n where table_name = '{{ relation.identifier }}'\n {% if relation.schema %}\n and table_schema = '{{ relation.schema }}'\n {% endif %}\n order by ordinal_position\n\n {% endcall %}\n {% set table = load_result('get_columns_in_relation').table %}\n {{ return(sql_convert_columns_in_relation(table)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement", "macro.dbt.sql_convert_columns_in_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9543302, "supported_languages": null}, "macro.dbt_postgres.postgres__list_relations_without_caching": {"name": "postgres__list_relations_without_caching", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__list_relations_without_caching", "macro_sql": "{% macro postgres__list_relations_without_caching(schema_relation) %}\n {% call statement('list_relations_without_caching', fetch_result=True) -%}\n select\n '{{ schema_relation.database }}' as database,\n tablename as name,\n schemaname as schema,\n 'table' as type\n from pg_tables\n where schemaname ilike '{{ schema_relation.schema }}'\n union all\n select\n '{{ schema_relation.database }}' as database,\n viewname as name,\n schemaname as schema,\n 'view' as type\n from pg_views\n where schemaname ilike '{{ schema_relation.schema }}'\n union all\n select\n '{{ schema_relation.database }}' as database,\n matviewname as name,\n schemaname as schema,\n 'materialized_view' as type\n from pg_matviews\n where schemaname ilike '{{ schema_relation.schema }}'\n {% endcall %}\n {{ return(load_result('list_relations_without_caching').table) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954336, "supported_languages": null}, "macro.dbt_postgres.postgres__information_schema_name": {"name": "postgres__information_schema_name", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__information_schema_name", "macro_sql": "{% macro postgres__information_schema_name(database) -%}\n {% if database_name -%}\n {{ adapter.verify_database(database_name) }}\n {%- endif -%}\n information_schema\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954345, "supported_languages": null}, "macro.dbt_postgres.postgres__list_schemas": {"name": "postgres__list_schemas", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__list_schemas", "macro_sql": "{% macro postgres__list_schemas(database) %}\n {% if database -%}\n {{ adapter.verify_database(database) }}\n {%- endif -%}\n {% call statement('list_schemas', fetch_result=True, auto_begin=False) %}\n select distinct nspname from pg_namespace\n {% endcall %}\n {{ return(load_result('list_schemas').table) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9543512, "supported_languages": null}, "macro.dbt_postgres.postgres__check_schema_exists": {"name": "postgres__check_schema_exists", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__check_schema_exists", "macro_sql": "{% macro postgres__check_schema_exists(information_schema, schema) -%}\n {% if information_schema.database -%}\n {{ adapter.verify_database(information_schema.database) }}\n {%- endif -%}\n {% call statement('check_schema_exists', fetch_result=True, auto_begin=False) %}\n select count(*) from pg_namespace where nspname = '{{ schema }}'\n {% endcall %}\n {{ return(load_result('check_schema_exists').table) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954356, "supported_languages": null}, "macro.dbt_postgres.postgres__make_relation_with_suffix": {"name": "postgres__make_relation_with_suffix", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__make_relation_with_suffix", "macro_sql": "{% macro postgres__make_relation_with_suffix(base_relation, suffix, dstring) %}\n {% if dstring %}\n {% set dt = modules.datetime.datetime.now() %}\n {% set dtstring = dt.strftime(\"%H%M%S%f\") %}\n {% set suffix = suffix ~ dtstring %}\n {% endif %}\n {% set suffix_length = suffix|length %}\n {% set relation_max_name_length = base_relation.relation_max_name_length() %}\n {% if suffix_length > relation_max_name_length %}\n {% do exceptions.raise_compiler_error('Relation suffix is too long (' ~ suffix_length ~ ' characters). Maximum length is ' ~ relation_max_name_length ~ ' characters.') %}\n {% endif %}\n {% set identifier = base_relation.identifier[:relation_max_name_length - suffix_length] ~ suffix %}\n\n {{ return(base_relation.incorporate(path={\"identifier\": identifier })) }}\n\n {% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9543638, "supported_languages": null}, "macro.dbt_postgres.postgres__make_intermediate_relation": {"name": "postgres__make_intermediate_relation", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__make_intermediate_relation", "macro_sql": "{% macro postgres__make_intermediate_relation(base_relation, suffix) %}\n {{ return(postgres__make_relation_with_suffix(base_relation, suffix, dstring=False)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__make_relation_with_suffix"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9543731, "supported_languages": null}, "macro.dbt_postgres.postgres__make_temp_relation": {"name": "postgres__make_temp_relation", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__make_temp_relation", "macro_sql": "{% macro postgres__make_temp_relation(base_relation, suffix) %}\n {% set temp_relation = postgres__make_relation_with_suffix(base_relation, suffix, dstring=True) %}\n {{ return(temp_relation.incorporate(path={\"schema\": none,\n \"database\": none})) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__make_relation_with_suffix"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954381, "supported_languages": null}, "macro.dbt_postgres.postgres__make_backup_relation": {"name": "postgres__make_backup_relation", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__make_backup_relation", "macro_sql": "{% macro postgres__make_backup_relation(base_relation, backup_relation_type, suffix) %}\n {% set backup_relation = postgres__make_relation_with_suffix(base_relation, suffix, dstring=False) %}\n {{ return(backup_relation.incorporate(type=backup_relation_type)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__make_relation_with_suffix"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954387, "supported_languages": null}, "macro.dbt_postgres.postgres_escape_comment": {"name": "postgres_escape_comment", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres_escape_comment", "macro_sql": "{% macro postgres_escape_comment(comment) -%}\n {% if comment is not string %}\n {% do exceptions.raise_compiler_error('cannot escape a non-string: ' ~ comment) %}\n {% endif %}\n {%- set magic = '$dbt_comment_literal_block$' -%}\n {%- if magic in comment -%}\n {%- do exceptions.raise_compiler_error('The string ' ~ magic ~ ' is not allowed in comments.') -%}\n {%- endif -%}\n {{ magic }}{{ comment }}{{ magic }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9543931, "supported_languages": null}, "macro.dbt_postgres.postgres__alter_relation_comment": {"name": "postgres__alter_relation_comment", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__alter_relation_comment", "macro_sql": "{% macro postgres__alter_relation_comment(relation, comment) %}\n {% set escaped_comment = postgres_escape_comment(comment) %}\n comment on {{ relation.type }} {{ relation }} is {{ escaped_comment }};\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres_escape_comment"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954404, "supported_languages": null}, "macro.dbt_postgres.postgres__alter_column_comment": {"name": "postgres__alter_column_comment", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__alter_column_comment", "macro_sql": "{% macro postgres__alter_column_comment(relation, column_dict) %}\n {% set existing_columns = adapter.get_columns_in_relation(relation) | map(attribute=\"name\") | list %}\n {% for column_name in column_dict if (column_name in existing_columns) %}\n {% set comment = column_dict[column_name]['description'] %}\n {% set escaped_comment = postgres_escape_comment(comment) %}\n comment on column {{ relation }}.{{ adapter.quote(column_name) if column_dict[column_name]['quote'] else column_name }} is {{ escaped_comment }};\n {% endfor %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres_escape_comment"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95441, "supported_languages": null}, "macro.dbt_postgres.postgres__get_show_grant_sql": {"name": "postgres__get_show_grant_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__get_show_grant_sql", "macro_sql": "\n\n{%- macro postgres__get_show_grant_sql(relation) -%}\n select grantee, privilege_type\n from {{ relation.information_schema('role_table_grants') }}\n where grantor = current_role\n and grantee != current_role\n and table_schema = '{{ relation.schema }}'\n and table_name = '{{ relation.identifier }}'\n{%- endmacro -%}\n\n", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954416, "supported_languages": null}, "macro.dbt_postgres.postgres__copy_grants": {"name": "postgres__copy_grants", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__copy_grants", "macro_sql": "{% macro postgres__copy_grants() %}\n {{ return(False) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954422, "supported_languages": null}, "macro.dbt_postgres.postgres__get_show_indexes_sql": {"name": "postgres__get_show_indexes_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__get_show_indexes_sql", "macro_sql": "{% macro postgres__get_show_indexes_sql(relation) %}\n select\n i.relname as name,\n m.amname as method,\n ix.indisunique as \"unique\",\n array_to_string(array_agg(a.attname), ',') as column_names\n from pg_index ix\n join pg_class i\n on i.oid = ix.indexrelid\n join pg_am m\n on m.oid=i.relam\n join pg_class t\n on t.oid = ix.indrelid\n join pg_namespace n\n on n.oid = t.relnamespace\n join pg_attribute a\n on a.attrelid = t.oid\n and a.attnum = ANY(ix.indkey)\n where t.relname = '{{ relation.identifier }}'\n and n.nspname = '{{ relation.schema }}'\n and t.relkind in ('r', 'm')\n group by 1, 2, 3\n order by 1, 2, 3\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9544299, "supported_languages": null}, "macro.dbt_postgres.postgres__get_drop_index_sql": {"name": "postgres__get_drop_index_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__get_drop_index_sql", "macro_sql": "\n\n\n{%- macro postgres__get_drop_index_sql(relation, index_name) -%}\n drop index if exists \"{{ relation.schema }}\".\"{{ index_name }}\"\n{%- endmacro -%}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954438, "supported_languages": null}, "macro.dbt_postgres.postgres__get_incremental_default_sql": {"name": "postgres__get_incremental_default_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/materializations/incremental_strategies.sql", "original_file_path": "macros/materializations/incremental_strategies.sql", "unique_id": "macro.dbt_postgres.postgres__get_incremental_default_sql", "macro_sql": "{% macro postgres__get_incremental_default_sql(arg_dict) %}\n\n {% if arg_dict[\"unique_key\"] %}\n {% do return(get_incremental_delete_insert_sql(arg_dict)) %}\n {% else %}\n {% do return(get_incremental_append_sql(arg_dict)) %}\n {% endif %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_incremental_delete_insert_sql", "macro.dbt.get_incremental_append_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9544508, "supported_languages": null}, "macro.dbt_postgres.postgres__snapshot_merge_sql": {"name": "postgres__snapshot_merge_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/materializations/snapshot_merge.sql", "original_file_path": "macros/materializations/snapshot_merge.sql", "unique_id": "macro.dbt_postgres.postgres__snapshot_merge_sql", "macro_sql": "{% macro postgres__snapshot_merge_sql(target, source, insert_cols) -%}\n {%- set insert_cols_csv = insert_cols | join(', ') -%}\n\n update {{ target }}\n set dbt_valid_to = DBT_INTERNAL_SOURCE.dbt_valid_to\n from {{ source }} as DBT_INTERNAL_SOURCE\n where DBT_INTERNAL_SOURCE.dbt_scd_id::text = {{ target }}.dbt_scd_id::text\n and DBT_INTERNAL_SOURCE.dbt_change_type::text in ('update'::text, 'delete'::text)\n and {{ target }}.dbt_valid_to is null;\n\n insert into {{ target }} ({{ insert_cols_csv }})\n select {% for column in insert_cols -%}\n DBT_INTERNAL_SOURCE.{{ column }} {%- if not loop.last %}, {%- endif %}\n {%- endfor %}\n from {{ source }} as DBT_INTERNAL_SOURCE\n where DBT_INTERNAL_SOURCE.dbt_change_type::text = 'insert'::text;\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954464, "supported_languages": null}, "macro.dbt_postgres.postgres__drop_materialized_view": {"name": "postgres__drop_materialized_view", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/materialized_view/drop.sql", "original_file_path": "macros/relations/materialized_view/drop.sql", "unique_id": "macro.dbt_postgres.postgres__drop_materialized_view", "macro_sql": "{% macro postgres__drop_materialized_view(relation) -%}\n drop materialized view if exists {{ relation }} cascade\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954475, "supported_languages": null}, "macro.dbt_postgres.postgres__describe_materialized_view": {"name": "postgres__describe_materialized_view", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/materialized_view/describe.sql", "original_file_path": "macros/relations/materialized_view/describe.sql", "unique_id": "macro.dbt_postgres.postgres__describe_materialized_view", "macro_sql": "{% macro postgres__describe_materialized_view(relation) %}\n -- for now just get the indexes, we don't need the name or the query yet\n {% set _indexes = run_query(get_show_indexes_sql(relation)) %}\n {% do return({'indexes': _indexes}) %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.run_query", "macro.dbt.get_show_indexes_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9544861, "supported_languages": null}, "macro.dbt_postgres.postgres__refresh_materialized_view": {"name": "postgres__refresh_materialized_view", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/materialized_view/refresh.sql", "original_file_path": "macros/relations/materialized_view/refresh.sql", "unique_id": "macro.dbt_postgres.postgres__refresh_materialized_view", "macro_sql": "{% macro postgres__refresh_materialized_view(relation) %}\n refresh materialized view {{ relation }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9545002, "supported_languages": null}, "macro.dbt_postgres.postgres__get_rename_materialized_view_sql": {"name": "postgres__get_rename_materialized_view_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/materialized_view/rename.sql", "original_file_path": "macros/relations/materialized_view/rename.sql", "unique_id": "macro.dbt_postgres.postgres__get_rename_materialized_view_sql", "macro_sql": "{% macro postgres__get_rename_materialized_view_sql(relation, new_name) %}\n alter materialized view {{ relation }} rename to {{ new_name }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9545112, "supported_languages": null}, "macro.dbt_postgres.postgres__get_alter_materialized_view_as_sql": {"name": "postgres__get_alter_materialized_view_as_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/materialized_view/alter.sql", "original_file_path": "macros/relations/materialized_view/alter.sql", "unique_id": "macro.dbt_postgres.postgres__get_alter_materialized_view_as_sql", "macro_sql": "{% macro postgres__get_alter_materialized_view_as_sql(\n relation,\n configuration_changes,\n sql,\n existing_relation,\n backup_relation,\n intermediate_relation\n) %}\n\n -- apply a full refresh immediately if needed\n {% if configuration_changes.requires_full_refresh %}\n\n {{ get_replace_sql(existing_relation, relation, sql) }}\n\n -- otherwise apply individual changes as needed\n {% else %}\n\n {{ postgres__update_indexes_on_materialized_view(relation, configuration_changes.indexes) }}\n\n {%- endif -%}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_replace_sql", "macro.dbt_postgres.postgres__update_indexes_on_materialized_view"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954525, "supported_languages": null}, "macro.dbt_postgres.postgres__update_indexes_on_materialized_view": {"name": "postgres__update_indexes_on_materialized_view", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/materialized_view/alter.sql", "original_file_path": "macros/relations/materialized_view/alter.sql", "unique_id": "macro.dbt_postgres.postgres__update_indexes_on_materialized_view", "macro_sql": "\n\n\n{%- macro postgres__update_indexes_on_materialized_view(relation, index_changes) -%}\n {{- log(\"Applying UPDATE INDEXES to: \" ~ relation) -}}\n\n {%- for _index_change in index_changes -%}\n {%- set _index = _index_change.context -%}\n\n {%- if _index_change.action == \"drop\" -%}\n\n {{ postgres__get_drop_index_sql(relation, _index.name) }};\n\n {%- elif _index_change.action == \"create\" -%}\n\n {{ postgres__get_create_index_sql(relation, _index.as_node_config) }}\n\n {%- endif -%}\n\n {%- endfor -%}\n\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_drop_index_sql", "macro.dbt_postgres.postgres__get_create_index_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95453, "supported_languages": null}, "macro.dbt_postgres.postgres__get_materialized_view_configuration_changes": {"name": "postgres__get_materialized_view_configuration_changes", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/materialized_view/alter.sql", "original_file_path": "macros/relations/materialized_view/alter.sql", "unique_id": "macro.dbt_postgres.postgres__get_materialized_view_configuration_changes", "macro_sql": "{% macro postgres__get_materialized_view_configuration_changes(existing_relation, new_config) %}\n {% set _existing_materialized_view = postgres__describe_materialized_view(existing_relation) %}\n {% set _configuration_changes = existing_relation.get_materialized_view_config_change_collection(_existing_materialized_view, new_config.model) %}\n {% do return(_configuration_changes) %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__describe_materialized_view"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9545379, "supported_languages": null}, "macro.dbt_postgres.postgres__get_create_materialized_view_as_sql": {"name": "postgres__get_create_materialized_view_as_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/materialized_view/create.sql", "original_file_path": "macros/relations/materialized_view/create.sql", "unique_id": "macro.dbt_postgres.postgres__get_create_materialized_view_as_sql", "macro_sql": "{% macro postgres__get_create_materialized_view_as_sql(relation, sql) %}\n create materialized view if not exists {{ relation }} as {{ sql }};\n\n {% for _index_dict in config.get('indexes', []) -%}\n {{- get_create_index_sql(relation, _index_dict) -}}\n {%- endfor -%}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_create_index_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954549, "supported_languages": null}, "macro.dbt_postgres.postgres__drop_table": {"name": "postgres__drop_table", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/table/drop.sql", "original_file_path": "macros/relations/table/drop.sql", "unique_id": "macro.dbt_postgres.postgres__drop_table", "macro_sql": "{% macro postgres__drop_table(relation) -%}\n drop table if exists {{ relation }} cascade\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9545631, "supported_languages": null}, "macro.dbt_postgres.postgres__get_replace_table_sql": {"name": "postgres__get_replace_table_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/table/replace.sql", "original_file_path": "macros/relations/table/replace.sql", "unique_id": "macro.dbt_postgres.postgres__get_replace_table_sql", "macro_sql": "{% macro postgres__get_replace_table_sql(relation, sql) -%}\n\n {%- set sql_header = config.get('sql_header', none) -%}\n {{ sql_header if sql_header is not none }}\n\n create or replace table {{ relation }}\n {% set contract_config = config.get('contract') %}\n {% if contract_config.enforced %}\n {{ get_assert_columns_equivalent(sql) }}\n {{ get_table_columns_and_constraints() }}\n {%- set sql = get_select_subquery(sql) %}\n {% endif %}\n as (\n {{ sql }}\n );\n\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.get_assert_columns_equivalent", "macro.dbt.get_table_columns_and_constraints", "macro.dbt.get_select_subquery"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95458, "supported_languages": null}, "macro.dbt_postgres.postgres__get_rename_table_sql": {"name": "postgres__get_rename_table_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/table/rename.sql", "original_file_path": "macros/relations/table/rename.sql", "unique_id": "macro.dbt_postgres.postgres__get_rename_table_sql", "macro_sql": "{% macro postgres__get_rename_table_sql(relation, new_name) %}\n alter table {{ relation }} rename to {{ new_name }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954592, "supported_languages": null}, "macro.dbt_postgres.postgres__drop_view": {"name": "postgres__drop_view", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/view/drop.sql", "original_file_path": "macros/relations/view/drop.sql", "unique_id": "macro.dbt_postgres.postgres__drop_view", "macro_sql": "{% macro postgres__drop_view(relation) -%}\n drop view if exists {{ relation }} cascade\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954602, "supported_languages": null}, "macro.dbt_postgres.postgres__get_replace_view_sql": {"name": "postgres__get_replace_view_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/view/replace.sql", "original_file_path": "macros/relations/view/replace.sql", "unique_id": "macro.dbt_postgres.postgres__get_replace_view_sql", "macro_sql": "{% macro postgres__get_replace_view_sql(relation, sql) -%}\n\n {%- set sql_header = config.get('sql_header', none) -%}\n {{ sql_header if sql_header is not none }}\n\n create or replace view {{ relation }}\n {% set contract_config = config.get('contract') %}\n {% if contract_config.enforced %}\n {{ get_assert_columns_equivalent(sql) }}\n {%- endif %}\n as (\n {{ sql }}\n );\n\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.get_assert_columns_equivalent"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954616, "supported_languages": null}, "macro.dbt_postgres.postgres__get_rename_view_sql": {"name": "postgres__get_rename_view_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/view/rename.sql", "original_file_path": "macros/relations/view/rename.sql", "unique_id": "macro.dbt_postgres.postgres__get_rename_view_sql", "macro_sql": "{% macro postgres__get_rename_view_sql(relation, new_name) %}\n alter view {{ relation }} rename to {{ new_name }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954626, "supported_languages": null}, "macro.dbt_postgres.postgres__dateadd": {"name": "postgres__dateadd", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/utils/dateadd.sql", "original_file_path": "macros/utils/dateadd.sql", "unique_id": "macro.dbt_postgres.postgres__dateadd", "macro_sql": "{% macro postgres__dateadd(datepart, interval, from_date_or_timestamp) %}\n\n {{ from_date_or_timestamp }} + ((interval '1 {{ datepart }}') * ({{ interval }}))\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9546409, "supported_languages": null}, "macro.dbt_postgres.postgres__listagg": {"name": "postgres__listagg", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/utils/listagg.sql", "original_file_path": "macros/utils/listagg.sql", "unique_id": "macro.dbt_postgres.postgres__listagg", "macro_sql": "{% macro postgres__listagg(measure, delimiter_text, order_by_clause, limit_num) -%}\n\n {% if limit_num -%}\n array_to_string(\n (array_agg(\n {{ measure }}\n {% if order_by_clause -%}\n {{ order_by_clause }}\n {%- endif %}\n ))[1:{{ limit_num }}],\n {{ delimiter_text }}\n )\n {%- else %}\n string_agg(\n {{ measure }},\n {{ delimiter_text }}\n {% if order_by_clause -%}\n {{ order_by_clause }}\n {%- endif %}\n )\n {%- endif %}\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954655, "supported_languages": null}, "macro.dbt_postgres.postgres__datediff": {"name": "postgres__datediff", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/utils/datediff.sql", "original_file_path": "macros/utils/datediff.sql", "unique_id": "macro.dbt_postgres.postgres__datediff", "macro_sql": "{% macro postgres__datediff(first_date, second_date, datepart) -%}\n\n {% if datepart == 'year' %}\n (date_part('year', ({{second_date}})::date) - date_part('year', ({{first_date}})::date))\n {% elif datepart == 'quarter' %}\n ({{ datediff(first_date, second_date, 'year') }} * 4 + date_part('quarter', ({{second_date}})::date) - date_part('quarter', ({{first_date}})::date))\n {% elif datepart == 'month' %}\n ({{ datediff(first_date, second_date, 'year') }} * 12 + date_part('month', ({{second_date}})::date) - date_part('month', ({{first_date}})::date))\n {% elif datepart == 'day' %}\n (({{second_date}})::date - ({{first_date}})::date)\n {% elif datepart == 'week' %}\n ({{ datediff(first_date, second_date, 'day') }} / 7 + case\n when date_part('dow', ({{first_date}})::timestamp) <= date_part('dow', ({{second_date}})::timestamp) then\n case when {{first_date}} <= {{second_date}} then 0 else -1 end\n else\n case when {{first_date}} <= {{second_date}} then 1 else 0 end\n end)\n {% elif datepart == 'hour' %}\n ({{ datediff(first_date, second_date, 'day') }} * 24 + date_part('hour', ({{second_date}})::timestamp) - date_part('hour', ({{first_date}})::timestamp))\n {% elif datepart == 'minute' %}\n ({{ datediff(first_date, second_date, 'hour') }} * 60 + date_part('minute', ({{second_date}})::timestamp) - date_part('minute', ({{first_date}})::timestamp))\n {% elif datepart == 'second' %}\n ({{ datediff(first_date, second_date, 'minute') }} * 60 + floor(date_part('second', ({{second_date}})::timestamp)) - floor(date_part('second', ({{first_date}})::timestamp)))\n {% elif datepart == 'millisecond' %}\n ({{ datediff(first_date, second_date, 'minute') }} * 60000 + floor(date_part('millisecond', ({{second_date}})::timestamp)) - floor(date_part('millisecond', ({{first_date}})::timestamp)))\n {% elif datepart == 'microsecond' %}\n ({{ datediff(first_date, second_date, 'minute') }} * 60000000 + floor(date_part('microsecond', ({{second_date}})::timestamp)) - floor(date_part('microsecond', ({{first_date}})::timestamp)))\n {% else %}\n {{ exceptions.raise_compiler_error(\"Unsupported datepart for macro datediff in postgres: {!r}\".format(datepart)) }}\n {% endif %}\n\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.datediff"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954672, "supported_languages": null}, "macro.dbt_postgres.postgres__any_value": {"name": "postgres__any_value", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/utils/any_value.sql", "original_file_path": "macros/utils/any_value.sql", "unique_id": "macro.dbt_postgres.postgres__any_value", "macro_sql": "{% macro postgres__any_value(expression) -%}\n\n min({{ expression }})\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954685, "supported_languages": null}, "macro.dbt_postgres.postgres__last_day": {"name": "postgres__last_day", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/utils/last_day.sql", "original_file_path": "macros/utils/last_day.sql", "unique_id": "macro.dbt_postgres.postgres__last_day", "macro_sql": "{% macro postgres__last_day(date, datepart) -%}\n\n {%- if datepart == 'quarter' -%}\n -- postgres dateadd does not support quarter interval.\n cast(\n {{dbt.dateadd('day', '-1',\n dbt.dateadd('month', '3', dbt.date_trunc(datepart, date))\n )}}\n as date)\n {%- else -%}\n {{dbt.default_last_day(date, datepart)}}\n {%- endif -%}\n\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.dateadd", "macro.dbt.date_trunc", "macro.dbt.default_last_day"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954695, "supported_languages": null}, "macro.dbt_postgres.postgres__split_part": {"name": "postgres__split_part", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/utils/split_part.sql", "original_file_path": "macros/utils/split_part.sql", "unique_id": "macro.dbt_postgres.postgres__split_part", "macro_sql": "{% macro postgres__split_part(string_text, delimiter_text, part_number) %}\n\n {% if part_number >= 0 %}\n {{ dbt.default__split_part(string_text, delimiter_text, part_number) }}\n {% else %}\n {{ dbt._split_part_negative(string_text, delimiter_text, part_number) }}\n {% endif %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__split_part", "macro.dbt._split_part_negative"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954706, "supported_languages": null}, "macro.dbt.run_hooks": {"name": "run_hooks", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/hooks.sql", "original_file_path": "macros/materializations/hooks.sql", "unique_id": "macro.dbt.run_hooks", "macro_sql": "{% macro run_hooks(hooks, inside_transaction=True) %}\n {% for hook in hooks | selectattr('transaction', 'equalto', inside_transaction) %}\n {% if not inside_transaction and loop.first %}\n {% call statement(auto_begin=inside_transaction) %}\n commit;\n {% endcall %}\n {% endif %}\n {% set rendered = render(hook.get('sql')) | trim %}\n {% if (rendered | length) > 0 %}\n {% call statement(auto_begin=inside_transaction) %}\n {{ rendered }}\n {% endcall %}\n {% endif %}\n {% endfor %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954721, "supported_languages": null}, "macro.dbt.make_hook_config": {"name": "make_hook_config", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/hooks.sql", "original_file_path": "macros/materializations/hooks.sql", "unique_id": "macro.dbt.make_hook_config", "macro_sql": "{% macro make_hook_config(sql, inside_transaction) %}\n {{ tojson({\"sql\": sql, \"transaction\": inside_transaction}) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954727, "supported_languages": null}, "macro.dbt.before_begin": {"name": "before_begin", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/hooks.sql", "original_file_path": "macros/materializations/hooks.sql", "unique_id": "macro.dbt.before_begin", "macro_sql": "{% macro before_begin(sql) %}\n {{ make_hook_config(sql, inside_transaction=False) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.make_hook_config"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954734, "supported_languages": null}, "macro.dbt.in_transaction": {"name": "in_transaction", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/hooks.sql", "original_file_path": "macros/materializations/hooks.sql", "unique_id": "macro.dbt.in_transaction", "macro_sql": "{% macro in_transaction(sql) %}\n {{ make_hook_config(sql, inside_transaction=True) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.make_hook_config"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954742, "supported_languages": null}, "macro.dbt.after_commit": {"name": "after_commit", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/hooks.sql", "original_file_path": "macros/materializations/hooks.sql", "unique_id": "macro.dbt.after_commit", "macro_sql": "{% macro after_commit(sql) %}\n {{ make_hook_config(sql, inside_transaction=False) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.make_hook_config"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954747, "supported_languages": null}, "macro.dbt.set_sql_header": {"name": "set_sql_header", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/configs.sql", "original_file_path": "macros/materializations/configs.sql", "unique_id": "macro.dbt.set_sql_header", "macro_sql": "{% macro set_sql_header(config) -%}\n {{ config.set('sql_header', caller()) }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954758, "supported_languages": null}, "macro.dbt.should_full_refresh": {"name": "should_full_refresh", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/configs.sql", "original_file_path": "macros/materializations/configs.sql", "unique_id": "macro.dbt.should_full_refresh", "macro_sql": "{% macro should_full_refresh() %}\n {% set config_full_refresh = config.get('full_refresh') %}\n {% if config_full_refresh is none %}\n {% set config_full_refresh = flags.FULL_REFRESH %}\n {% endif %}\n {% do return(config_full_refresh) %}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9547641, "supported_languages": null}, "macro.dbt.should_store_failures": {"name": "should_store_failures", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/configs.sql", "original_file_path": "macros/materializations/configs.sql", "unique_id": "macro.dbt.should_store_failures", "macro_sql": "{% macro should_store_failures() %}\n {% set config_store_failures = config.get('store_failures') %}\n {% if config_store_failures is none %}\n {% set config_store_failures = flags.STORE_FAILURES %}\n {% endif %}\n {% do return(config_store_failures) %}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954772, "supported_languages": null}, "macro.dbt.snapshot_merge_sql": {"name": "snapshot_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/snapshot_merge.sql", "original_file_path": "macros/materializations/snapshots/snapshot_merge.sql", "unique_id": "macro.dbt.snapshot_merge_sql", "macro_sql": "{% macro snapshot_merge_sql(target, source, insert_cols) -%}\n {{ adapter.dispatch('snapshot_merge_sql', 'dbt')(target, source, insert_cols) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__snapshot_merge_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954792, "supported_languages": null}, "macro.dbt.default__snapshot_merge_sql": {"name": "default__snapshot_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/snapshot_merge.sql", "original_file_path": "macros/materializations/snapshots/snapshot_merge.sql", "unique_id": "macro.dbt.default__snapshot_merge_sql", "macro_sql": "{% macro default__snapshot_merge_sql(target, source, insert_cols) -%}\n {%- set insert_cols_csv = insert_cols | join(', ') -%}\n\n merge into {{ target }} as DBT_INTERNAL_DEST\n using {{ source }} as DBT_INTERNAL_SOURCE\n on DBT_INTERNAL_SOURCE.dbt_scd_id = DBT_INTERNAL_DEST.dbt_scd_id\n\n when matched\n and DBT_INTERNAL_DEST.dbt_valid_to is null\n and DBT_INTERNAL_SOURCE.dbt_change_type in ('update', 'delete')\n then update\n set dbt_valid_to = DBT_INTERNAL_SOURCE.dbt_valid_to\n\n when not matched\n and DBT_INTERNAL_SOURCE.dbt_change_type = 'insert'\n then insert ({{ insert_cols_csv }})\n values ({{ insert_cols_csv }})\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9548008, "supported_languages": null}, "macro.dbt.strategy_dispatch": {"name": "strategy_dispatch", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/strategies.sql", "original_file_path": "macros/materializations/snapshots/strategies.sql", "unique_id": "macro.dbt.strategy_dispatch", "macro_sql": "{% macro strategy_dispatch(name) -%}\n{% set original_name = name %}\n {% if '.' in name %}\n {% set package_name, name = name.split(\".\", 1) %}\n {% else %}\n {% set package_name = none %}\n {% endif %}\n\n {% if package_name is none %}\n {% set package_context = context %}\n {% elif package_name in context %}\n {% set package_context = context[package_name] %}\n {% else %}\n {% set error_msg %}\n Could not find package '{{package_name}}', called with '{{original_name}}'\n {% endset %}\n {{ exceptions.raise_compiler_error(error_msg | trim) }}\n {% endif %}\n\n {%- set search_name = 'snapshot_' ~ name ~ '_strategy' -%}\n\n {% if search_name not in package_context %}\n {% set error_msg %}\n The specified strategy macro '{{name}}' was not found in package '{{ package_name }}'\n {% endset %}\n {{ exceptions.raise_compiler_error(error_msg | trim) }}\n {% endif %}\n {{ return(package_context[search_name]) }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9548218, "supported_languages": null}, "macro.dbt.snapshot_hash_arguments": {"name": "snapshot_hash_arguments", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/strategies.sql", "original_file_path": "macros/materializations/snapshots/strategies.sql", "unique_id": "macro.dbt.snapshot_hash_arguments", "macro_sql": "{% macro snapshot_hash_arguments(args) -%}\n {{ adapter.dispatch('snapshot_hash_arguments', 'dbt')(args) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__snapshot_hash_arguments"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954828, "supported_languages": null}, "macro.dbt.default__snapshot_hash_arguments": {"name": "default__snapshot_hash_arguments", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/strategies.sql", "original_file_path": "macros/materializations/snapshots/strategies.sql", "unique_id": "macro.dbt.default__snapshot_hash_arguments", "macro_sql": "{% macro default__snapshot_hash_arguments(args) -%}\n md5({%- for arg in args -%}\n coalesce(cast({{ arg }} as varchar ), '')\n {% if not loop.last %} || '|' || {% endif %}\n {%- endfor -%})\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954836, "supported_languages": null}, "macro.dbt.snapshot_timestamp_strategy": {"name": "snapshot_timestamp_strategy", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/strategies.sql", "original_file_path": "macros/materializations/snapshots/strategies.sql", "unique_id": "macro.dbt.snapshot_timestamp_strategy", "macro_sql": "{% macro snapshot_timestamp_strategy(node, snapshotted_rel, current_rel, config, target_exists) %}\n {% set primary_key = config['unique_key'] %}\n {% set updated_at = config['updated_at'] %}\n {% set invalidate_hard_deletes = config.get('invalidate_hard_deletes', false) %}\n\n {#/*\n The snapshot relation might not have an {{ updated_at }} value if the\n snapshot strategy is changed from `check` to `timestamp`. We\n should use a dbt-created column for the comparison in the snapshot\n table instead of assuming that the user-supplied {{ updated_at }}\n will be present in the historical data.\n\n See https://github.com/dbt-labs/dbt-core/issues/2350\n */ #}\n {% set row_changed_expr -%}\n ({{ snapshotted_rel }}.dbt_valid_from < {{ current_rel }}.{{ updated_at }})\n {%- endset %}\n\n {% set scd_id_expr = snapshot_hash_arguments([primary_key, updated_at]) %}\n\n {% do return({\n \"unique_key\": primary_key,\n \"updated_at\": updated_at,\n \"row_changed\": row_changed_expr,\n \"scd_id\": scd_id_expr,\n \"invalidate_hard_deletes\": invalidate_hard_deletes\n }) %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.snapshot_hash_arguments"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9548512, "supported_languages": null}, "macro.dbt.snapshot_string_as_time": {"name": "snapshot_string_as_time", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/strategies.sql", "original_file_path": "macros/materializations/snapshots/strategies.sql", "unique_id": "macro.dbt.snapshot_string_as_time", "macro_sql": "{% macro snapshot_string_as_time(timestamp) -%}\n {{ adapter.dispatch('snapshot_string_as_time', 'dbt')(timestamp) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__snapshot_string_as_time"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954856, "supported_languages": null}, "macro.dbt.default__snapshot_string_as_time": {"name": "default__snapshot_string_as_time", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/strategies.sql", "original_file_path": "macros/materializations/snapshots/strategies.sql", "unique_id": "macro.dbt.default__snapshot_string_as_time", "macro_sql": "{% macro default__snapshot_string_as_time(timestamp) %}\n {% do exceptions.raise_not_implemented(\n 'snapshot_string_as_time macro not implemented for adapter '+adapter.type()\n ) %}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954865, "supported_languages": null}, "macro.dbt.snapshot_check_all_get_existing_columns": {"name": "snapshot_check_all_get_existing_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/strategies.sql", "original_file_path": "macros/materializations/snapshots/strategies.sql", "unique_id": "macro.dbt.snapshot_check_all_get_existing_columns", "macro_sql": "{% macro snapshot_check_all_get_existing_columns(node, target_exists, check_cols_config) -%}\n {%- if not target_exists -%}\n {#-- no table yet -> return whatever the query does --#}\n {{ return((false, query_columns)) }}\n {%- endif -%}\n\n {#-- handle any schema changes --#}\n {%- set target_relation = adapter.get_relation(database=node.database, schema=node.schema, identifier=node.alias) -%}\n\n {% if check_cols_config == 'all' %}\n {%- set query_columns = get_columns_in_query(node['compiled_code']) -%}\n\n {% elif check_cols_config is iterable and (check_cols_config | length) > 0 %}\n {#-- query for proper casing/quoting, to support comparison below --#}\n {%- set select_check_cols_from_target -%}\n {#-- N.B. The whitespace below is necessary to avoid edge case issue with comments --#}\n {#-- See: https://github.com/dbt-labs/dbt-core/issues/6781 --#}\n select {{ check_cols_config | join(', ') }} from (\n {{ node['compiled_code'] }}\n ) subq\n {%- endset -%}\n {% set query_columns = get_columns_in_query(select_check_cols_from_target) %}\n\n {% else %}\n {% do exceptions.raise_compiler_error(\"Invalid value for 'check_cols': \" ~ check_cols_config) %}\n {% endif %}\n\n {%- set existing_cols = adapter.get_columns_in_relation(target_relation) | map(attribute = 'name') | list -%}\n {%- set ns = namespace() -%} {#-- handle for-loop scoping with a namespace --#}\n {%- set ns.column_added = false -%}\n\n {%- set intersection = [] -%}\n {%- for col in query_columns -%}\n {%- if col in existing_cols -%}\n {%- do intersection.append(adapter.quote(col)) -%}\n {%- else -%}\n {% set ns.column_added = true %}\n {%- endif -%}\n {%- endfor -%}\n {{ return((ns.column_added, intersection)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.get_columns_in_query"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954873, "supported_languages": null}, "macro.dbt.snapshot_check_strategy": {"name": "snapshot_check_strategy", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/strategies.sql", "original_file_path": "macros/materializations/snapshots/strategies.sql", "unique_id": "macro.dbt.snapshot_check_strategy", "macro_sql": "{% macro snapshot_check_strategy(node, snapshotted_rel, current_rel, config, target_exists) %}\n {% set check_cols_config = config['check_cols'] %}\n {% set primary_key = config['unique_key'] %}\n {% set invalidate_hard_deletes = config.get('invalidate_hard_deletes', false) %}\n {% set updated_at = config.get('updated_at', snapshot_get_time()) %}\n\n {% set column_added = false %}\n\n {% set column_added, check_cols = snapshot_check_all_get_existing_columns(node, target_exists, check_cols_config) %}\n\n {%- set row_changed_expr -%}\n (\n {%- if column_added -%}\n {{ get_true_sql() }}\n {%- else -%}\n {%- for col in check_cols -%}\n {{ snapshotted_rel }}.{{ col }} != {{ current_rel }}.{{ col }}\n or\n (\n (({{ snapshotted_rel }}.{{ col }} is null) and not ({{ current_rel }}.{{ col }} is null))\n or\n ((not {{ snapshotted_rel }}.{{ col }} is null) and ({{ current_rel }}.{{ col }} is null))\n )\n {%- if not loop.last %} or {% endif -%}\n {%- endfor -%}\n {%- endif -%}\n )\n {%- endset %}\n\n {% set scd_id_expr = snapshot_hash_arguments([primary_key, updated_at]) %}\n\n {% do return({\n \"unique_key\": primary_key,\n \"updated_at\": updated_at,\n \"row_changed\": row_changed_expr,\n \"scd_id\": scd_id_expr,\n \"invalidate_hard_deletes\": invalidate_hard_deletes\n }) %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.snapshot_get_time", "macro.dbt.snapshot_check_all_get_existing_columns", "macro.dbt.get_true_sql", "macro.dbt.snapshot_hash_arguments"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9548821, "supported_languages": null}, "macro.dbt.create_columns": {"name": "create_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.create_columns", "macro_sql": "{% macro create_columns(relation, columns) %}\n {{ adapter.dispatch('create_columns', 'dbt')(relation, columns) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__create_columns"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9549062, "supported_languages": null}, "macro.dbt.default__create_columns": {"name": "default__create_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.default__create_columns", "macro_sql": "{% macro default__create_columns(relation, columns) %}\n {% for column in columns %}\n {% call statement() %}\n alter table {{ relation }} add column \"{{ column.name }}\" {{ column.data_type }};\n {% endcall %}\n {% endfor %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954915, "supported_languages": null}, "macro.dbt.post_snapshot": {"name": "post_snapshot", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.post_snapshot", "macro_sql": "{% macro post_snapshot(staging_relation) %}\n {{ adapter.dispatch('post_snapshot', 'dbt')(staging_relation) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__post_snapshot"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954925, "supported_languages": null}, "macro.dbt.default__post_snapshot": {"name": "default__post_snapshot", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.default__post_snapshot", "macro_sql": "{% macro default__post_snapshot(staging_relation) %}\n {# no-op #}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954931, "supported_languages": null}, "macro.dbt.get_true_sql": {"name": "get_true_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.get_true_sql", "macro_sql": "{% macro get_true_sql() %}\n {{ adapter.dispatch('get_true_sql', 'dbt')() }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_true_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954937, "supported_languages": null}, "macro.dbt.default__get_true_sql": {"name": "default__get_true_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.default__get_true_sql", "macro_sql": "{% macro default__get_true_sql() %}\n {{ return('TRUE') }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954944, "supported_languages": null}, "macro.dbt.snapshot_staging_table": {"name": "snapshot_staging_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.snapshot_staging_table", "macro_sql": "{% macro snapshot_staging_table(strategy, source_sql, target_relation) -%}\n {{ adapter.dispatch('snapshot_staging_table', 'dbt')(strategy, source_sql, target_relation) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__snapshot_staging_table"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.954953, "supported_languages": null}, "macro.dbt.default__snapshot_staging_table": {"name": "default__snapshot_staging_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.default__snapshot_staging_table", "macro_sql": "{% macro default__snapshot_staging_table(strategy, source_sql, target_relation) -%}\n\n with snapshot_query as (\n\n {{ source_sql }}\n\n ),\n\n snapshotted_data as (\n\n select *,\n {{ strategy.unique_key }} as dbt_unique_key\n\n from {{ target_relation }}\n where dbt_valid_to is null\n\n ),\n\n insertions_source_data as (\n\n select\n *,\n {{ strategy.unique_key }} as dbt_unique_key,\n {{ strategy.updated_at }} as dbt_updated_at,\n {{ strategy.updated_at }} as dbt_valid_from,\n nullif({{ strategy.updated_at }}, {{ strategy.updated_at }}) as dbt_valid_to,\n {{ strategy.scd_id }} as dbt_scd_id\n\n from snapshot_query\n ),\n\n updates_source_data as (\n\n select\n *,\n {{ strategy.unique_key }} as dbt_unique_key,\n {{ strategy.updated_at }} as dbt_updated_at,\n {{ strategy.updated_at }} as dbt_valid_from,\n {{ strategy.updated_at }} as dbt_valid_to\n\n from snapshot_query\n ),\n\n {%- if strategy.invalidate_hard_deletes %}\n\n deletes_source_data as (\n\n select\n *,\n {{ strategy.unique_key }} as dbt_unique_key\n from snapshot_query\n ),\n {% endif %}\n\n insertions as (\n\n select\n 'insert' as dbt_change_type,\n source_data.*\n\n from insertions_source_data as source_data\n left outer join snapshotted_data on snapshotted_data.dbt_unique_key = source_data.dbt_unique_key\n where snapshotted_data.dbt_unique_key is null\n or (\n snapshotted_data.dbt_unique_key is not null\n and (\n {{ strategy.row_changed }}\n )\n )\n\n ),\n\n updates as (\n\n select\n 'update' as dbt_change_type,\n source_data.*,\n snapshotted_data.dbt_scd_id\n\n from updates_source_data as source_data\n join snapshotted_data on snapshotted_data.dbt_unique_key = source_data.dbt_unique_key\n where (\n {{ strategy.row_changed }}\n )\n )\n\n {%- if strategy.invalidate_hard_deletes -%}\n ,\n\n deletes as (\n\n select\n 'delete' as dbt_change_type,\n source_data.*,\n {{ snapshot_get_time() }} as dbt_valid_from,\n {{ snapshot_get_time() }} as dbt_updated_at,\n {{ snapshot_get_time() }} as dbt_valid_to,\n snapshotted_data.dbt_scd_id\n\n from snapshotted_data\n left join deletes_source_data as source_data on snapshotted_data.dbt_unique_key = source_data.dbt_unique_key\n where source_data.dbt_unique_key is null\n )\n {%- endif %}\n\n select * from insertions\n union all\n select * from updates\n {%- if strategy.invalidate_hard_deletes %}\n union all\n select * from deletes\n {%- endif %}\n\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.snapshot_get_time"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9549599, "supported_languages": null}, "macro.dbt.build_snapshot_table": {"name": "build_snapshot_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.build_snapshot_table", "macro_sql": "{% macro build_snapshot_table(strategy, sql) -%}\n {{ adapter.dispatch('build_snapshot_table', 'dbt')(strategy, sql) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__build_snapshot_table"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9549649, "supported_languages": null}, "macro.dbt.default__build_snapshot_table": {"name": "default__build_snapshot_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.default__build_snapshot_table", "macro_sql": "{% macro default__build_snapshot_table(strategy, sql) %}\n\n select *,\n {{ strategy.scd_id }} as dbt_scd_id,\n {{ strategy.updated_at }} as dbt_updated_at,\n {{ strategy.updated_at }} as dbt_valid_from,\n nullif({{ strategy.updated_at }}, {{ strategy.updated_at }}) as dbt_valid_to\n from (\n {{ sql }}\n ) sbq\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9549701, "supported_languages": null}, "macro.dbt.build_snapshot_staging_table": {"name": "build_snapshot_staging_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.build_snapshot_staging_table", "macro_sql": "{% macro build_snapshot_staging_table(strategy, sql, target_relation) %}\n {% set temp_relation = make_temp_relation(target_relation) %}\n\n {% set select = snapshot_staging_table(strategy, sql, target_relation) %}\n\n {% call statement('build_snapshot_staging_relation') %}\n {{ create_table_as(True, temp_relation, select) }}\n {% endcall %}\n\n {% do return(temp_relation) %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.make_temp_relation", "macro.dbt.snapshot_staging_table", "macro.dbt.statement", "macro.dbt.create_table_as"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9550028, "supported_languages": null}, "macro.dbt.materialization_snapshot_default": {"name": "materialization_snapshot_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/snapshot.sql", "original_file_path": "macros/materializations/snapshots/snapshot.sql", "unique_id": "macro.dbt.materialization_snapshot_default", "macro_sql": "{% materialization snapshot, default %}\n {%- set config = model['config'] -%}\n\n {%- set target_table = model.get('alias', model.get('name')) -%}\n\n {%- set strategy_name = config.get('strategy') -%}\n {%- set unique_key = config.get('unique_key') %}\n -- grab current tables grants config for comparision later on\n {%- set grant_config = config.get('grants') -%}\n\n {% set target_relation_exists, target_relation = get_or_create_relation(\n database=model.database,\n schema=model.schema,\n identifier=target_table,\n type='table') -%}\n\n {%- if not target_relation.is_table -%}\n {% do exceptions.relation_wrong_type(target_relation, 'table') %}\n {%- endif -%}\n\n\n {{ run_hooks(pre_hooks, inside_transaction=False) }}\n\n {{ run_hooks(pre_hooks, inside_transaction=True) }}\n\n {% set strategy_macro = strategy_dispatch(strategy_name) %}\n {% set strategy = strategy_macro(model, \"snapshotted_data\", \"source_data\", config, target_relation_exists) %}\n\n {% if not target_relation_exists %}\n\n {% set build_sql = build_snapshot_table(strategy, model['compiled_code']) %}\n {% set final_sql = create_table_as(False, target_relation, build_sql) %}\n\n {% else %}\n\n {{ adapter.valid_snapshot_target(target_relation) }}\n\n {% set staging_table = build_snapshot_staging_table(strategy, sql, target_relation) %}\n\n -- this may no-op if the database does not require column expansion\n {% do adapter.expand_target_column_types(from_relation=staging_table,\n to_relation=target_relation) %}\n\n {% set missing_columns = adapter.get_missing_columns(staging_table, target_relation)\n | rejectattr('name', 'equalto', 'dbt_change_type')\n | rejectattr('name', 'equalto', 'DBT_CHANGE_TYPE')\n | rejectattr('name', 'equalto', 'dbt_unique_key')\n | rejectattr('name', 'equalto', 'DBT_UNIQUE_KEY')\n | list %}\n\n {% do create_columns(target_relation, missing_columns) %}\n\n {% set source_columns = adapter.get_columns_in_relation(staging_table)\n | rejectattr('name', 'equalto', 'dbt_change_type')\n | rejectattr('name', 'equalto', 'DBT_CHANGE_TYPE')\n | rejectattr('name', 'equalto', 'dbt_unique_key')\n | rejectattr('name', 'equalto', 'DBT_UNIQUE_KEY')\n | list %}\n\n {% set quoted_source_columns = [] %}\n {% for column in source_columns %}\n {% do quoted_source_columns.append(adapter.quote(column.name)) %}\n {% endfor %}\n\n {% set final_sql = snapshot_merge_sql(\n target = target_relation,\n source = staging_table,\n insert_cols = quoted_source_columns\n )\n %}\n\n {% endif %}\n\n {% call statement('main') %}\n {{ final_sql }}\n {% endcall %}\n\n {% set should_revoke = should_revoke(target_relation_exists, full_refresh_mode=False) %}\n {% do apply_grants(target_relation, grant_config, should_revoke=should_revoke) %}\n\n {% do persist_docs(target_relation, model) %}\n\n {% if not target_relation_exists %}\n {% do create_indexes(target_relation) %}\n {% endif %}\n\n {{ run_hooks(post_hooks, inside_transaction=True) }}\n\n {{ adapter.commit() }}\n\n {% if staging_table is defined %}\n {% do post_snapshot(staging_table) %}\n {% endif %}\n\n {{ run_hooks(post_hooks, inside_transaction=False) }}\n\n {{ return({'relations': [target_relation]}) }}\n\n{% endmaterialization %}", "depends_on": {"macros": ["macro.dbt.get_or_create_relation", "macro.dbt.run_hooks", "macro.dbt.strategy_dispatch", "macro.dbt.build_snapshot_table", "macro.dbt.create_table_as", "macro.dbt.build_snapshot_staging_table", "macro.dbt.create_columns", "macro.dbt.snapshot_merge_sql", "macro.dbt.statement", "macro.dbt.should_revoke", "macro.dbt.apply_grants", "macro.dbt.persist_docs", "macro.dbt.create_indexes", "macro.dbt.post_snapshot"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9550211, "supported_languages": ["sql"]}, "macro.dbt.materialization_test_default": {"name": "materialization_test_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/tests/test.sql", "original_file_path": "macros/materializations/tests/test.sql", "unique_id": "macro.dbt.materialization_test_default", "macro_sql": "{%- materialization test, default -%}\n\n {% set relations = [] %}\n\n {% if should_store_failures() %}\n\n {% set identifier = model['alias'] %}\n {% set old_relation = adapter.get_relation(database=database, schema=schema, identifier=identifier) %}\n\n {% set store_failures_as = config.get('store_failures_as') %}\n -- if `--store-failures` is invoked via command line and `store_failures_as` is not set,\n -- config.get('store_failures_as', 'table') returns None, not 'table'\n {% if store_failures_as == none %}{% set store_failures_as = 'table' %}{% endif %}\n {% if store_failures_as not in ['table', 'view'] %}\n {{ exceptions.raise_compiler_error(\n \"'\" ~ store_failures_as ~ \"' is not a valid value for `store_failures_as`. \"\n \"Accepted values are: ['ephemeral', 'table', 'view']\"\n ) }}\n {% endif %}\n\n {% set target_relation = api.Relation.create(\n identifier=identifier, schema=schema, database=database, type=store_failures_as) -%} %}\n\n {% if old_relation %}\n {% do adapter.drop_relation(old_relation) %}\n {% endif %}\n\n {% call statement(auto_begin=True) %}\n {{ get_create_sql(target_relation, sql) }}\n {% endcall %}\n\n {% do relations.append(target_relation) %}\n\n {% set main_sql %}\n select *\n from {{ target_relation }}\n {% endset %}\n\n {{ adapter.commit() }}\n\n {% else %}\n\n {% set main_sql = sql %}\n\n {% endif %}\n\n {% set limit = config.get('limit') %}\n {% set fail_calc = config.get('fail_calc') %}\n {% set warn_if = config.get('warn_if') %}\n {% set error_if = config.get('error_if') %}\n\n {% call statement('main', fetch_result=True) -%}\n\n {{ get_test_sql(main_sql, fail_calc, warn_if, error_if, limit)}}\n\n {%- endcall %}\n\n {{ return({'relations': relations}) }}\n\n{%- endmaterialization -%}", "depends_on": {"macros": ["macro.dbt.should_store_failures", "macro.dbt.statement", "macro.dbt.get_create_sql", "macro.dbt.get_test_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95505, "supported_languages": ["sql"]}, "macro.dbt.get_test_sql": {"name": "get_test_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/tests/helpers.sql", "original_file_path": "macros/materializations/tests/helpers.sql", "unique_id": "macro.dbt.get_test_sql", "macro_sql": "{% macro get_test_sql(main_sql, fail_calc, warn_if, error_if, limit) -%}\n {{ adapter.dispatch('get_test_sql', 'dbt')(main_sql, fail_calc, warn_if, error_if, limit) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_test_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955074, "supported_languages": null}, "macro.dbt.default__get_test_sql": {"name": "default__get_test_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/tests/helpers.sql", "original_file_path": "macros/materializations/tests/helpers.sql", "unique_id": "macro.dbt.default__get_test_sql", "macro_sql": "{% macro default__get_test_sql(main_sql, fail_calc, warn_if, error_if, limit) -%}\n select\n {{ fail_calc }} as failures,\n {{ fail_calc }} {{ warn_if }} as should_warn,\n {{ fail_calc }} {{ error_if }} as should_error\n from (\n {{ main_sql }}\n {{ \"limit \" ~ limit if limit != none }}\n ) dbt_internal_test\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955084, "supported_languages": null}, "macro.dbt.get_unit_test_sql": {"name": "get_unit_test_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/tests/helpers.sql", "original_file_path": "macros/materializations/tests/helpers.sql", "unique_id": "macro.dbt.get_unit_test_sql", "macro_sql": "{% macro get_unit_test_sql(main_sql, expected_fixture_sql, expected_column_names) -%}\n {{ adapter.dispatch('get_unit_test_sql', 'dbt')(main_sql, expected_fixture_sql, expected_column_names) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_unit_test_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955091, "supported_languages": null}, "macro.dbt.default__get_unit_test_sql": {"name": "default__get_unit_test_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/tests/helpers.sql", "original_file_path": "macros/materializations/tests/helpers.sql", "unique_id": "macro.dbt.default__get_unit_test_sql", "macro_sql": "{% macro default__get_unit_test_sql(main_sql, expected_fixture_sql, expected_column_names) -%}\n-- Build actual result given inputs\nwith dbt_internal_unit_test_actual as (\n select\n {% for expected_column_name in expected_column_names %}{{expected_column_name}}{% if not loop.last -%},{% endif %}{%- endfor -%}, {{ dbt.string_literal(\"actual\") }} as {{ adapter.quote(\"actual_or_expected\") }}\n from (\n {{ main_sql }}\n ) _dbt_internal_unit_test_actual\n),\n-- Build expected result\ndbt_internal_unit_test_expected as (\n select\n {% for expected_column_name in expected_column_names %}{{expected_column_name}}{% if not loop.last -%}, {% endif %}{%- endfor -%}, {{ dbt.string_literal(\"expected\") }} as {{ adapter.quote(\"actual_or_expected\") }}\n from (\n {{ expected_fixture_sql }}\n ) _dbt_internal_unit_test_expected\n)\n-- Union actual and expected results\nselect * from dbt_internal_unit_test_actual\nunion all\nselect * from dbt_internal_unit_test_expected\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.string_literal"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9550982, "supported_languages": null}, "macro.dbt.get_where_subquery": {"name": "get_where_subquery", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/tests/where_subquery.sql", "original_file_path": "macros/materializations/tests/where_subquery.sql", "unique_id": "macro.dbt.get_where_subquery", "macro_sql": "{% macro get_where_subquery(relation) -%}\n {% do return(adapter.dispatch('get_where_subquery', 'dbt')(relation)) %}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_where_subquery"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955111, "supported_languages": null}, "macro.dbt.default__get_where_subquery": {"name": "default__get_where_subquery", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/tests/where_subquery.sql", "original_file_path": "macros/materializations/tests/where_subquery.sql", "unique_id": "macro.dbt.default__get_where_subquery", "macro_sql": "{% macro default__get_where_subquery(relation) -%}\n {% set where = config.get('where', '') %}\n {% if where %}\n {%- set filtered -%}\n (select * from {{ relation }} where {{ where }}) dbt_subquery\n {%- endset -%}\n {% do return(filtered) %}\n {%- else -%}\n {% do return(relation) %}\n {%- endif -%}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955116, "supported_languages": null}, "macro.dbt.materialization_unit_default": {"name": "materialization_unit_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/tests/unit.sql", "original_file_path": "macros/materializations/tests/unit.sql", "unique_id": "macro.dbt.materialization_unit_default", "macro_sql": "{%- materialization unit, default -%}\n\n {% set relations = [] %}\n\n {% set expected_rows = config.get('expected_rows') %}\n {% set expected_sql = config.get('expected_sql') %}\n {% set tested_expected_column_names = expected_rows[0].keys() if (expected_rows | length ) > 0 else get_columns_in_query(sql) %} %}\n\n {%- set target_relation = this.incorporate(type='table') -%}\n {%- set temp_relation = make_temp_relation(target_relation)-%}\n {% do run_query(get_create_table_as_sql(True, temp_relation, get_empty_subquery_sql(sql))) %}\n {%- set columns_in_relation = adapter.get_columns_in_relation(temp_relation) -%}\n {%- set column_name_to_data_types = {} -%}\n {%- for column in columns_in_relation -%}\n {%- do column_name_to_data_types.update({column.name|lower: column.data_type}) -%}\n {%- endfor -%}\n\n {% if not expected_sql %}\n {% set expected_sql = get_expected_sql(expected_rows, column_name_to_data_types) %}\n {% endif %}\n {% set unit_test_sql = get_unit_test_sql(sql, expected_sql, tested_expected_column_names) %}\n\n {% call statement('main', fetch_result=True) -%}\n\n {{ unit_test_sql }}\n\n {%- endcall %}\n\n {% do adapter.drop_relation(temp_relation) %}\n\n {{ return({'relations': relations}) }}\n\n{%- endmaterialization -%}", "depends_on": {"macros": ["macro.dbt.get_columns_in_query", "macro.dbt.make_temp_relation", "macro.dbt.run_query", "macro.dbt.get_create_table_as_sql", "macro.dbt.get_empty_subquery_sql", "macro.dbt.get_expected_sql", "macro.dbt.get_unit_test_sql", "macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955128, "supported_languages": ["sql"]}, "macro.dbt.materialization_materialized_view_default": {"name": "materialization_materialized_view_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/materialized_view.sql", "original_file_path": "macros/materializations/models/materialized_view.sql", "unique_id": "macro.dbt.materialization_materialized_view_default", "macro_sql": "{% materialization materialized_view, default %}\n {% set existing_relation = load_cached_relation(this) %}\n {% set target_relation = this.incorporate(type=this.MaterializedView) %}\n {% set intermediate_relation = make_intermediate_relation(target_relation) %}\n {% set backup_relation_type = target_relation.MaterializedView if existing_relation is none else existing_relation.type %}\n {% set backup_relation = make_backup_relation(target_relation, backup_relation_type) %}\n\n {{ materialized_view_setup(backup_relation, intermediate_relation, pre_hooks) }}\n\n {% set build_sql = materialized_view_get_build_sql(existing_relation, target_relation, backup_relation, intermediate_relation) %}\n\n {% if build_sql == '' %}\n {{ materialized_view_execute_no_op(target_relation) }}\n {% else %}\n {{ materialized_view_execute_build_sql(build_sql, existing_relation, target_relation, post_hooks) }}\n {% endif %}\n\n {{ materialized_view_teardown(backup_relation, intermediate_relation, post_hooks) }}\n\n {{ return({'relations': [target_relation]}) }}\n\n{% endmaterialization %}", "depends_on": {"macros": ["macro.dbt.load_cached_relation", "macro.dbt.make_intermediate_relation", "macro.dbt.make_backup_relation", "macro.dbt.materialized_view_setup", "macro.dbt.materialized_view_get_build_sql", "macro.dbt.materialized_view_execute_no_op", "macro.dbt.materialized_view_execute_build_sql", "macro.dbt.materialized_view_teardown"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9551501, "supported_languages": ["sql"]}, "macro.dbt.materialized_view_setup": {"name": "materialized_view_setup", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/materialized_view.sql", "original_file_path": "macros/materializations/models/materialized_view.sql", "unique_id": "macro.dbt.materialized_view_setup", "macro_sql": "{% macro materialized_view_setup(backup_relation, intermediate_relation, pre_hooks) %}\n\n -- backup_relation and intermediate_relation should not already exist in the database\n -- it's possible these exist because of a previous run that exited unexpectedly\n {% set preexisting_backup_relation = load_cached_relation(backup_relation) %}\n {% set preexisting_intermediate_relation = load_cached_relation(intermediate_relation) %}\n\n -- drop the temp relations if they exist already in the database\n {{ drop_relation_if_exists(preexisting_backup_relation) }}\n {{ drop_relation_if_exists(preexisting_intermediate_relation) }}\n\n {{ run_hooks(pre_hooks, inside_transaction=False) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.load_cached_relation", "macro.dbt.drop_relation_if_exists", "macro.dbt.run_hooks"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9551601, "supported_languages": null}, "macro.dbt.materialized_view_teardown": {"name": "materialized_view_teardown", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/materialized_view.sql", "original_file_path": "macros/materializations/models/materialized_view.sql", "unique_id": "macro.dbt.materialized_view_teardown", "macro_sql": "{% macro materialized_view_teardown(backup_relation, intermediate_relation, post_hooks) %}\n\n -- drop the temp relations if they exist to leave the database clean for the next run\n {{ drop_relation_if_exists(backup_relation) }}\n {{ drop_relation_if_exists(intermediate_relation) }}\n\n {{ run_hooks(post_hooks, inside_transaction=False) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.drop_relation_if_exists", "macro.dbt.run_hooks"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955165, "supported_languages": null}, "macro.dbt.materialized_view_get_build_sql": {"name": "materialized_view_get_build_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/materialized_view.sql", "original_file_path": "macros/materializations/models/materialized_view.sql", "unique_id": "macro.dbt.materialized_view_get_build_sql", "macro_sql": "{% macro materialized_view_get_build_sql(existing_relation, target_relation, backup_relation, intermediate_relation) %}\n\n {% set full_refresh_mode = should_full_refresh() %}\n\n -- determine the scenario we're in: create, full_refresh, alter, refresh data\n {% if existing_relation is none %}\n {% set build_sql = get_create_materialized_view_as_sql(target_relation, sql) %}\n {% elif full_refresh_mode or not existing_relation.is_materialized_view %}\n {% set build_sql = get_replace_sql(existing_relation, target_relation, sql) %}\n {% else %}\n\n -- get config options\n {% set on_configuration_change = config.get('on_configuration_change') %}\n {% set configuration_changes = get_materialized_view_configuration_changes(existing_relation, config) %}\n\n {% if configuration_changes is none %}\n {% set build_sql = refresh_materialized_view(target_relation) %}\n\n {% elif on_configuration_change == 'apply' %}\n {% set build_sql = get_alter_materialized_view_as_sql(target_relation, configuration_changes, sql, existing_relation, backup_relation, intermediate_relation) %}\n {% elif on_configuration_change == 'continue' %}\n {% set build_sql = '' %}\n {{ exceptions.warn(\"Configuration changes were identified and `on_configuration_change` was set to `continue` for `\" ~ target_relation ~ \"`\") }}\n {% elif on_configuration_change == 'fail' %}\n {{ exceptions.raise_fail_fast_error(\"Configuration changes were identified and `on_configuration_change` was set to `fail` for `\" ~ target_relation ~ \"`\") }}\n\n {% else %}\n -- this only happens if the user provides a value other than `apply`, 'skip', 'fail'\n {{ exceptions.raise_compiler_error(\"Unexpected configuration scenario\") }}\n\n {% endif %}\n\n {% endif %}\n\n {% do return(build_sql) %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.should_full_refresh", "macro.dbt.get_create_materialized_view_as_sql", "macro.dbt.get_replace_sql", "macro.dbt.get_materialized_view_configuration_changes", "macro.dbt.refresh_materialized_view", "macro.dbt.get_alter_materialized_view_as_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955171, "supported_languages": null}, "macro.dbt.materialized_view_execute_no_op": {"name": "materialized_view_execute_no_op", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/materialized_view.sql", "original_file_path": "macros/materializations/models/materialized_view.sql", "unique_id": "macro.dbt.materialized_view_execute_no_op", "macro_sql": "{% macro materialized_view_execute_no_op(target_relation) %}\n {% do store_raw_result(\n name=\"main\",\n message=\"skip \" ~ target_relation,\n code=\"skip\",\n rows_affected=\"-1\"\n ) %}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955182, "supported_languages": null}, "macro.dbt.materialized_view_execute_build_sql": {"name": "materialized_view_execute_build_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/materialized_view.sql", "original_file_path": "macros/materializations/models/materialized_view.sql", "unique_id": "macro.dbt.materialized_view_execute_build_sql", "macro_sql": "{% macro materialized_view_execute_build_sql(build_sql, existing_relation, target_relation, post_hooks) %}\n\n -- `BEGIN` happens here:\n {{ run_hooks(pre_hooks, inside_transaction=True) }}\n\n {% set grant_config = config.get('grants') %}\n\n {% call statement(name=\"main\") %}\n {{ build_sql }}\n {% endcall %}\n\n {% set should_revoke = should_revoke(existing_relation, full_refresh_mode=True) %}\n {% do apply_grants(target_relation, grant_config, should_revoke=should_revoke) %}\n\n {% do persist_docs(target_relation, model) %}\n\n {{ run_hooks(post_hooks, inside_transaction=True) }}\n\n {{ adapter.commit() }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.run_hooks", "macro.dbt.statement", "macro.dbt.should_revoke", "macro.dbt.apply_grants", "macro.dbt.persist_docs"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9551911, "supported_languages": null}, "macro.dbt.materialization_view_default": {"name": "materialization_view_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/view.sql", "original_file_path": "macros/materializations/models/view.sql", "unique_id": "macro.dbt.materialization_view_default", "macro_sql": "{%- materialization view, default -%}\n\n {%- set existing_relation = load_cached_relation(this) -%}\n {%- set target_relation = this.incorporate(type='view') -%}\n {%- set intermediate_relation = make_intermediate_relation(target_relation) -%}\n\n -- the intermediate_relation should not already exist in the database; get_relation\n -- will return None in that case. Otherwise, we get a relation that we can drop\n -- later, before we try to use this name for the current operation\n {%- set preexisting_intermediate_relation = load_cached_relation(intermediate_relation) -%}\n /*\n This relation (probably) doesn't exist yet. If it does exist, it's a leftover from\n a previous run, and we're going to try to drop it immediately. At the end of this\n materialization, we're going to rename the \"existing_relation\" to this identifier,\n and then we're going to drop it. In order to make sure we run the correct one of:\n - drop view ...\n - drop table ...\n\n We need to set the type of this relation to be the type of the existing_relation, if it exists,\n or else \"view\" as a sane default if it does not. Note that if the existing_relation does not\n exist, then there is nothing to move out of the way and subsequentally drop. In that case,\n this relation will be effectively unused.\n */\n {%- set backup_relation_type = 'view' if existing_relation is none else existing_relation.type -%}\n {%- set backup_relation = make_backup_relation(target_relation, backup_relation_type) -%}\n -- as above, the backup_relation should not already exist\n {%- set preexisting_backup_relation = load_cached_relation(backup_relation) -%}\n -- grab current tables grants config for comparision later on\n {% set grant_config = config.get('grants') %}\n\n {{ run_hooks(pre_hooks, inside_transaction=False) }}\n\n -- drop the temp relations if they exist already in the database\n {{ drop_relation_if_exists(preexisting_intermediate_relation) }}\n {{ drop_relation_if_exists(preexisting_backup_relation) }}\n\n -- `BEGIN` happens here:\n {{ run_hooks(pre_hooks, inside_transaction=True) }}\n\n -- build model\n {% call statement('main') -%}\n {{ get_create_view_as_sql(intermediate_relation, sql) }}\n {%- endcall %}\n\n -- cleanup\n -- move the existing view out of the way\n {% if existing_relation is not none %}\n /* Do the equivalent of rename_if_exists. 'existing_relation' could have been dropped\n since the variable was first set. */\n {% set existing_relation = load_cached_relation(existing_relation) %}\n {% if existing_relation is not none %}\n {{ adapter.rename_relation(existing_relation, backup_relation) }}\n {% endif %}\n {% endif %}\n {{ adapter.rename_relation(intermediate_relation, target_relation) }}\n\n {% set should_revoke = should_revoke(existing_relation, full_refresh_mode=True) %}\n {% do apply_grants(target_relation, grant_config, should_revoke=should_revoke) %}\n\n {% do persist_docs(target_relation, model) %}\n\n {{ run_hooks(post_hooks, inside_transaction=True) }}\n\n {{ adapter.commit() }}\n\n {{ drop_relation_if_exists(backup_relation) }}\n\n {{ run_hooks(post_hooks, inside_transaction=False) }}\n\n {{ return({'relations': [target_relation]}) }}\n\n{%- endmaterialization -%}", "depends_on": {"macros": ["macro.dbt.load_cached_relation", "macro.dbt.make_intermediate_relation", "macro.dbt.make_backup_relation", "macro.dbt.run_hooks", "macro.dbt.drop_relation_if_exists", "macro.dbt.statement", "macro.dbt.get_create_view_as_sql", "macro.dbt.should_revoke", "macro.dbt.apply_grants", "macro.dbt.persist_docs"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9552078, "supported_languages": ["sql"]}, "macro.dbt.materialization_table_default": {"name": "materialization_table_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/table.sql", "original_file_path": "macros/materializations/models/table.sql", "unique_id": "macro.dbt.materialization_table_default", "macro_sql": "{% materialization table, default %}\n\n {%- set existing_relation = load_cached_relation(this) -%}\n {%- set target_relation = this.incorporate(type='table') %}\n {%- set intermediate_relation = make_intermediate_relation(target_relation) -%}\n -- the intermediate_relation should not already exist in the database; get_relation\n -- will return None in that case. Otherwise, we get a relation that we can drop\n -- later, before we try to use this name for the current operation\n {%- set preexisting_intermediate_relation = load_cached_relation(intermediate_relation) -%}\n /*\n See ../view/view.sql for more information about this relation.\n */\n {%- set backup_relation_type = 'table' if existing_relation is none else existing_relation.type -%}\n {%- set backup_relation = make_backup_relation(target_relation, backup_relation_type) -%}\n -- as above, the backup_relation should not already exist\n {%- set preexisting_backup_relation = load_cached_relation(backup_relation) -%}\n -- grab current tables grants config for comparision later on\n {% set grant_config = config.get('grants') %}\n\n -- drop the temp relations if they exist already in the database\n {{ drop_relation_if_exists(preexisting_intermediate_relation) }}\n {{ drop_relation_if_exists(preexisting_backup_relation) }}\n\n {{ run_hooks(pre_hooks, inside_transaction=False) }}\n\n -- `BEGIN` happens here:\n {{ run_hooks(pre_hooks, inside_transaction=True) }}\n\n -- build model\n {% call statement('main') -%}\n {{ get_create_table_as_sql(False, intermediate_relation, sql) }}\n {%- endcall %}\n\n -- cleanup\n {% if existing_relation is not none %}\n /* Do the equivalent of rename_if_exists. 'existing_relation' could have been dropped\n since the variable was first set. */\n {% set existing_relation = load_cached_relation(existing_relation) %}\n {% if existing_relation is not none %}\n {{ adapter.rename_relation(existing_relation, backup_relation) }}\n {% endif %}\n {% endif %}\n\n {{ adapter.rename_relation(intermediate_relation, target_relation) }}\n\n {% do create_indexes(target_relation) %}\n\n {{ run_hooks(post_hooks, inside_transaction=True) }}\n\n {% set should_revoke = should_revoke(existing_relation, full_refresh_mode=True) %}\n {% do apply_grants(target_relation, grant_config, should_revoke=should_revoke) %}\n\n {% do persist_docs(target_relation, model) %}\n\n -- `COMMIT` happens here\n {{ adapter.commit() }}\n\n -- finally, drop the existing/backup relation after the commit\n {{ drop_relation_if_exists(backup_relation) }}\n\n {{ run_hooks(post_hooks, inside_transaction=False) }}\n\n {{ return({'relations': [target_relation]}) }}\n{% endmaterialization %}", "depends_on": {"macros": ["macro.dbt.load_cached_relation", "macro.dbt.make_intermediate_relation", "macro.dbt.make_backup_relation", "macro.dbt.drop_relation_if_exists", "macro.dbt.run_hooks", "macro.dbt.statement", "macro.dbt.get_create_table_as_sql", "macro.dbt.create_indexes", "macro.dbt.should_revoke", "macro.dbt.apply_grants", "macro.dbt.persist_docs"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955231, "supported_languages": ["sql"]}, "macro.dbt.get_quoted_csv": {"name": "get_quoted_csv", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/column_helpers.sql", "original_file_path": "macros/materializations/models/incremental/column_helpers.sql", "unique_id": "macro.dbt.get_quoted_csv", "macro_sql": "{% macro get_quoted_csv(column_names) %}\n\n {% set quoted = [] %}\n {% for col in column_names -%}\n {%- do quoted.append(adapter.quote(col)) -%}\n {%- endfor %}\n\n {%- set dest_cols_csv = quoted | join(', ') -%}\n {{ return(dest_cols_csv) }}\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9552588, "supported_languages": null}, "macro.dbt.diff_columns": {"name": "diff_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/column_helpers.sql", "original_file_path": "macros/materializations/models/incremental/column_helpers.sql", "unique_id": "macro.dbt.diff_columns", "macro_sql": "{% macro diff_columns(source_columns, target_columns) %}\n\n {% set result = [] %}\n {% set source_names = source_columns | map(attribute = 'column') | list %}\n {% set target_names = target_columns | map(attribute = 'column') | list %}\n\n {# --check whether the name attribute exists in the target - this does not perform a data type check #}\n {% for sc in source_columns %}\n {% if sc.name not in target_names %}\n {{ result.append(sc) }}\n {% endif %}\n {% endfor %}\n\n {{ return(result) }}\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955266, "supported_languages": null}, "macro.dbt.diff_column_data_types": {"name": "diff_column_data_types", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/column_helpers.sql", "original_file_path": "macros/materializations/models/incremental/column_helpers.sql", "unique_id": "macro.dbt.diff_column_data_types", "macro_sql": "{% macro diff_column_data_types(source_columns, target_columns) %}\n\n {% set result = [] %}\n {% for sc in source_columns %}\n {% set tc = target_columns | selectattr(\"name\", \"equalto\", sc.name) | list | first %}\n {% if tc %}\n {% if sc.data_type != tc.data_type and not sc.can_expand_to(other_column=tc) %}\n {{ result.append( { 'column_name': tc.name, 'new_type': sc.data_type } ) }}\n {% endif %}\n {% endif %}\n {% endfor %}\n\n {{ return(result) }}\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955271, "supported_languages": null}, "macro.dbt.get_merge_update_columns": {"name": "get_merge_update_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/column_helpers.sql", "original_file_path": "macros/materializations/models/incremental/column_helpers.sql", "unique_id": "macro.dbt.get_merge_update_columns", "macro_sql": "{% macro get_merge_update_columns(merge_update_columns, merge_exclude_columns, dest_columns) %}\n {{ return(adapter.dispatch('get_merge_update_columns', 'dbt')(merge_update_columns, merge_exclude_columns, dest_columns)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_merge_update_columns"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955279, "supported_languages": null}, "macro.dbt.default__get_merge_update_columns": {"name": "default__get_merge_update_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/column_helpers.sql", "original_file_path": "macros/materializations/models/incremental/column_helpers.sql", "unique_id": "macro.dbt.default__get_merge_update_columns", "macro_sql": "{% macro default__get_merge_update_columns(merge_update_columns, merge_exclude_columns, dest_columns) %}\n {%- set default_cols = dest_columns | map(attribute=\"quoted\") | list -%}\n\n {%- if merge_update_columns and merge_exclude_columns -%}\n {{ exceptions.raise_compiler_error(\n 'Model cannot specify merge_update_columns and merge_exclude_columns. Please update model to use only one config'\n )}}\n {%- elif merge_update_columns -%}\n {%- set update_columns = merge_update_columns -%}\n {%- elif merge_exclude_columns -%}\n {%- set update_columns = [] -%}\n {%- for column in dest_columns -%}\n {% if column.column | lower not in merge_exclude_columns | map(\"lower\") | list %}\n {%- do update_columns.append(column.quoted) -%}\n {% endif %}\n {%- endfor -%}\n {%- else -%}\n {%- set update_columns = default_cols -%}\n {%- endif -%}\n\n {{ return(update_columns) }}\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9552839, "supported_languages": null}, "macro.dbt.get_merge_sql": {"name": "get_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/merge.sql", "original_file_path": "macros/materializations/models/incremental/merge.sql", "unique_id": "macro.dbt.get_merge_sql", "macro_sql": "{% macro get_merge_sql(target, source, unique_key, dest_columns, incremental_predicates=none) -%}\n -- back compat for old kwarg name\n {% set incremental_predicates = kwargs.get('predicates', incremental_predicates) %}\n {{ adapter.dispatch('get_merge_sql', 'dbt')(target, source, unique_key, dest_columns, incremental_predicates) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_merge_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955302, "supported_languages": null}, "macro.dbt.default__get_merge_sql": {"name": "default__get_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/merge.sql", "original_file_path": "macros/materializations/models/incremental/merge.sql", "unique_id": "macro.dbt.default__get_merge_sql", "macro_sql": "{% macro default__get_merge_sql(target, source, unique_key, dest_columns, incremental_predicates=none) -%}\n {%- set predicates = [] if incremental_predicates is none else [] + incremental_predicates -%}\n {%- set dest_cols_csv = get_quoted_csv(dest_columns | map(attribute=\"name\")) -%}\n {%- set merge_update_columns = config.get('merge_update_columns') -%}\n {%- set merge_exclude_columns = config.get('merge_exclude_columns') -%}\n {%- set update_columns = get_merge_update_columns(merge_update_columns, merge_exclude_columns, dest_columns) -%}\n {%- set sql_header = config.get('sql_header', none) -%}\n\n {% if unique_key %}\n {% if unique_key is sequence and unique_key is not mapping and unique_key is not string %}\n {% for key in unique_key %}\n {% set this_key_match %}\n DBT_INTERNAL_SOURCE.{{ key }} = DBT_INTERNAL_DEST.{{ key }}\n {% endset %}\n {% do predicates.append(this_key_match) %}\n {% endfor %}\n {% else %}\n {% set unique_key_match %}\n DBT_INTERNAL_SOURCE.{{ unique_key }} = DBT_INTERNAL_DEST.{{ unique_key }}\n {% endset %}\n {% do predicates.append(unique_key_match) %}\n {% endif %}\n {% else %}\n {% do predicates.append('FALSE') %}\n {% endif %}\n\n {{ sql_header if sql_header is not none }}\n\n merge into {{ target }} as DBT_INTERNAL_DEST\n using {{ source }} as DBT_INTERNAL_SOURCE\n on {{\"(\" ~ predicates | join(\") and (\") ~ \")\"}}\n\n {% if unique_key %}\n when matched then update set\n {% for column_name in update_columns -%}\n {{ column_name }} = DBT_INTERNAL_SOURCE.{{ column_name }}\n {%- if not loop.last %}, {%- endif %}\n {%- endfor %}\n {% endif %}\n\n when not matched then insert\n ({{ dest_cols_csv }})\n values\n ({{ dest_cols_csv }})\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_quoted_csv", "macro.dbt.get_merge_update_columns"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955311, "supported_languages": null}, "macro.dbt.get_delete_insert_merge_sql": {"name": "get_delete_insert_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/merge.sql", "original_file_path": "macros/materializations/models/incremental/merge.sql", "unique_id": "macro.dbt.get_delete_insert_merge_sql", "macro_sql": "{% macro get_delete_insert_merge_sql(target, source, unique_key, dest_columns, incremental_predicates) -%}\n {{ adapter.dispatch('get_delete_insert_merge_sql', 'dbt')(target, source, unique_key, dest_columns, incremental_predicates) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_delete_insert_merge_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955322, "supported_languages": null}, "macro.dbt.default__get_delete_insert_merge_sql": {"name": "default__get_delete_insert_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/merge.sql", "original_file_path": "macros/materializations/models/incremental/merge.sql", "unique_id": "macro.dbt.default__get_delete_insert_merge_sql", "macro_sql": "{% macro default__get_delete_insert_merge_sql(target, source, unique_key, dest_columns, incremental_predicates) -%}\n\n {%- set dest_cols_csv = get_quoted_csv(dest_columns | map(attribute=\"name\")) -%}\n\n {% if unique_key %}\n {% if unique_key is sequence and unique_key is not string %}\n delete from {{target }}\n using {{ source }}\n where (\n {% for key in unique_key %}\n {{ source }}.{{ key }} = {{ target }}.{{ key }}\n {{ \"and \" if not loop.last}}\n {% endfor %}\n {% if incremental_predicates %}\n {% for predicate in incremental_predicates %}\n and {{ predicate }}\n {% endfor %}\n {% endif %}\n );\n {% else %}\n delete from {{ target }}\n where (\n {{ unique_key }}) in (\n select ({{ unique_key }})\n from {{ source }}\n )\n {%- if incremental_predicates %}\n {% for predicate in incremental_predicates %}\n and {{ predicate }}\n {% endfor %}\n {%- endif -%};\n\n {% endif %}\n {% endif %}\n\n insert into {{ target }} ({{ dest_cols_csv }})\n (\n select {{ dest_cols_csv }}\n from {{ source }}\n )\n\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.get_quoted_csv"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955331, "supported_languages": null}, "macro.dbt.get_insert_overwrite_merge_sql": {"name": "get_insert_overwrite_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/merge.sql", "original_file_path": "macros/materializations/models/incremental/merge.sql", "unique_id": "macro.dbt.get_insert_overwrite_merge_sql", "macro_sql": "{% macro get_insert_overwrite_merge_sql(target, source, dest_columns, predicates, include_sql_header=false) -%}\n {{ adapter.dispatch('get_insert_overwrite_merge_sql', 'dbt')(target, source, dest_columns, predicates, include_sql_header) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_insert_overwrite_merge_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955342, "supported_languages": null}, "macro.dbt.default__get_insert_overwrite_merge_sql": {"name": "default__get_insert_overwrite_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/merge.sql", "original_file_path": "macros/materializations/models/incremental/merge.sql", "unique_id": "macro.dbt.default__get_insert_overwrite_merge_sql", "macro_sql": "{% macro default__get_insert_overwrite_merge_sql(target, source, dest_columns, predicates, include_sql_header) -%}\n {#-- The only time include_sql_header is True: --#}\n {#-- BigQuery + insert_overwrite strategy + \"static\" partitions config --#}\n {#-- We should consider including the sql header at the materialization level instead --#}\n\n {%- set predicates = [] if predicates is none else [] + predicates -%}\n {%- set dest_cols_csv = get_quoted_csv(dest_columns | map(attribute=\"name\")) -%}\n {%- set sql_header = config.get('sql_header', none) -%}\n\n {{ sql_header if sql_header is not none and include_sql_header }}\n\n merge into {{ target }} as DBT_INTERNAL_DEST\n using {{ source }} as DBT_INTERNAL_SOURCE\n on FALSE\n\n when not matched by source\n {% if predicates %} and {{ predicates | join(' and ') }} {% endif %}\n then delete\n\n when not matched then insert\n ({{ dest_cols_csv }})\n values\n ({{ dest_cols_csv }})\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_quoted_csv"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955348, "supported_languages": null}, "macro.dbt.is_incremental": {"name": "is_incremental", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/is_incremental.sql", "original_file_path": "macros/materializations/models/incremental/is_incremental.sql", "unique_id": "macro.dbt.is_incremental", "macro_sql": "{% macro is_incremental() %}\n {#-- do not run introspective queries in parsing #}\n {% if not execute %}\n {{ return(False) }}\n {% else %}\n {% set relation = adapter.get_relation(this.database, this.schema, this.table) %}\n {{ return(relation is not none\n and relation.type == 'table'\n and model.config.materialized == 'incremental'\n and not should_full_refresh()) }}\n {% endif %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.should_full_refresh"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9553668, "supported_languages": null}, "macro.dbt.get_incremental_append_sql": {"name": "get_incremental_append_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.get_incremental_append_sql", "macro_sql": "{% macro get_incremental_append_sql(arg_dict) %}\n\n {{ return(adapter.dispatch('get_incremental_append_sql', 'dbt')(arg_dict)) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_incremental_append_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955382, "supported_languages": null}, "macro.dbt.default__get_incremental_append_sql": {"name": "default__get_incremental_append_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.default__get_incremental_append_sql", "macro_sql": "{% macro default__get_incremental_append_sql(arg_dict) %}\n\n {% do return(get_insert_into_sql(arg_dict[\"target_relation\"], arg_dict[\"temp_relation\"], arg_dict[\"dest_columns\"])) %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_insert_into_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9553878, "supported_languages": null}, "macro.dbt.get_incremental_delete_insert_sql": {"name": "get_incremental_delete_insert_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.get_incremental_delete_insert_sql", "macro_sql": "{% macro get_incremental_delete_insert_sql(arg_dict) %}\n\n {{ return(adapter.dispatch('get_incremental_delete_insert_sql', 'dbt')(arg_dict)) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_incremental_delete_insert_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9553962, "supported_languages": null}, "macro.dbt.default__get_incremental_delete_insert_sql": {"name": "default__get_incremental_delete_insert_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.default__get_incremental_delete_insert_sql", "macro_sql": "{% macro default__get_incremental_delete_insert_sql(arg_dict) %}\n\n {% do return(get_delete_insert_merge_sql(arg_dict[\"target_relation\"], arg_dict[\"temp_relation\"], arg_dict[\"unique_key\"], arg_dict[\"dest_columns\"], arg_dict[\"incremental_predicates\"])) %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_delete_insert_merge_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955402, "supported_languages": null}, "macro.dbt.get_incremental_merge_sql": {"name": "get_incremental_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.get_incremental_merge_sql", "macro_sql": "{% macro get_incremental_merge_sql(arg_dict) %}\n\n {{ return(adapter.dispatch('get_incremental_merge_sql', 'dbt')(arg_dict)) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_incremental_merge_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9554088, "supported_languages": null}, "macro.dbt.default__get_incremental_merge_sql": {"name": "default__get_incremental_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.default__get_incremental_merge_sql", "macro_sql": "{% macro default__get_incremental_merge_sql(arg_dict) %}\n\n {% do return(get_merge_sql(arg_dict[\"target_relation\"], arg_dict[\"temp_relation\"], arg_dict[\"unique_key\"], arg_dict[\"dest_columns\"], arg_dict[\"incremental_predicates\"])) %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_merge_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955414, "supported_languages": null}, "macro.dbt.get_incremental_insert_overwrite_sql": {"name": "get_incremental_insert_overwrite_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.get_incremental_insert_overwrite_sql", "macro_sql": "{% macro get_incremental_insert_overwrite_sql(arg_dict) %}\n\n {{ return(adapter.dispatch('get_incremental_insert_overwrite_sql', 'dbt')(arg_dict)) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_incremental_insert_overwrite_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955422, "supported_languages": null}, "macro.dbt.default__get_incremental_insert_overwrite_sql": {"name": "default__get_incremental_insert_overwrite_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.default__get_incremental_insert_overwrite_sql", "macro_sql": "{% macro default__get_incremental_insert_overwrite_sql(arg_dict) %}\n\n {% do return(get_insert_overwrite_merge_sql(arg_dict[\"target_relation\"], arg_dict[\"temp_relation\"], arg_dict[\"dest_columns\"], arg_dict[\"incremental_predicates\"])) %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_insert_overwrite_merge_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9554272, "supported_languages": null}, "macro.dbt.get_incremental_default_sql": {"name": "get_incremental_default_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.get_incremental_default_sql", "macro_sql": "{% macro get_incremental_default_sql(arg_dict) %}\n\n {{ return(adapter.dispatch('get_incremental_default_sql', 'dbt')(arg_dict)) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_incremental_default_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955432, "supported_languages": null}, "macro.dbt.default__get_incremental_default_sql": {"name": "default__get_incremental_default_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.default__get_incremental_default_sql", "macro_sql": "{% macro default__get_incremental_default_sql(arg_dict) %}\n\n {% do return(get_incremental_append_sql(arg_dict)) %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_incremental_append_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9554381, "supported_languages": null}, "macro.dbt.get_insert_into_sql": {"name": "get_insert_into_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.get_insert_into_sql", "macro_sql": "{% macro get_insert_into_sql(target_relation, temp_relation, dest_columns) %}\n\n {%- set dest_cols_csv = get_quoted_csv(dest_columns | map(attribute=\"name\")) -%}\n\n insert into {{ target_relation }} ({{ dest_cols_csv }})\n (\n select {{ dest_cols_csv }}\n from {{ temp_relation }}\n )\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_quoted_csv"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955447, "supported_languages": null}, "macro.dbt.materialization_incremental_default": {"name": "materialization_incremental_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/incremental.sql", "original_file_path": "macros/materializations/models/incremental/incremental.sql", "unique_id": "macro.dbt.materialization_incremental_default", "macro_sql": "{% materialization incremental, default -%}\n\n -- relations\n {%- set existing_relation = load_cached_relation(this) -%}\n {%- set target_relation = this.incorporate(type='table') -%}\n {%- set temp_relation = make_temp_relation(target_relation)-%}\n {%- set intermediate_relation = make_intermediate_relation(target_relation)-%}\n {%- set backup_relation_type = 'table' if existing_relation is none else existing_relation.type -%}\n {%- set backup_relation = make_backup_relation(target_relation, backup_relation_type) -%}\n\n -- configs\n {%- set unique_key = config.get('unique_key') -%}\n {%- set full_refresh_mode = (should_full_refresh() or existing_relation.is_view) -%}\n {%- set on_schema_change = incremental_validate_on_schema_change(config.get('on_schema_change'), default='ignore') -%}\n\n -- the temp_ and backup_ relations should not already exist in the database; get_relation\n -- will return None in that case. Otherwise, we get a relation that we can drop\n -- later, before we try to use this name for the current operation. This has to happen before\n -- BEGIN, in a separate transaction\n {%- set preexisting_intermediate_relation = load_cached_relation(intermediate_relation)-%}\n {%- set preexisting_backup_relation = load_cached_relation(backup_relation) -%}\n -- grab current tables grants config for comparision later on\n {% set grant_config = config.get('grants') %}\n {{ drop_relation_if_exists(preexisting_intermediate_relation) }}\n {{ drop_relation_if_exists(preexisting_backup_relation) }}\n\n {{ run_hooks(pre_hooks, inside_transaction=False) }}\n\n -- `BEGIN` happens here:\n {{ run_hooks(pre_hooks, inside_transaction=True) }}\n\n {% set to_drop = [] %}\n\n {% if existing_relation is none %}\n {% set build_sql = get_create_table_as_sql(False, target_relation, sql) %}\n {% elif full_refresh_mode %}\n {% set build_sql = get_create_table_as_sql(False, intermediate_relation, sql) %}\n {% set need_swap = true %}\n {% else %}\n {% do run_query(get_create_table_as_sql(True, temp_relation, sql)) %}\n {% do adapter.expand_target_column_types(\n from_relation=temp_relation,\n to_relation=target_relation) %}\n {#-- Process schema changes. Returns dict of changes if successful. Use source columns for upserting/merging --#}\n {% set dest_columns = process_schema_changes(on_schema_change, temp_relation, existing_relation) %}\n {% if not dest_columns %}\n {% set dest_columns = adapter.get_columns_in_relation(existing_relation) %}\n {% endif %}\n\n {#-- Get the incremental_strategy, the macro to use for the strategy, and build the sql --#}\n {% set incremental_strategy = config.get('incremental_strategy') or 'default' %}\n {% set incremental_predicates = config.get('predicates', none) or config.get('incremental_predicates', none) %}\n {% set strategy_sql_macro_func = adapter.get_incremental_strategy_macro(context, incremental_strategy) %}\n {% set strategy_arg_dict = ({'target_relation': target_relation, 'temp_relation': temp_relation, 'unique_key': unique_key, 'dest_columns': dest_columns, 'incremental_predicates': incremental_predicates }) %}\n {% set build_sql = strategy_sql_macro_func(strategy_arg_dict) %}\n\n {% endif %}\n\n {% call statement(\"main\") %}\n {{ build_sql }}\n {% endcall %}\n\n {% if need_swap %}\n {% do adapter.rename_relation(target_relation, backup_relation) %}\n {% do adapter.rename_relation(intermediate_relation, target_relation) %}\n {% do to_drop.append(backup_relation) %}\n {% endif %}\n\n {% set should_revoke = should_revoke(existing_relation, full_refresh_mode) %}\n {% do apply_grants(target_relation, grant_config, should_revoke=should_revoke) %}\n\n {% do persist_docs(target_relation, model) %}\n\n {% if existing_relation is none or existing_relation.is_view or should_full_refresh() %}\n {% do create_indexes(target_relation) %}\n {% endif %}\n\n {{ run_hooks(post_hooks, inside_transaction=True) }}\n\n -- `COMMIT` happens here\n {% do adapter.commit() %}\n\n {% for rel in to_drop %}\n {% do adapter.drop_relation(rel) %}\n {% endfor %}\n\n {{ run_hooks(post_hooks, inside_transaction=False) }}\n\n {{ return({'relations': [target_relation]}) }}\n\n{%- endmaterialization %}", "depends_on": {"macros": ["macro.dbt.load_cached_relation", "macro.dbt.make_temp_relation", "macro.dbt.make_intermediate_relation", "macro.dbt.make_backup_relation", "macro.dbt.should_full_refresh", "macro.dbt.incremental_validate_on_schema_change", "macro.dbt.drop_relation_if_exists", "macro.dbt.run_hooks", "macro.dbt.get_create_table_as_sql", "macro.dbt.run_query", "macro.dbt.process_schema_changes", "macro.dbt.statement", "macro.dbt.should_revoke", "macro.dbt.apply_grants", "macro.dbt.persist_docs", "macro.dbt.create_indexes"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95547, "supported_languages": ["sql"]}, "macro.dbt.incremental_validate_on_schema_change": {"name": "incremental_validate_on_schema_change", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/on_schema_change.sql", "original_file_path": "macros/materializations/models/incremental/on_schema_change.sql", "unique_id": "macro.dbt.incremental_validate_on_schema_change", "macro_sql": "{% macro incremental_validate_on_schema_change(on_schema_change, default='ignore') %}\n\n {% if on_schema_change not in ['sync_all_columns', 'append_new_columns', 'fail', 'ignore'] %}\n\n {% set log_message = 'Invalid value for on_schema_change (%s) specified. Setting default value of %s.' % (on_schema_change, default) %}\n {% do log(log_message) %}\n\n {{ return(default) }}\n\n {% else %}\n\n {{ return(on_schema_change) }}\n\n {% endif %}\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9554932, "supported_languages": null}, "macro.dbt.check_for_schema_changes": {"name": "check_for_schema_changes", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/on_schema_change.sql", "original_file_path": "macros/materializations/models/incremental/on_schema_change.sql", "unique_id": "macro.dbt.check_for_schema_changes", "macro_sql": "{% macro check_for_schema_changes(source_relation, target_relation) %}\n\n {% set schema_changed = False %}\n\n {%- set source_columns = adapter.get_columns_in_relation(source_relation) -%}\n {%- set target_columns = adapter.get_columns_in_relation(target_relation) -%}\n {%- set source_not_in_target = diff_columns(source_columns, target_columns) -%}\n {%- set target_not_in_source = diff_columns(target_columns, source_columns) -%}\n\n {% set new_target_types = diff_column_data_types(source_columns, target_columns) %}\n\n {% if source_not_in_target != [] %}\n {% set schema_changed = True %}\n {% elif target_not_in_source != [] or new_target_types != [] %}\n {% set schema_changed = True %}\n {% elif new_target_types != [] %}\n {% set schema_changed = True %}\n {% endif %}\n\n {% set changes_dict = {\n 'schema_changed': schema_changed,\n 'source_not_in_target': source_not_in_target,\n 'target_not_in_source': target_not_in_source,\n 'source_columns': source_columns,\n 'target_columns': target_columns,\n 'new_target_types': new_target_types\n } %}\n\n {% set msg %}\n In {{ target_relation }}:\n Schema changed: {{ schema_changed }}\n Source columns not in target: {{ source_not_in_target }}\n Target columns not in source: {{ target_not_in_source }}\n New column types: {{ new_target_types }}\n {% endset %}\n\n {% do log(msg) %}\n\n {{ return(changes_dict) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.diff_columns", "macro.dbt.diff_column_data_types"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955505, "supported_languages": null}, "macro.dbt.sync_column_schemas": {"name": "sync_column_schemas", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/on_schema_change.sql", "original_file_path": "macros/materializations/models/incremental/on_schema_change.sql", "unique_id": "macro.dbt.sync_column_schemas", "macro_sql": "{% macro sync_column_schemas(on_schema_change, target_relation, schema_changes_dict) %}\n\n {%- set add_to_target_arr = schema_changes_dict['source_not_in_target'] -%}\n\n {%- if on_schema_change == 'append_new_columns'-%}\n {%- if add_to_target_arr | length > 0 -%}\n {%- do alter_relation_add_remove_columns(target_relation, add_to_target_arr, none) -%}\n {%- endif -%}\n\n {% elif on_schema_change == 'sync_all_columns' %}\n {%- set remove_from_target_arr = schema_changes_dict['target_not_in_source'] -%}\n {%- set new_target_types = schema_changes_dict['new_target_types'] -%}\n\n {% if add_to_target_arr | length > 0 or remove_from_target_arr | length > 0 %}\n {%- do alter_relation_add_remove_columns(target_relation, add_to_target_arr, remove_from_target_arr) -%}\n {% endif %}\n\n {% if new_target_types != [] %}\n {% for ntt in new_target_types %}\n {% set column_name = ntt['column_name'] %}\n {% set new_type = ntt['new_type'] %}\n {% do alter_column_type(target_relation, column_name, new_type) %}\n {% endfor %}\n {% endif %}\n\n {% endif %}\n\n {% set schema_change_message %}\n In {{ target_relation }}:\n Schema change approach: {{ on_schema_change }}\n Columns added: {{ add_to_target_arr }}\n Columns removed: {{ remove_from_target_arr }}\n Data types changed: {{ new_target_types }}\n {% endset %}\n\n {% do log(schema_change_message) %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.alter_relation_add_remove_columns", "macro.dbt.alter_column_type"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955515, "supported_languages": null}, "macro.dbt.process_schema_changes": {"name": "process_schema_changes", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/on_schema_change.sql", "original_file_path": "macros/materializations/models/incremental/on_schema_change.sql", "unique_id": "macro.dbt.process_schema_changes", "macro_sql": "{% macro process_schema_changes(on_schema_change, source_relation, target_relation) %}\n\n {% if on_schema_change == 'ignore' %}\n\n {{ return({}) }}\n\n {% else %}\n\n {% set schema_changes_dict = check_for_schema_changes(source_relation, target_relation) %}\n\n {% if schema_changes_dict['schema_changed'] %}\n\n {% if on_schema_change == 'fail' %}\n\n {% set fail_msg %}\n The source and target schemas on this incremental model are out of sync!\n They can be reconciled in several ways:\n - set the `on_schema_change` config to either append_new_columns or sync_all_columns, depending on your situation.\n - Re-run the incremental model with `full_refresh: True` to update the target schema.\n - update the schema manually and re-run the process.\n\n Additional troubleshooting context:\n Source columns not in target: {{ schema_changes_dict['source_not_in_target'] }}\n Target columns not in source: {{ schema_changes_dict['target_not_in_source'] }}\n New column types: {{ schema_changes_dict['new_target_types'] }}\n {% endset %}\n\n {% do exceptions.raise_compiler_error(fail_msg) %}\n\n {# -- unless we ignore, run the sync operation per the config #}\n {% else %}\n\n {% do sync_column_schemas(on_schema_change, target_relation, schema_changes_dict) %}\n\n {% endif %}\n\n {% endif %}\n\n {{ return(schema_changes_dict['source_columns']) }}\n\n {% endif %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.check_for_schema_changes", "macro.dbt.sync_column_schemas"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955522, "supported_languages": null}, "macro.dbt.can_clone_table": {"name": "can_clone_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/clone/can_clone_table.sql", "original_file_path": "macros/materializations/models/clone/can_clone_table.sql", "unique_id": "macro.dbt.can_clone_table", "macro_sql": "{% macro can_clone_table() %}\n {{ return(adapter.dispatch('can_clone_table', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__can_clone_table"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955536, "supported_languages": null}, "macro.dbt.default__can_clone_table": {"name": "default__can_clone_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/clone/can_clone_table.sql", "original_file_path": "macros/materializations/models/clone/can_clone_table.sql", "unique_id": "macro.dbt.default__can_clone_table", "macro_sql": "{% macro default__can_clone_table() %}\n {{ return(False) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955542, "supported_languages": null}, "macro.dbt.create_or_replace_clone": {"name": "create_or_replace_clone", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/clone/create_or_replace_clone.sql", "original_file_path": "macros/materializations/models/clone/create_or_replace_clone.sql", "unique_id": "macro.dbt.create_or_replace_clone", "macro_sql": "{% macro create_or_replace_clone(this_relation, defer_relation) %}\n {{ return(adapter.dispatch('create_or_replace_clone', 'dbt')(this_relation, defer_relation)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__create_or_replace_clone"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955552, "supported_languages": null}, "macro.dbt.default__create_or_replace_clone": {"name": "default__create_or_replace_clone", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/clone/create_or_replace_clone.sql", "original_file_path": "macros/materializations/models/clone/create_or_replace_clone.sql", "unique_id": "macro.dbt.default__create_or_replace_clone", "macro_sql": "{% macro default__create_or_replace_clone(this_relation, defer_relation) %}\n create or replace table {{ this_relation }} clone {{ defer_relation }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95556, "supported_languages": null}, "macro.dbt.materialization_clone_default": {"name": "materialization_clone_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/clone/clone.sql", "original_file_path": "macros/materializations/models/clone/clone.sql", "unique_id": "macro.dbt.materialization_clone_default", "macro_sql": "{%- materialization clone, default -%}\n\n {%- set relations = {'relations': []} -%}\n\n {%- if not defer_relation -%}\n -- nothing to do\n {{ log(\"No relation found in state manifest for \" ~ model.unique_id, info=True) }}\n {{ return(relations) }}\n {%- endif -%}\n\n {%- set existing_relation = load_cached_relation(this) -%}\n\n {%- if existing_relation and not flags.FULL_REFRESH -%}\n -- noop!\n {{ log(\"Relation \" ~ existing_relation ~ \" already exists\", info=True) }}\n {{ return(relations) }}\n {%- endif -%}\n\n {%- set other_existing_relation = load_cached_relation(defer_relation) -%}\n\n -- If this is a database that can do zero-copy cloning of tables, and the other relation is a table, then this will be a table\n -- Otherwise, this will be a view\n\n {% set can_clone_table = can_clone_table() %}\n\n {%- if other_existing_relation and other_existing_relation.type == 'table' and can_clone_table -%}\n\n {%- set target_relation = this.incorporate(type='table') -%}\n {% if existing_relation is not none and not existing_relation.is_table %}\n {{ log(\"Dropping relation \" ~ existing_relation ~ \" because it is of type \" ~ existing_relation.type) }}\n {{ drop_relation_if_exists(existing_relation) }}\n {% endif %}\n\n -- as a general rule, data platforms that can clone tables can also do atomic 'create or replace'\n {% call statement('main') %}\n {% if target_relation and defer_relation and target_relation == defer_relation %}\n {{ log(\"Target relation and defer relation are the same, skipping clone for relation: \" ~ target_relation) }}\n {% else %}\n {{ create_or_replace_clone(target_relation, defer_relation) }}\n {% endif %}\n\n {% endcall %}\n\n {% set should_revoke = should_revoke(existing_relation, full_refresh_mode=True) %}\n {% do apply_grants(target_relation, grant_config, should_revoke=should_revoke) %}\n {% do persist_docs(target_relation, model) %}\n\n {{ return({'relations': [target_relation]}) }}\n\n {%- else -%}\n\n {%- set target_relation = this.incorporate(type='view') -%}\n\n -- reuse the view materialization\n -- TODO: support actual dispatch for materialization macros\n -- Tracking ticket: https://github.com/dbt-labs/dbt-core/issues/7799\n {% set search_name = \"materialization_view_\" ~ adapter.type() %}\n {% if not search_name in context %}\n {% set search_name = \"materialization_view_default\" %}\n {% endif %}\n {% set materialization_macro = context[search_name] %}\n {% set relations = materialization_macro() %}\n {{ return(relations) }}\n\n {%- endif -%}\n\n{%- endmaterialization -%}", "depends_on": {"macros": ["macro.dbt.load_cached_relation", "macro.dbt.can_clone_table", "macro.dbt.drop_relation_if_exists", "macro.dbt.statement", "macro.dbt.create_or_replace_clone", "macro.dbt.should_revoke", "macro.dbt.apply_grants", "macro.dbt.persist_docs"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9555771, "supported_languages": ["sql"]}, "macro.dbt.materialization_seed_default": {"name": "materialization_seed_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/seed.sql", "original_file_path": "macros/materializations/seeds/seed.sql", "unique_id": "macro.dbt.materialization_seed_default", "macro_sql": "{% materialization seed, default %}\n\n {%- set identifier = model['alias'] -%}\n {%- set full_refresh_mode = (should_full_refresh()) -%}\n\n {%- set old_relation = adapter.get_relation(database=database, schema=schema, identifier=identifier) -%}\n\n {%- set exists_as_table = (old_relation is not none and old_relation.is_table) -%}\n {%- set exists_as_view = (old_relation is not none and old_relation.is_view) -%}\n\n {%- set grant_config = config.get('grants') -%}\n {%- set agate_table = load_agate_table() -%}\n -- grab current tables grants config for comparison later on\n\n {%- do store_result('agate_table', response='OK', agate_table=agate_table) -%}\n\n {{ run_hooks(pre_hooks, inside_transaction=False) }}\n\n -- `BEGIN` happens here:\n {{ run_hooks(pre_hooks, inside_transaction=True) }}\n\n -- build model\n {% set create_table_sql = \"\" %}\n {% if exists_as_view %}\n {{ exceptions.raise_compiler_error(\"Cannot seed to '{}', it is a view\".format(old_relation)) }}\n {% elif exists_as_table %}\n {% set create_table_sql = reset_csv_table(model, full_refresh_mode, old_relation, agate_table) %}\n {% else %}\n {% set create_table_sql = create_csv_table(model, agate_table) %}\n {% endif %}\n\n {% set code = 'CREATE' if full_refresh_mode else 'INSERT' %}\n {% set rows_affected = (agate_table.rows | length) %}\n {% set sql = load_csv_rows(model, agate_table) %}\n\n {% call noop_statement('main', code ~ ' ' ~ rows_affected, code, rows_affected) %}\n {{ get_csv_sql(create_table_sql, sql) }};\n {% endcall %}\n\n {% set target_relation = this.incorporate(type='table') %}\n\n {% set should_revoke = should_revoke(old_relation, full_refresh_mode) %}\n {% do apply_grants(target_relation, grant_config, should_revoke=should_revoke) %}\n\n {% do persist_docs(target_relation, model) %}\n\n {% if full_refresh_mode or not exists_as_table %}\n {% do create_indexes(target_relation) %}\n {% endif %}\n\n {{ run_hooks(post_hooks, inside_transaction=True) }}\n\n -- `COMMIT` happens here\n {{ adapter.commit() }}\n\n {{ run_hooks(post_hooks, inside_transaction=False) }}\n\n {{ return({'relations': [target_relation]}) }}\n\n{% endmaterialization %}", "depends_on": {"macros": ["macro.dbt.should_full_refresh", "macro.dbt.run_hooks", "macro.dbt.reset_csv_table", "macro.dbt.create_csv_table", "macro.dbt.load_csv_rows", "macro.dbt.noop_statement", "macro.dbt.get_csv_sql", "macro.dbt.should_revoke", "macro.dbt.apply_grants", "macro.dbt.persist_docs", "macro.dbt.create_indexes"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9555998, "supported_languages": ["sql"]}, "macro.dbt.create_csv_table": {"name": "create_csv_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.create_csv_table", "macro_sql": "{% macro create_csv_table(model, agate_table) -%}\n {{ adapter.dispatch('create_csv_table', 'dbt')(model, agate_table) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__create_csv_table"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955629, "supported_languages": null}, "macro.dbt.default__create_csv_table": {"name": "default__create_csv_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.default__create_csv_table", "macro_sql": "{% macro default__create_csv_table(model, agate_table) %}\n {%- set column_override = model['config'].get('column_types', {}) -%}\n {%- set quote_seed_column = model['config'].get('quote_columns', None) -%}\n\n {% set sql %}\n create table {{ this.render() }} (\n {%- for col_name in agate_table.column_names -%}\n {%- set inferred_type = adapter.convert_type(agate_table, loop.index0) -%}\n {%- set type = column_override.get(col_name, inferred_type) -%}\n {%- set column_name = (col_name | string) -%}\n {{ adapter.quote_seed_column(column_name, quote_seed_column) }} {{ type }} {%- if not loop.last -%}, {%- endif -%}\n {%- endfor -%}\n )\n {% endset %}\n\n {% call statement('_') -%}\n {{ sql }}\n {%- endcall %}\n\n {{ return(sql) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955638, "supported_languages": null}, "macro.dbt.reset_csv_table": {"name": "reset_csv_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.reset_csv_table", "macro_sql": "{% macro reset_csv_table(model, full_refresh, old_relation, agate_table) -%}\n {{ adapter.dispatch('reset_csv_table', 'dbt')(model, full_refresh, old_relation, agate_table) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__reset_csv_table"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9556448, "supported_languages": null}, "macro.dbt.default__reset_csv_table": {"name": "default__reset_csv_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.default__reset_csv_table", "macro_sql": "{% macro default__reset_csv_table(model, full_refresh, old_relation, agate_table) %}\n {% set sql = \"\" %}\n {% if full_refresh %}\n {{ adapter.drop_relation(old_relation) }}\n {% set sql = create_csv_table(model, agate_table) %}\n {% else %}\n {{ adapter.truncate_relation(old_relation) }}\n {% set sql = \"truncate table \" ~ old_relation %}\n {% endif %}\n\n {{ return(sql) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.create_csv_table"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9556532, "supported_languages": null}, "macro.dbt.get_csv_sql": {"name": "get_csv_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.get_csv_sql", "macro_sql": "{% macro get_csv_sql(create_or_truncate_sql, insert_sql) %}\n {{ adapter.dispatch('get_csv_sql', 'dbt')(create_or_truncate_sql, insert_sql) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_csv_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955659, "supported_languages": null}, "macro.dbt.default__get_csv_sql": {"name": "default__get_csv_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.default__get_csv_sql", "macro_sql": "{% macro default__get_csv_sql(create_or_truncate_sql, insert_sql) %}\n {{ create_or_truncate_sql }};\n -- dbt seed --\n {{ insert_sql }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955667, "supported_languages": null}, "macro.dbt.get_binding_char": {"name": "get_binding_char", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.get_binding_char", "macro_sql": "{% macro get_binding_char() -%}\n {{ adapter.dispatch('get_binding_char', 'dbt')() }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_binding_char"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955682, "supported_languages": null}, "macro.dbt.default__get_binding_char": {"name": "default__get_binding_char", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.default__get_binding_char", "macro_sql": "{% macro default__get_binding_char() %}\n {{ return('%s') }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955688, "supported_languages": null}, "macro.dbt.get_batch_size": {"name": "get_batch_size", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.get_batch_size", "macro_sql": "{% macro get_batch_size() -%}\n {{ return(adapter.dispatch('get_batch_size', 'dbt')()) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_batch_size"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955693, "supported_languages": null}, "macro.dbt.default__get_batch_size": {"name": "default__get_batch_size", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.default__get_batch_size", "macro_sql": "{% macro default__get_batch_size() %}\n {{ return(10000) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9557, "supported_languages": null}, "macro.dbt.get_seed_column_quoted_csv": {"name": "get_seed_column_quoted_csv", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.get_seed_column_quoted_csv", "macro_sql": "{% macro get_seed_column_quoted_csv(model, column_names) %}\n {%- set quote_seed_column = model['config'].get('quote_columns', None) -%}\n {% set quoted = [] %}\n {% for col in column_names -%}\n {%- do quoted.append(adapter.quote_seed_column(col, quote_seed_column)) -%}\n {%- endfor %}\n\n {%- set dest_cols_csv = quoted | join(', ') -%}\n {{ return(dest_cols_csv) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9557052, "supported_languages": null}, "macro.dbt.load_csv_rows": {"name": "load_csv_rows", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.load_csv_rows", "macro_sql": "{% macro load_csv_rows(model, agate_table) -%}\n {{ adapter.dispatch('load_csv_rows', 'dbt')(model, agate_table) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__load_csv_rows"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955714, "supported_languages": null}, "macro.dbt.default__load_csv_rows": {"name": "default__load_csv_rows", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.default__load_csv_rows", "macro_sql": "{% macro default__load_csv_rows(model, agate_table) %}\n\n {% set batch_size = get_batch_size() %}\n\n {% set cols_sql = get_seed_column_quoted_csv(model, agate_table.column_names) %}\n {% set bindings = [] %}\n\n {% set statements = [] %}\n\n {% for chunk in agate_table.rows | batch(batch_size) %}\n {% set bindings = [] %}\n\n {% for row in chunk %}\n {% do bindings.extend(row) %}\n {% endfor %}\n\n {% set sql %}\n insert into {{ this.render() }} ({{ cols_sql }}) values\n {% for row in chunk -%}\n ({%- for column in agate_table.column_names -%}\n {{ get_binding_char() }}\n {%- if not loop.last%},{%- endif %}\n {%- endfor -%})\n {%- if not loop.last%},{%- endif %}\n {%- endfor %}\n {% endset %}\n\n {% do adapter.add_query(sql, bindings=bindings, abridge_sql_log=True) %}\n\n {% if loop.index0 == 0 %}\n {% do statements.append(sql) %}\n {% endif %}\n {% endfor %}\n\n {# Return SQL so we can render it out into the compiled files #}\n {{ return(statements[0]) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_batch_size", "macro.dbt.get_seed_column_quoted_csv", "macro.dbt.get_binding_char"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955719, "supported_languages": null}, "macro.dbt.generate_alias_name": {"name": "generate_alias_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/get_custom_name/get_custom_alias.sql", "original_file_path": "macros/get_custom_name/get_custom_alias.sql", "unique_id": "macro.dbt.generate_alias_name", "macro_sql": "{% macro generate_alias_name(custom_alias_name=none, node=none) -%}\n {% do return(adapter.dispatch('generate_alias_name', 'dbt')(custom_alias_name, node)) %}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__generate_alias_name"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955734, "supported_languages": null}, "macro.dbt.default__generate_alias_name": {"name": "default__generate_alias_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/get_custom_name/get_custom_alias.sql", "original_file_path": "macros/get_custom_name/get_custom_alias.sql", "unique_id": "macro.dbt.default__generate_alias_name", "macro_sql": "{% macro default__generate_alias_name(custom_alias_name=none, node=none) -%}\n\n {%- if custom_alias_name -%}\n\n {{ custom_alias_name | trim }}\n\n {%- elif node.version -%}\n\n {{ return(node.name ~ \"_v\" ~ (node.version | replace(\".\", \"_\"))) }}\n\n {%- else -%}\n\n {{ node.name }}\n\n {%- endif -%}\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955761, "supported_languages": null}, "macro.dbt.generate_schema_name": {"name": "generate_schema_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/get_custom_name/get_custom_schema.sql", "original_file_path": "macros/get_custom_name/get_custom_schema.sql", "unique_id": "macro.dbt.generate_schema_name", "macro_sql": "{% macro generate_schema_name(custom_schema_name=none, node=none) -%}\n {{ return(adapter.dispatch('generate_schema_name', 'dbt')(custom_schema_name, node)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__generate_schema_name"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955776, "supported_languages": null}, "macro.dbt.default__generate_schema_name": {"name": "default__generate_schema_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/get_custom_name/get_custom_schema.sql", "original_file_path": "macros/get_custom_name/get_custom_schema.sql", "unique_id": "macro.dbt.default__generate_schema_name", "macro_sql": "{% macro default__generate_schema_name(custom_schema_name, node) -%}\n\n {%- set default_schema = target.schema -%}\n {%- if custom_schema_name is none -%}\n\n {{ default_schema }}\n\n {%- else -%}\n\n {{ default_schema }}_{{ custom_schema_name | trim }}\n\n {%- endif -%}\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955782, "supported_languages": null}, "macro.dbt.generate_schema_name_for_env": {"name": "generate_schema_name_for_env", "resource_type": "macro", "package_name": "dbt", "path": "macros/get_custom_name/get_custom_schema.sql", "original_file_path": "macros/get_custom_name/get_custom_schema.sql", "unique_id": "macro.dbt.generate_schema_name_for_env", "macro_sql": "{% macro generate_schema_name_for_env(custom_schema_name, node) -%}\n\n {%- set default_schema = target.schema -%}\n {%- if target.name == 'prod' and custom_schema_name is not none -%}\n\n {{ custom_schema_name | trim }}\n\n {%- else -%}\n\n {{ default_schema }}\n\n {%- endif -%}\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955789, "supported_languages": null}, "macro.dbt.generate_database_name": {"name": "generate_database_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/get_custom_name/get_custom_database.sql", "original_file_path": "macros/get_custom_name/get_custom_database.sql", "unique_id": "macro.dbt.generate_database_name", "macro_sql": "{% macro generate_database_name(custom_database_name=none, node=none) -%}\n {% do return(adapter.dispatch('generate_database_name', 'dbt')(custom_database_name, node)) %}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__generate_database_name"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955803, "supported_languages": null}, "macro.dbt.default__generate_database_name": {"name": "default__generate_database_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/get_custom_name/get_custom_database.sql", "original_file_path": "macros/get_custom_name/get_custom_database.sql", "unique_id": "macro.dbt.default__generate_database_name", "macro_sql": "{% macro default__generate_database_name(custom_database_name=none, node=none) -%}\n {%- set default_database = target.database -%}\n {%- if custom_database_name is none -%}\n\n {{ default_database }}\n\n {%- else -%}\n\n {{ custom_database_name }}\n\n {%- endif -%}\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955809, "supported_languages": null}, "macro.dbt.get_drop_sql": {"name": "get_drop_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/drop.sql", "original_file_path": "macros/relations/drop.sql", "unique_id": "macro.dbt.get_drop_sql", "macro_sql": "{%- macro get_drop_sql(relation) -%}\n {{- log('Applying DROP to: ' ~ relation) -}}\n {{- adapter.dispatch('get_drop_sql', 'dbt')(relation) -}}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.default__get_drop_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95582, "supported_languages": null}, "macro.dbt.default__get_drop_sql": {"name": "default__get_drop_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/drop.sql", "original_file_path": "macros/relations/drop.sql", "unique_id": "macro.dbt.default__get_drop_sql", "macro_sql": "{%- macro default__get_drop_sql(relation) -%}\n\n {%- if relation.is_view -%}\n {{ drop_view(relation) }}\n\n {%- elif relation.is_table -%}\n {{ drop_table(relation) }}\n\n {%- elif relation.is_materialized_view -%}\n {{ drop_materialized_view(relation) }}\n\n {%- else -%}\n drop {{ relation.type }} if exists {{ relation }} cascade\n\n {%- endif -%}\n\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.drop_view", "macro.dbt.drop_table", "macro.dbt.drop_materialized_view"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955827, "supported_languages": null}, "macro.dbt.drop_relation": {"name": "drop_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/drop.sql", "original_file_path": "macros/relations/drop.sql", "unique_id": "macro.dbt.drop_relation", "macro_sql": "{% macro drop_relation(relation) -%}\n {{ return(adapter.dispatch('drop_relation', 'dbt')(relation)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__drop_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955837, "supported_languages": null}, "macro.dbt.default__drop_relation": {"name": "default__drop_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/drop.sql", "original_file_path": "macros/relations/drop.sql", "unique_id": "macro.dbt.default__drop_relation", "macro_sql": "{% macro default__drop_relation(relation) -%}\n {% call statement('drop_relation', auto_begin=False) -%}\n {{ get_drop_sql(relation) }}\n {%- endcall %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement", "macro.dbt.get_drop_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955842, "supported_languages": null}, "macro.dbt.drop_relation_if_exists": {"name": "drop_relation_if_exists", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/drop.sql", "original_file_path": "macros/relations/drop.sql", "unique_id": "macro.dbt.drop_relation_if_exists", "macro_sql": "{% macro drop_relation_if_exists(relation) %}\n {% if relation is not none %}\n {{ adapter.drop_relation(relation) }}\n {% endif %}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955847, "supported_languages": null}, "macro.dbt.get_replace_sql": {"name": "get_replace_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/replace.sql", "original_file_path": "macros/relations/replace.sql", "unique_id": "macro.dbt.get_replace_sql", "macro_sql": "{% macro get_replace_sql(existing_relation, target_relation, sql) %}\n {{- log('Applying REPLACE to: ' ~ existing_relation) -}}\n {{- adapter.dispatch('get_replace_sql', 'dbt')(existing_relation, target_relation, sql) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_replace_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955864, "supported_languages": null}, "macro.dbt.default__get_replace_sql": {"name": "default__get_replace_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/replace.sql", "original_file_path": "macros/relations/replace.sql", "unique_id": "macro.dbt.default__get_replace_sql", "macro_sql": "{% macro default__get_replace_sql(existing_relation, target_relation, sql) %}\n\n {# /* use a create or replace statement if possible */ #}\n\n {% set is_replaceable = existing_relation.type == target_relation_type and existing_relation.can_be_replaced %}\n\n {% if is_replaceable and existing_relation.is_view %}\n {{ get_replace_view_sql(target_relation, sql) }}\n\n {% elif is_replaceable and existing_relation.is_table %}\n {{ get_replace_table_sql(target_relation, sql) }}\n\n {% elif is_replaceable and existing_relation.is_materialized_view %}\n {{ get_replace_materialized_view_sql(target_relation, sql) }}\n\n {# /* a create or replace statement is not possible, so try to stage and/or backup to be safe */ #}\n\n {# /* create target_relation as an intermediate relation, then swap it out with the existing one using a backup */ #}\n {%- elif target_relation.can_be_renamed and existing_relation.can_be_renamed -%}\n {{ get_create_intermediate_sql(target_relation, sql) }};\n {{ get_create_backup_sql(existing_relation) }};\n {{ get_rename_intermediate_sql(target_relation) }};\n {{ get_drop_backup_sql(existing_relation) }}\n\n {# /* create target_relation as an intermediate relation, then swap it out with the existing one without using a backup */ #}\n {%- elif target_relation.can_be_renamed -%}\n {{ get_create_intermediate_sql(target_relation, sql) }};\n {{ get_drop_sql(existing_relation) }};\n {{ get_rename_intermediate_sql(target_relation) }}\n\n {# /* create target_relation in place by first backing up the existing relation */ #}\n {%- elif existing_relation.can_be_renamed -%}\n {{ get_create_backup_sql(existing_relation) }};\n {{ get_create_sql(target_relation, sql) }};\n {{ get_drop_backup_sql(existing_relation) }}\n\n {# /* no renaming is allowed, so just drop and create */ #}\n {%- else -%}\n {{ get_drop_sql(existing_relation) }};\n {{ get_create_sql(target_relation, sql) }}\n\n {%- endif -%}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_replace_view_sql", "macro.dbt.get_replace_table_sql", "macro.dbt.get_replace_materialized_view_sql", "macro.dbt.get_create_intermediate_sql", "macro.dbt.get_create_backup_sql", "macro.dbt.get_rename_intermediate_sql", "macro.dbt.get_drop_backup_sql", "macro.dbt.get_drop_sql", "macro.dbt.get_create_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955874, "supported_languages": null}, "macro.dbt.get_create_intermediate_sql": {"name": "get_create_intermediate_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/create_intermediate.sql", "original_file_path": "macros/relations/create_intermediate.sql", "unique_id": "macro.dbt.get_create_intermediate_sql", "macro_sql": "{%- macro get_create_intermediate_sql(relation, sql) -%}\n {{- log('Applying CREATE INTERMEDIATE to: ' ~ relation) -}}\n {{- adapter.dispatch('get_create_intermediate_sql', 'dbt')(relation, sql) -}}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.default__get_create_intermediate_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955892, "supported_languages": null}, "macro.dbt.default__get_create_intermediate_sql": {"name": "default__get_create_intermediate_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/create_intermediate.sql", "original_file_path": "macros/relations/create_intermediate.sql", "unique_id": "macro.dbt.default__get_create_intermediate_sql", "macro_sql": "{%- macro default__get_create_intermediate_sql(relation, sql) -%}\n\n -- get the standard intermediate name\n {% set intermediate_relation = make_intermediate_relation(relation) %}\n\n -- drop any pre-existing intermediate\n {{ get_drop_sql(intermediate_relation) }};\n\n {{ get_create_sql(intermediate_relation, sql) }}\n\n{%- endmacro -%}", "depends_on": {"macros": ["macro.dbt.make_intermediate_relation", "macro.dbt.get_drop_sql", "macro.dbt.get_create_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955897, "supported_languages": null}, "macro.dbt.drop_schema_named": {"name": "drop_schema_named", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/schema.sql", "original_file_path": "macros/relations/schema.sql", "unique_id": "macro.dbt.drop_schema_named", "macro_sql": "{% macro drop_schema_named(schema_name) %}\n {{ return(adapter.dispatch('drop_schema_named', 'dbt') (schema_name)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__drop_schema_named"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955909, "supported_languages": null}, "macro.dbt.default__drop_schema_named": {"name": "default__drop_schema_named", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/schema.sql", "original_file_path": "macros/relations/schema.sql", "unique_id": "macro.dbt.default__drop_schema_named", "macro_sql": "{% macro default__drop_schema_named(schema_name) %}\n {% set schema_relation = api.Relation.create(schema=schema_name) %}\n {{ adapter.drop_schema(schema_relation) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9559171, "supported_languages": null}, "macro.dbt.get_drop_backup_sql": {"name": "get_drop_backup_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/drop_backup.sql", "original_file_path": "macros/relations/drop_backup.sql", "unique_id": "macro.dbt.get_drop_backup_sql", "macro_sql": "{%- macro get_drop_backup_sql(relation) -%}\n {{- log('Applying DROP BACKUP to: ' ~ relation) -}}\n {{- adapter.dispatch('get_drop_backup_sql', 'dbt')(relation) -}}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.default__get_drop_backup_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95593, "supported_languages": null}, "macro.dbt.default__get_drop_backup_sql": {"name": "default__get_drop_backup_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/drop_backup.sql", "original_file_path": "macros/relations/drop_backup.sql", "unique_id": "macro.dbt.default__get_drop_backup_sql", "macro_sql": "{%- macro default__get_drop_backup_sql(relation) -%}\n\n -- get the standard backup name\n {% set backup_relation = make_backup_relation(relation, relation.type) %}\n\n {{ get_drop_sql(backup_relation) }}\n\n{%- endmacro -%}", "depends_on": {"macros": ["macro.dbt.make_backup_relation", "macro.dbt.get_drop_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955936, "supported_languages": null}, "macro.dbt.get_rename_sql": {"name": "get_rename_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/rename.sql", "original_file_path": "macros/relations/rename.sql", "unique_id": "macro.dbt.get_rename_sql", "macro_sql": "{%- macro get_rename_sql(relation, new_name) -%}\n {{- log('Applying RENAME to: ' ~ relation) -}}\n {{- adapter.dispatch('get_rename_sql', 'dbt')(relation, new_name) -}}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.default__get_rename_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955948, "supported_languages": null}, "macro.dbt.default__get_rename_sql": {"name": "default__get_rename_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/rename.sql", "original_file_path": "macros/relations/rename.sql", "unique_id": "macro.dbt.default__get_rename_sql", "macro_sql": "{%- macro default__get_rename_sql(relation, new_name) -%}\n\n {%- if relation.is_view -%}\n {{ get_rename_view_sql(relation, new_name) }}\n\n {%- elif relation.is_table -%}\n {{ get_rename_table_sql(relation, new_name) }}\n\n {%- elif relation.is_materialized_view -%}\n {{ get_rename_materialized_view_sql(relation, new_name) }}\n\n {%- else -%}\n {{- exceptions.raise_compiler_error(\"`get_rename_sql` has not been implemented for: \" ~ relation.type ) -}}\n\n {%- endif -%}\n\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.get_rename_view_sql", "macro.dbt.get_rename_table_sql", "macro.dbt.get_rename_materialized_view_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955955, "supported_languages": null}, "macro.dbt.rename_relation": {"name": "rename_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/rename.sql", "original_file_path": "macros/relations/rename.sql", "unique_id": "macro.dbt.rename_relation", "macro_sql": "{% macro rename_relation(from_relation, to_relation) -%}\n {{ return(adapter.dispatch('rename_relation', 'dbt')(from_relation, to_relation)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__rename_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955966, "supported_languages": null}, "macro.dbt.default__rename_relation": {"name": "default__rename_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/rename.sql", "original_file_path": "macros/relations/rename.sql", "unique_id": "macro.dbt.default__rename_relation", "macro_sql": "{% macro default__rename_relation(from_relation, to_relation) -%}\n {% set target_name = adapter.quote_as_configured(to_relation.identifier, 'identifier') %}\n {% call statement('rename_relation') -%}\n alter table {{ from_relation }} rename to {{ target_name }}\n {%- endcall %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955971, "supported_languages": null}, "macro.dbt.get_create_backup_sql": {"name": "get_create_backup_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/create_backup.sql", "original_file_path": "macros/relations/create_backup.sql", "unique_id": "macro.dbt.get_create_backup_sql", "macro_sql": "{%- macro get_create_backup_sql(relation) -%}\n {{- log('Applying CREATE BACKUP to: ' ~ relation) -}}\n {{- adapter.dispatch('get_create_backup_sql', 'dbt')(relation) -}}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.default__get_create_backup_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.955989, "supported_languages": null}, "macro.dbt.default__get_create_backup_sql": {"name": "default__get_create_backup_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/create_backup.sql", "original_file_path": "macros/relations/create_backup.sql", "unique_id": "macro.dbt.default__get_create_backup_sql", "macro_sql": "{%- macro default__get_create_backup_sql(relation) -%}\n\n -- get the standard backup name\n {% set backup_relation = make_backup_relation(relation, relation.type) %}\n\n -- drop any pre-existing backup\n {{ get_drop_sql(backup_relation) }};\n\n {{ get_rename_sql(relation, backup_relation.identifier) }}\n\n{%- endmacro -%}", "depends_on": {"macros": ["macro.dbt.make_backup_relation", "macro.dbt.get_drop_sql", "macro.dbt.get_rename_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9559941, "supported_languages": null}, "macro.dbt.get_create_sql": {"name": "get_create_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/create.sql", "original_file_path": "macros/relations/create.sql", "unique_id": "macro.dbt.get_create_sql", "macro_sql": "{%- macro get_create_sql(relation, sql) -%}\n {{- log('Applying CREATE to: ' ~ relation) -}}\n {{- adapter.dispatch('get_create_sql', 'dbt')(relation, sql) -}}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.default__get_create_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956007, "supported_languages": null}, "macro.dbt.default__get_create_sql": {"name": "default__get_create_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/create.sql", "original_file_path": "macros/relations/create.sql", "unique_id": "macro.dbt.default__get_create_sql", "macro_sql": "{%- macro default__get_create_sql(relation, sql) -%}\n\n {%- if relation.is_view -%}\n {{ get_create_view_as_sql(relation, sql) }}\n\n {%- elif relation.is_table -%}\n {{ get_create_table_as_sql(False, relation, sql) }}\n\n {%- elif relation.is_materialized_view -%}\n {{ get_create_materialized_view_as_sql(relation, sql) }}\n\n {%- else -%}\n {{- exceptions.raise_compiler_error(\"`get_create_sql` has not been implemented for: \" ~ relation.type ) -}}\n\n {%- endif -%}\n\n{%- endmacro -%}", "depends_on": {"macros": ["macro.dbt.get_create_view_as_sql", "macro.dbt.get_create_table_as_sql", "macro.dbt.get_create_materialized_view_as_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956013, "supported_languages": null}, "macro.dbt.get_rename_intermediate_sql": {"name": "get_rename_intermediate_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/rename_intermediate.sql", "original_file_path": "macros/relations/rename_intermediate.sql", "unique_id": "macro.dbt.get_rename_intermediate_sql", "macro_sql": "{%- macro get_rename_intermediate_sql(relation) -%}\n {{- log('Applying RENAME INTERMEDIATE to: ' ~ relation) -}}\n {{- adapter.dispatch('get_rename_intermediate_sql', 'dbt')(relation) -}}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.default__get_rename_intermediate_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956028, "supported_languages": null}, "macro.dbt.default__get_rename_intermediate_sql": {"name": "default__get_rename_intermediate_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/rename_intermediate.sql", "original_file_path": "macros/relations/rename_intermediate.sql", "unique_id": "macro.dbt.default__get_rename_intermediate_sql", "macro_sql": "{%- macro default__get_rename_intermediate_sql(relation) -%}\n\n -- get the standard intermediate name\n {% set intermediate_relation = make_intermediate_relation(relation) %}\n\n {{ get_rename_sql(intermediate_relation, relation.identifier) }}\n\n{%- endmacro -%}", "depends_on": {"macros": ["macro.dbt.make_intermediate_relation", "macro.dbt.get_rename_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956036, "supported_languages": null}, "macro.dbt.drop_materialized_view": {"name": "drop_materialized_view", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/drop.sql", "original_file_path": "macros/relations/materialized_view/drop.sql", "unique_id": "macro.dbt.drop_materialized_view", "macro_sql": "{% macro drop_materialized_view(relation) -%}\n {{- adapter.dispatch('drop_materialized_view', 'dbt')(relation) -}}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__drop_materialized_view"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956047, "supported_languages": null}, "macro.dbt.default__drop_materialized_view": {"name": "default__drop_materialized_view", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/drop.sql", "original_file_path": "macros/relations/materialized_view/drop.sql", "unique_id": "macro.dbt.default__drop_materialized_view", "macro_sql": "{% macro default__drop_materialized_view(relation) -%}\n drop materialized view if exists {{ relation }} cascade\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956052, "supported_languages": null}, "macro.dbt.get_replace_materialized_view_sql": {"name": "get_replace_materialized_view_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/replace.sql", "original_file_path": "macros/relations/materialized_view/replace.sql", "unique_id": "macro.dbt.get_replace_materialized_view_sql", "macro_sql": "{% macro get_replace_materialized_view_sql(relation, sql) %}\n {{- adapter.dispatch('get_replace_materialized_view_sql', 'dbt')(relation, sql) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_replace_materialized_view_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9560661, "supported_languages": null}, "macro.dbt.default__get_replace_materialized_view_sql": {"name": "default__get_replace_materialized_view_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/replace.sql", "original_file_path": "macros/relations/materialized_view/replace.sql", "unique_id": "macro.dbt.default__get_replace_materialized_view_sql", "macro_sql": "{% macro default__get_replace_materialized_view_sql(relation, sql) %}\n {{ exceptions.raise_compiler_error(\n \"`get_replace_materialized_view_sql` has not been implemented for this adapter.\"\n ) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956071, "supported_languages": null}, "macro.dbt.refresh_materialized_view": {"name": "refresh_materialized_view", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/refresh.sql", "original_file_path": "macros/relations/materialized_view/refresh.sql", "unique_id": "macro.dbt.refresh_materialized_view", "macro_sql": "{% macro refresh_materialized_view(relation) %}\n {{- log('Applying REFRESH to: ' ~ relation) -}}\n {{- adapter.dispatch('refresh_materialized_view', 'dbt')(relation) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__refresh_materialized_view"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956081, "supported_languages": null}, "macro.dbt.default__refresh_materialized_view": {"name": "default__refresh_materialized_view", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/refresh.sql", "original_file_path": "macros/relations/materialized_view/refresh.sql", "unique_id": "macro.dbt.default__refresh_materialized_view", "macro_sql": "{% macro default__refresh_materialized_view(relation) %}\n {{ exceptions.raise_compiler_error(\"`refresh_materialized_view` has not been implemented for this adapter.\") }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956087, "supported_languages": null}, "macro.dbt.get_rename_materialized_view_sql": {"name": "get_rename_materialized_view_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/rename.sql", "original_file_path": "macros/relations/materialized_view/rename.sql", "unique_id": "macro.dbt.get_rename_materialized_view_sql", "macro_sql": "{% macro get_rename_materialized_view_sql(relation, new_name) %}\n {{- adapter.dispatch('get_rename_materialized_view_sql', 'dbt')(relation, new_name) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_rename_materialized_view_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9560971, "supported_languages": null}, "macro.dbt.default__get_rename_materialized_view_sql": {"name": "default__get_rename_materialized_view_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/rename.sql", "original_file_path": "macros/relations/materialized_view/rename.sql", "unique_id": "macro.dbt.default__get_rename_materialized_view_sql", "macro_sql": "{% macro default__get_rename_materialized_view_sql(relation, new_name) %}\n {{ exceptions.raise_compiler_error(\n \"`get_rename_materialized_view_sql` has not been implemented for this adapter.\"\n ) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9561028, "supported_languages": null}, "macro.dbt.get_alter_materialized_view_as_sql": {"name": "get_alter_materialized_view_as_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/alter.sql", "original_file_path": "macros/relations/materialized_view/alter.sql", "unique_id": "macro.dbt.get_alter_materialized_view_as_sql", "macro_sql": "{% macro get_alter_materialized_view_as_sql(\n relation,\n configuration_changes,\n sql,\n existing_relation,\n backup_relation,\n intermediate_relation\n) %}\n {{- log('Applying ALTER to: ' ~ relation) -}}\n {{- adapter.dispatch('get_alter_materialized_view_as_sql', 'dbt')(\n relation,\n configuration_changes,\n sql,\n existing_relation,\n backup_relation,\n intermediate_relation\n ) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_alter_materialized_view_as_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956117, "supported_languages": null}, "macro.dbt.default__get_alter_materialized_view_as_sql": {"name": "default__get_alter_materialized_view_as_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/alter.sql", "original_file_path": "macros/relations/materialized_view/alter.sql", "unique_id": "macro.dbt.default__get_alter_materialized_view_as_sql", "macro_sql": "{% macro default__get_alter_materialized_view_as_sql(\n relation,\n configuration_changes,\n sql,\n existing_relation,\n backup_relation,\n intermediate_relation\n) %}\n {{ exceptions.raise_compiler_error(\"Materialized views have not been implemented for this adapter.\") }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9561229, "supported_languages": null}, "macro.dbt.get_materialized_view_configuration_changes": {"name": "get_materialized_view_configuration_changes", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/alter.sql", "original_file_path": "macros/relations/materialized_view/alter.sql", "unique_id": "macro.dbt.get_materialized_view_configuration_changes", "macro_sql": "{% macro get_materialized_view_configuration_changes(existing_relation, new_config) %}\n /* {#\n It's recommended that configuration changes be formatted as follows:\n {\"\": [{\"action\": \"\", \"context\": ...}]}\n\n For example:\n {\n \"indexes\": [\n {\"action\": \"drop\", \"context\": \"index_abc\"},\n {\"action\": \"create\", \"context\": {\"columns\": [\"column_1\", \"column_2\"], \"type\": \"hash\", \"unique\": True}},\n ],\n }\n\n Either way, `get_materialized_view_configuration_changes` needs to align with `get_alter_materialized_view_as_sql`.\n #} */\n {{- log('Determining configuration changes on: ' ~ existing_relation) -}}\n {%- do return(adapter.dispatch('get_materialized_view_configuration_changes', 'dbt')(existing_relation, new_config)) -%}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_materialized_view_configuration_changes"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956128, "supported_languages": null}, "macro.dbt.default__get_materialized_view_configuration_changes": {"name": "default__get_materialized_view_configuration_changes", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/alter.sql", "original_file_path": "macros/relations/materialized_view/alter.sql", "unique_id": "macro.dbt.default__get_materialized_view_configuration_changes", "macro_sql": "{% macro default__get_materialized_view_configuration_changes(existing_relation, new_config) %}\n {{ exceptions.raise_compiler_error(\"Materialized views have not been implemented for this adapter.\") }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9561338, "supported_languages": null}, "macro.dbt.get_create_materialized_view_as_sql": {"name": "get_create_materialized_view_as_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/create.sql", "original_file_path": "macros/relations/materialized_view/create.sql", "unique_id": "macro.dbt.get_create_materialized_view_as_sql", "macro_sql": "{% macro get_create_materialized_view_as_sql(relation, sql) -%}\n {{- adapter.dispatch('get_create_materialized_view_as_sql', 'dbt')(relation, sql) -}}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_create_materialized_view_as_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9561448, "supported_languages": null}, "macro.dbt.default__get_create_materialized_view_as_sql": {"name": "default__get_create_materialized_view_as_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/create.sql", "original_file_path": "macros/relations/materialized_view/create.sql", "unique_id": "macro.dbt.default__get_create_materialized_view_as_sql", "macro_sql": "{% macro default__get_create_materialized_view_as_sql(relation, sql) -%}\n {{ exceptions.raise_compiler_error(\n \"`get_create_materialized_view_as_sql` has not been implemented for this adapter.\"\n ) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956152, "supported_languages": null}, "macro.dbt.get_table_columns_and_constraints": {"name": "get_table_columns_and_constraints", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/column/columns_spec_ddl.sql", "original_file_path": "macros/relations/column/columns_spec_ddl.sql", "unique_id": "macro.dbt.get_table_columns_and_constraints", "macro_sql": "{%- macro get_table_columns_and_constraints() -%}\n {{ adapter.dispatch('get_table_columns_and_constraints', 'dbt')() }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__get_table_columns_and_constraints"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956168, "supported_languages": null}, "macro.dbt.default__get_table_columns_and_constraints": {"name": "default__get_table_columns_and_constraints", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/column/columns_spec_ddl.sql", "original_file_path": "macros/relations/column/columns_spec_ddl.sql", "unique_id": "macro.dbt.default__get_table_columns_and_constraints", "macro_sql": "{% macro default__get_table_columns_and_constraints() -%}\n {{ return(table_columns_and_constraints()) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.table_columns_and_constraints"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956178, "supported_languages": null}, "macro.dbt.table_columns_and_constraints": {"name": "table_columns_and_constraints", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/column/columns_spec_ddl.sql", "original_file_path": "macros/relations/column/columns_spec_ddl.sql", "unique_id": "macro.dbt.table_columns_and_constraints", "macro_sql": "{% macro table_columns_and_constraints() %}\n {# loop through user_provided_columns to create DDL with data types and constraints #}\n {%- set raw_column_constraints = adapter.render_raw_columns_constraints(raw_columns=model['columns']) -%}\n {%- set raw_model_constraints = adapter.render_raw_model_constraints(raw_constraints=model['constraints']) -%}\n (\n {% for c in raw_column_constraints -%}\n {{ c }}{{ \",\" if not loop.last or raw_model_constraints }}\n {% endfor %}\n {% for c in raw_model_constraints -%}\n {{ c }}{{ \",\" if not loop.last }}\n {% endfor -%}\n )\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9561841, "supported_languages": null}, "macro.dbt.get_assert_columns_equivalent": {"name": "get_assert_columns_equivalent", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/column/columns_spec_ddl.sql", "original_file_path": "macros/relations/column/columns_spec_ddl.sql", "unique_id": "macro.dbt.get_assert_columns_equivalent", "macro_sql": "\n\n{%- macro get_assert_columns_equivalent(sql) -%}\n {{ adapter.dispatch('get_assert_columns_equivalent', 'dbt')(sql) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__get_assert_columns_equivalent"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9561942, "supported_languages": null}, "macro.dbt.default__get_assert_columns_equivalent": {"name": "default__get_assert_columns_equivalent", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/column/columns_spec_ddl.sql", "original_file_path": "macros/relations/column/columns_spec_ddl.sql", "unique_id": "macro.dbt.default__get_assert_columns_equivalent", "macro_sql": "{% macro default__get_assert_columns_equivalent(sql) -%}\n {{ return(assert_columns_equivalent(sql)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.assert_columns_equivalent"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9562, "supported_languages": null}, "macro.dbt.assert_columns_equivalent": {"name": "assert_columns_equivalent", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/column/columns_spec_ddl.sql", "original_file_path": "macros/relations/column/columns_spec_ddl.sql", "unique_id": "macro.dbt.assert_columns_equivalent", "macro_sql": "{% macro assert_columns_equivalent(sql) %}\n\n {#-- First ensure the user has defined 'columns' in yaml specification --#}\n {%- set user_defined_columns = model['columns'] -%}\n {%- if not user_defined_columns -%}\n {{ exceptions.raise_contract_error([], []) }}\n {%- endif -%}\n\n {#-- Obtain the column schema provided by sql file. #}\n {%- set sql_file_provided_columns = get_column_schema_from_query(sql, config.get('sql_header', none)) -%}\n {#--Obtain the column schema provided by the schema file by generating an 'empty schema' query from the model's columns. #}\n {%- set schema_file_provided_columns = get_column_schema_from_query(get_empty_schema_sql(user_defined_columns)) -%}\n\n {#-- create dictionaries with name and formatted data type and strings for exception #}\n {%- set sql_columns = format_columns(sql_file_provided_columns) -%}\n {%- set yaml_columns = format_columns(schema_file_provided_columns) -%}\n\n {%- if sql_columns|length != yaml_columns|length -%}\n {%- do exceptions.raise_contract_error(yaml_columns, sql_columns) -%}\n {%- endif -%}\n\n {%- for sql_col in sql_columns -%}\n {%- set yaml_col = [] -%}\n {%- for this_col in yaml_columns -%}\n {%- if this_col['name'] == sql_col['name'] -%}\n {%- do yaml_col.append(this_col) -%}\n {%- break -%}\n {%- endif -%}\n {%- endfor -%}\n {%- if not yaml_col -%}\n {#-- Column with name not found in yaml #}\n {%- do exceptions.raise_contract_error(yaml_columns, sql_columns) -%}\n {%- endif -%}\n {%- if sql_col['formatted'] != yaml_col[0]['formatted'] -%}\n {#-- Column data types don't match #}\n {%- do exceptions.raise_contract_error(yaml_columns, sql_columns) -%}\n {%- endif -%}\n {%- endfor -%}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_column_schema_from_query", "macro.dbt.get_empty_schema_sql", "macro.dbt.format_columns"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956212, "supported_languages": null}, "macro.dbt.format_columns": {"name": "format_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/column/columns_spec_ddl.sql", "original_file_path": "macros/relations/column/columns_spec_ddl.sql", "unique_id": "macro.dbt.format_columns", "macro_sql": "{% macro format_columns(columns) %}\n {% set formatted_columns = [] %}\n {% for column in columns %}\n {%- set formatted_column = adapter.dispatch('format_column', 'dbt')(column) -%}\n {%- do formatted_columns.append(formatted_column) -%}\n {% endfor %}\n {{ return(formatted_columns) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__format_column"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956223, "supported_languages": null}, "macro.dbt.default__format_column": {"name": "default__format_column", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/column/columns_spec_ddl.sql", "original_file_path": "macros/relations/column/columns_spec_ddl.sql", "unique_id": "macro.dbt.default__format_column", "macro_sql": "{% macro default__format_column(column) -%}\n {% set data_type = column.dtype %}\n {% set formatted = column.column.lower() ~ \" \" ~ data_type %}\n {{ return({'name': column.name, 'data_type': data_type, 'formatted': formatted}) }}\n{%- endmacro -%}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95623, "supported_languages": null}, "macro.dbt.drop_table": {"name": "drop_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/drop.sql", "original_file_path": "macros/relations/table/drop.sql", "unique_id": "macro.dbt.drop_table", "macro_sql": "{% macro drop_table(relation) -%}\n {{- adapter.dispatch('drop_table', 'dbt')(relation) -}}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__drop_table"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9562418, "supported_languages": null}, "macro.dbt.default__drop_table": {"name": "default__drop_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/drop.sql", "original_file_path": "macros/relations/table/drop.sql", "unique_id": "macro.dbt.default__drop_table", "macro_sql": "{% macro default__drop_table(relation) -%}\n drop table if exists {{ relation }} cascade\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956247, "supported_languages": null}, "macro.dbt.get_replace_table_sql": {"name": "get_replace_table_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/replace.sql", "original_file_path": "macros/relations/table/replace.sql", "unique_id": "macro.dbt.get_replace_table_sql", "macro_sql": "{% macro get_replace_table_sql(relation, sql) %}\n {{- adapter.dispatch('get_replace_table_sql', 'dbt')(relation, sql) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_replace_table_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956259, "supported_languages": null}, "macro.dbt.default__get_replace_table_sql": {"name": "default__get_replace_table_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/replace.sql", "original_file_path": "macros/relations/table/replace.sql", "unique_id": "macro.dbt.default__get_replace_table_sql", "macro_sql": "{% macro default__get_replace_table_sql(relation, sql) %}\n {{ exceptions.raise_compiler_error(\n \"`get_replace_table_sql` has not been implemented for this adapter.\"\n ) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956268, "supported_languages": null}, "macro.dbt.get_rename_table_sql": {"name": "get_rename_table_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/rename.sql", "original_file_path": "macros/relations/table/rename.sql", "unique_id": "macro.dbt.get_rename_table_sql", "macro_sql": "{% macro get_rename_table_sql(relation, new_name) %}\n {{- adapter.dispatch('get_rename_table_sql', 'dbt')(relation, new_name) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_rename_table_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95628, "supported_languages": null}, "macro.dbt.default__get_rename_table_sql": {"name": "default__get_rename_table_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/rename.sql", "original_file_path": "macros/relations/table/rename.sql", "unique_id": "macro.dbt.default__get_rename_table_sql", "macro_sql": "{% macro default__get_rename_table_sql(relation, new_name) %}\n {{ exceptions.raise_compiler_error(\n \"`get_rename_table_sql` has not been implemented for this adapter.\"\n ) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956286, "supported_languages": null}, "macro.dbt.get_create_table_as_sql": {"name": "get_create_table_as_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/create.sql", "original_file_path": "macros/relations/table/create.sql", "unique_id": "macro.dbt.get_create_table_as_sql", "macro_sql": "{% macro get_create_table_as_sql(temporary, relation, sql) -%}\n {{ adapter.dispatch('get_create_table_as_sql', 'dbt')(temporary, relation, sql) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_create_table_as_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956301, "supported_languages": null}, "macro.dbt.default__get_create_table_as_sql": {"name": "default__get_create_table_as_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/create.sql", "original_file_path": "macros/relations/table/create.sql", "unique_id": "macro.dbt.default__get_create_table_as_sql", "macro_sql": "{% macro default__get_create_table_as_sql(temporary, relation, sql) -%}\n {{ return(create_table_as(temporary, relation, sql)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.create_table_as"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956308, "supported_languages": null}, "macro.dbt.create_table_as": {"name": "create_table_as", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/create.sql", "original_file_path": "macros/relations/table/create.sql", "unique_id": "macro.dbt.create_table_as", "macro_sql": "{% macro create_table_as(temporary, relation, compiled_code, language='sql') -%}\n {# backward compatibility for create_table_as that does not support language #}\n {% if language == \"sql\" %}\n {{ adapter.dispatch('create_table_as', 'dbt')(temporary, relation, compiled_code)}}\n {% else %}\n {{ adapter.dispatch('create_table_as', 'dbt')(temporary, relation, compiled_code, language) }}\n {% endif %}\n\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__create_table_as"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956314, "supported_languages": null}, "macro.dbt.default__create_table_as": {"name": "default__create_table_as", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/create.sql", "original_file_path": "macros/relations/table/create.sql", "unique_id": "macro.dbt.default__create_table_as", "macro_sql": "{% macro default__create_table_as(temporary, relation, sql) -%}\n {%- set sql_header = config.get('sql_header', none) -%}\n\n {{ sql_header if sql_header is not none }}\n\n create {% if temporary: -%}temporary{%- endif %} table\n {{ relation.include(database=(not temporary), schema=(not temporary)) }}\n {% set contract_config = config.get('contract') %}\n {% if contract_config.enforced and (not temporary) %}\n {{ get_assert_columns_equivalent(sql) }}\n {{ get_table_columns_and_constraints() }}\n {%- set sql = get_select_subquery(sql) %}\n {% endif %}\n as (\n {{ sql }}\n );\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.get_assert_columns_equivalent", "macro.dbt.get_table_columns_and_constraints", "macro.dbt.get_select_subquery"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956325, "supported_languages": null}, "macro.dbt.default__get_column_names": {"name": "default__get_column_names", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/create.sql", "original_file_path": "macros/relations/table/create.sql", "unique_id": "macro.dbt.default__get_column_names", "macro_sql": "{% macro default__get_column_names() %}\n {#- loop through user_provided_columns to get column names -#}\n {%- set user_provided_columns = model['columns'] -%}\n {%- for i in user_provided_columns %}\n {%- set col = user_provided_columns[i] -%}\n {%- set col_name = adapter.quote(col['name']) if col.get('quote') else col['name'] -%}\n {{ col_name }}{{ \", \" if not loop.last }}\n {%- endfor -%}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95633, "supported_languages": null}, "macro.dbt.get_select_subquery": {"name": "get_select_subquery", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/create.sql", "original_file_path": "macros/relations/table/create.sql", "unique_id": "macro.dbt.get_select_subquery", "macro_sql": "{% macro get_select_subquery(sql) %}\n {{ return(adapter.dispatch('get_select_subquery', 'dbt')(sql)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_select_subquery"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956337, "supported_languages": null}, "macro.dbt.default__get_select_subquery": {"name": "default__get_select_subquery", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/create.sql", "original_file_path": "macros/relations/table/create.sql", "unique_id": "macro.dbt.default__get_select_subquery", "macro_sql": "{% macro default__get_select_subquery(sql) %}\n select {{ adapter.dispatch('get_column_names', 'dbt')() }}\n from (\n {{ sql }}\n ) as model_subq\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_column_names"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956345, "supported_languages": null}, "macro.dbt.drop_view": {"name": "drop_view", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/drop.sql", "original_file_path": "macros/relations/view/drop.sql", "unique_id": "macro.dbt.drop_view", "macro_sql": "{% macro drop_view(relation) -%}\n {{- adapter.dispatch('drop_view', 'dbt')(relation) -}}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__drop_view"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956357, "supported_languages": null}, "macro.dbt.default__drop_view": {"name": "default__drop_view", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/drop.sql", "original_file_path": "macros/relations/view/drop.sql", "unique_id": "macro.dbt.default__drop_view", "macro_sql": "{% macro default__drop_view(relation) -%}\n drop view if exists {{ relation }} cascade\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956364, "supported_languages": null}, "macro.dbt.get_replace_view_sql": {"name": "get_replace_view_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/replace.sql", "original_file_path": "macros/relations/view/replace.sql", "unique_id": "macro.dbt.get_replace_view_sql", "macro_sql": "{% macro get_replace_view_sql(relation, sql) %}\n {{- adapter.dispatch('get_replace_view_sql', 'dbt')(relation, sql) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_replace_view_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956377, "supported_languages": null}, "macro.dbt.default__get_replace_view_sql": {"name": "default__get_replace_view_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/replace.sql", "original_file_path": "macros/relations/view/replace.sql", "unique_id": "macro.dbt.default__get_replace_view_sql", "macro_sql": "{% macro default__get_replace_view_sql(relation, sql) %}\n {{ exceptions.raise_compiler_error(\n \"`get_replace_view_sql` has not been implemented for this adapter.\"\n ) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956382, "supported_languages": null}, "macro.dbt.create_or_replace_view": {"name": "create_or_replace_view", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/replace.sql", "original_file_path": "macros/relations/view/replace.sql", "unique_id": "macro.dbt.create_or_replace_view", "macro_sql": "{% macro create_or_replace_view() %}\n {%- set identifier = model['alias'] -%}\n\n {%- set old_relation = adapter.get_relation(database=database, schema=schema, identifier=identifier) -%}\n {%- set exists_as_view = (old_relation is not none and old_relation.is_view) -%}\n\n {%- set target_relation = api.Relation.create(\n identifier=identifier, schema=schema, database=database,\n type='view') -%}\n {% set grant_config = config.get('grants') %}\n\n {{ run_hooks(pre_hooks) }}\n\n -- If there's a table with the same name and we weren't told to full refresh,\n -- that's an error. If we were told to full refresh, drop it. This behavior differs\n -- for Snowflake and BigQuery, so multiple dispatch is used.\n {%- if old_relation is not none and old_relation.is_table -%}\n {{ handle_existing_table(should_full_refresh(), old_relation) }}\n {%- endif -%}\n\n -- build model\n {% call statement('main') -%}\n {{ get_create_view_as_sql(target_relation, sql) }}\n {%- endcall %}\n\n {% set should_revoke = should_revoke(exists_as_view, full_refresh_mode=True) %}\n {% do apply_grants(target_relation, grant_config, should_revoke=should_revoke) %}\n\n {{ run_hooks(post_hooks) }}\n\n {{ return({'relations': [target_relation]}) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.run_hooks", "macro.dbt.handle_existing_table", "macro.dbt.should_full_refresh", "macro.dbt.statement", "macro.dbt.get_create_view_as_sql", "macro.dbt.should_revoke", "macro.dbt.apply_grants"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956392, "supported_languages": null}, "macro.dbt.handle_existing_table": {"name": "handle_existing_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/replace.sql", "original_file_path": "macros/relations/view/replace.sql", "unique_id": "macro.dbt.handle_existing_table", "macro_sql": "{% macro handle_existing_table(full_refresh, old_relation) %}\n {{ adapter.dispatch('handle_existing_table', 'dbt')(full_refresh, old_relation) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__handle_existing_table"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9564018, "supported_languages": null}, "macro.dbt.default__handle_existing_table": {"name": "default__handle_existing_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/replace.sql", "original_file_path": "macros/relations/view/replace.sql", "unique_id": "macro.dbt.default__handle_existing_table", "macro_sql": "{% macro default__handle_existing_table(full_refresh, old_relation) %}\n {{ log(\"Dropping relation \" ~ old_relation ~ \" because it is of type \" ~ old_relation.type) }}\n {{ adapter.drop_relation(old_relation) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956948, "supported_languages": null}, "macro.dbt.get_rename_view_sql": {"name": "get_rename_view_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/rename.sql", "original_file_path": "macros/relations/view/rename.sql", "unique_id": "macro.dbt.get_rename_view_sql", "macro_sql": "{% macro get_rename_view_sql(relation, new_name) %}\n {{- adapter.dispatch('get_rename_view_sql', 'dbt')(relation, new_name) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_rename_view_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956962, "supported_languages": null}, "macro.dbt.default__get_rename_view_sql": {"name": "default__get_rename_view_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/rename.sql", "original_file_path": "macros/relations/view/rename.sql", "unique_id": "macro.dbt.default__get_rename_view_sql", "macro_sql": "{% macro default__get_rename_view_sql(relation, new_name) %}\n {{ exceptions.raise_compiler_error(\n \"`get_rename_view_sql` has not been implemented for this adapter.\"\n ) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9569669, "supported_languages": null}, "macro.dbt.get_create_view_as_sql": {"name": "get_create_view_as_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/create.sql", "original_file_path": "macros/relations/view/create.sql", "unique_id": "macro.dbt.get_create_view_as_sql", "macro_sql": "{% macro get_create_view_as_sql(relation, sql) -%}\n {{ adapter.dispatch('get_create_view_as_sql', 'dbt')(relation, sql) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_create_view_as_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956979, "supported_languages": null}, "macro.dbt.default__get_create_view_as_sql": {"name": "default__get_create_view_as_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/create.sql", "original_file_path": "macros/relations/view/create.sql", "unique_id": "macro.dbt.default__get_create_view_as_sql", "macro_sql": "{% macro default__get_create_view_as_sql(relation, sql) -%}\n {{ return(create_view_as(relation, sql)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.create_view_as"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956987, "supported_languages": null}, "macro.dbt.create_view_as": {"name": "create_view_as", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/create.sql", "original_file_path": "macros/relations/view/create.sql", "unique_id": "macro.dbt.create_view_as", "macro_sql": "{% macro create_view_as(relation, sql) -%}\n {{ adapter.dispatch('create_view_as', 'dbt')(relation, sql) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__create_view_as"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.956994, "supported_languages": null}, "macro.dbt.default__create_view_as": {"name": "default__create_view_as", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/create.sql", "original_file_path": "macros/relations/view/create.sql", "unique_id": "macro.dbt.default__create_view_as", "macro_sql": "{% macro default__create_view_as(relation, sql) -%}\n {%- set sql_header = config.get('sql_header', none) -%}\n\n {{ sql_header if sql_header is not none }}\n create view {{ relation }}\n {% set contract_config = config.get('contract') %}\n {% if contract_config.enforced %}\n {{ get_assert_columns_equivalent(sql) }}\n {%- endif %}\n as (\n {{ sql }}\n );\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.get_assert_columns_equivalent"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957, "supported_languages": null}, "macro.dbt.default__test_relationships": {"name": "default__test_relationships", "resource_type": "macro", "package_name": "dbt", "path": "macros/generic_test_sql/relationships.sql", "original_file_path": "macros/generic_test_sql/relationships.sql", "unique_id": "macro.dbt.default__test_relationships", "macro_sql": "{% macro default__test_relationships(model, column_name, to, field) %}\n\nwith child as (\n select {{ column_name }} as from_field\n from {{ model }}\n where {{ column_name }} is not null\n),\n\nparent as (\n select {{ field }} as to_field\n from {{ to }}\n)\n\nselect\n from_field\n\nfrom child\nleft join parent\n on child.from_field = parent.to_field\n\nwhere parent.to_field is null\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9570198, "supported_languages": null}, "macro.dbt.default__test_not_null": {"name": "default__test_not_null", "resource_type": "macro", "package_name": "dbt", "path": "macros/generic_test_sql/not_null.sql", "original_file_path": "macros/generic_test_sql/not_null.sql", "unique_id": "macro.dbt.default__test_not_null", "macro_sql": "{% macro default__test_not_null(model, column_name) %}\n\n{% set column_list = '*' if should_store_failures() else column_name %}\n\nselect {{ column_list }}\nfrom {{ model }}\nwhere {{ column_name }} is null\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.should_store_failures"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9570298, "supported_languages": null}, "macro.dbt.default__test_unique": {"name": "default__test_unique", "resource_type": "macro", "package_name": "dbt", "path": "macros/generic_test_sql/unique.sql", "original_file_path": "macros/generic_test_sql/unique.sql", "unique_id": "macro.dbt.default__test_unique", "macro_sql": "{% macro default__test_unique(model, column_name) %}\n\nselect\n {{ column_name }} as unique_field,\n count(*) as n_records\n\nfrom {{ model }}\nwhere {{ column_name }} is not null\ngroup by {{ column_name }}\nhaving count(*) > 1\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957046, "supported_languages": null}, "macro.dbt.default__test_accepted_values": {"name": "default__test_accepted_values", "resource_type": "macro", "package_name": "dbt", "path": "macros/generic_test_sql/accepted_values.sql", "original_file_path": "macros/generic_test_sql/accepted_values.sql", "unique_id": "macro.dbt.default__test_accepted_values", "macro_sql": "{% macro default__test_accepted_values(model, column_name, values, quote=True) %}\n\nwith all_values as (\n\n select\n {{ column_name }} as value_field,\n count(*) as n_records\n\n from {{ model }}\n group by {{ column_name }}\n\n)\n\nselect *\nfrom all_values\nwhere value_field not in (\n {% for value in values -%}\n {% if quote -%}\n '{{ value }}'\n {%- else -%}\n {{ value }}\n {%- endif -%}\n {%- if not loop.last -%},{%- endif %}\n {%- endfor %}\n)\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957057, "supported_languages": null}, "macro.dbt.statement": {"name": "statement", "resource_type": "macro", "package_name": "dbt", "path": "macros/etc/statement.sql", "original_file_path": "macros/etc/statement.sql", "unique_id": "macro.dbt.statement", "macro_sql": "\n{%- macro statement(name=None, fetch_result=False, auto_begin=True, language='sql') -%}\n {%- if execute: -%}\n {%- set compiled_code = caller() -%}\n\n {%- if name == 'main' -%}\n {{ log('Writing runtime {} for node \"{}\"'.format(language, model['unique_id'])) }}\n {{ write(compiled_code) }}\n {%- endif -%}\n {%- if language == 'sql'-%}\n {%- set res, table = adapter.execute(compiled_code, auto_begin=auto_begin, fetch=fetch_result) -%}\n {%- elif language == 'python' -%}\n {%- set res = submit_python_job(model, compiled_code) -%}\n {#-- TODO: What should table be for python models? --#}\n {%- set table = None -%}\n {%- else -%}\n {% do exceptions.raise_compiler_error(\"statement macro didn't get supported language\") %}\n {%- endif -%}\n\n {%- if name is not none -%}\n {{ store_result(name, response=res, agate_table=table) }}\n {%- endif -%}\n\n {%- endif -%}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957077, "supported_languages": null}, "macro.dbt.noop_statement": {"name": "noop_statement", "resource_type": "macro", "package_name": "dbt", "path": "macros/etc/statement.sql", "original_file_path": "macros/etc/statement.sql", "unique_id": "macro.dbt.noop_statement", "macro_sql": "{% macro noop_statement(name=None, message=None, code=None, rows_affected=None, res=None) -%}\n {%- set sql = caller() -%}\n\n {%- if name == 'main' -%}\n {{ log('Writing runtime SQL for node \"{}\"'.format(model['unique_id'])) }}\n {{ write(sql) }}\n {%- endif -%}\n\n {%- if name is not none -%}\n {{ store_raw_result(name, message=message, code=code, rows_affected=rows_affected, agate_table=res) }}\n {%- endif -%}\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957084, "supported_languages": null}, "macro.dbt.run_query": {"name": "run_query", "resource_type": "macro", "package_name": "dbt", "path": "macros/etc/statement.sql", "original_file_path": "macros/etc/statement.sql", "unique_id": "macro.dbt.run_query", "macro_sql": "{% macro run_query(sql) %}\n {% call statement(\"run_query_statement\", fetch_result=true, auto_begin=false) %}\n {{ sql }}\n {% endcall %}\n\n {% do return(load_result(\"run_query_statement\").table) %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957089, "supported_languages": null}, "macro.dbt.convert_datetime": {"name": "convert_datetime", "resource_type": "macro", "package_name": "dbt", "path": "macros/etc/datetime.sql", "original_file_path": "macros/etc/datetime.sql", "unique_id": "macro.dbt.convert_datetime", "macro_sql": "{% macro convert_datetime(date_str, date_fmt) %}\n\n {% set error_msg -%}\n The provided partition date '{{ date_str }}' does not match the expected format '{{ date_fmt }}'\n {%- endset %}\n\n {% set res = try_or_compiler_error(error_msg, modules.datetime.datetime.strptime, date_str.strip(), date_fmt) %}\n {{ return(res) }}\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9571042, "supported_languages": null}, "macro.dbt.dates_in_range": {"name": "dates_in_range", "resource_type": "macro", "package_name": "dbt", "path": "macros/etc/datetime.sql", "original_file_path": "macros/etc/datetime.sql", "unique_id": "macro.dbt.dates_in_range", "macro_sql": "{% macro dates_in_range(start_date_str, end_date_str=none, in_fmt=\"%Y%m%d\", out_fmt=\"%Y%m%d\") %}\n {% set end_date_str = start_date_str if end_date_str is none else end_date_str %}\n\n {% set start_date = convert_datetime(start_date_str, in_fmt) %}\n {% set end_date = convert_datetime(end_date_str, in_fmt) %}\n\n {% set day_count = (end_date - start_date).days %}\n {% if day_count < 0 %}\n {% set msg -%}\n Partition start date is after the end date ({{ start_date }}, {{ end_date }})\n {%- endset %}\n\n {{ exceptions.raise_compiler_error(msg, model) }}\n {% endif %}\n\n {% set date_list = [] %}\n {% for i in range(0, day_count + 1) %}\n {% set the_date = (modules.datetime.timedelta(days=i) + start_date) %}\n {% if not out_fmt %}\n {% set _ = date_list.append(the_date) %}\n {% else %}\n {% set _ = date_list.append(the_date.strftime(out_fmt)) %}\n {% endif %}\n {% endfor %}\n\n {{ return(date_list) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.convert_datetime"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957112, "supported_languages": null}, "macro.dbt.partition_range": {"name": "partition_range", "resource_type": "macro", "package_name": "dbt", "path": "macros/etc/datetime.sql", "original_file_path": "macros/etc/datetime.sql", "unique_id": "macro.dbt.partition_range", "macro_sql": "{% macro partition_range(raw_partition_date, date_fmt='%Y%m%d') %}\n {% set partition_range = (raw_partition_date | string).split(\",\") %}\n\n {% if (partition_range | length) == 1 %}\n {% set start_date = partition_range[0] %}\n {% set end_date = none %}\n {% elif (partition_range | length) == 2 %}\n {% set start_date = partition_range[0] %}\n {% set end_date = partition_range[1] %}\n {% else %}\n {{ exceptions.raise_compiler_error(\"Invalid partition time. Expected format: {Start Date}[,{End Date}]. Got: \" ~ raw_partition_date) }}\n {% endif %}\n\n {{ return(dates_in_range(start_date, end_date, in_fmt=date_fmt)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.dates_in_range"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957119, "supported_languages": null}, "macro.dbt.py_current_timestring": {"name": "py_current_timestring", "resource_type": "macro", "package_name": "dbt", "path": "macros/etc/datetime.sql", "original_file_path": "macros/etc/datetime.sql", "unique_id": "macro.dbt.py_current_timestring", "macro_sql": "{% macro py_current_timestring() %}\n {% set dt = modules.datetime.datetime.now() %}\n {% do return(dt.strftime(\"%Y%m%d%H%M%S%f\")) %}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957124, "supported_languages": null}, "macro.dbt.except": {"name": "except", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/except.sql", "original_file_path": "macros/utils/except.sql", "unique_id": "macro.dbt.except", "macro_sql": "{% macro except() %}\n {{ return(adapter.dispatch('except', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__except"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9571369, "supported_languages": null}, "macro.dbt.default__except": {"name": "default__except", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/except.sql", "original_file_path": "macros/utils/except.sql", "unique_id": "macro.dbt.default__except", "macro_sql": "{% macro default__except() %}\n\n except\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957142, "supported_languages": null}, "macro.dbt.get_intervals_between": {"name": "get_intervals_between", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/date_spine.sql", "original_file_path": "macros/utils/date_spine.sql", "unique_id": "macro.dbt.get_intervals_between", "macro_sql": "{% macro get_intervals_between(start_date, end_date, datepart) -%}\n {{ return(adapter.dispatch('get_intervals_between', 'dbt')(start_date, end_date, datepart)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_intervals_between"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957155, "supported_languages": null}, "macro.dbt.default__get_intervals_between": {"name": "default__get_intervals_between", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/date_spine.sql", "original_file_path": "macros/utils/date_spine.sql", "unique_id": "macro.dbt.default__get_intervals_between", "macro_sql": "{% macro default__get_intervals_between(start_date, end_date, datepart) -%}\n {%- call statement('get_intervals_between', fetch_result=True) %}\n\n select {{ dbt.datediff(start_date, end_date, datepart) }}\n\n {%- endcall -%}\n\n {%- set value_list = load_result('get_intervals_between') -%}\n\n {%- if value_list and value_list['data'] -%}\n {%- set values = value_list['data'] | map(attribute=0) | list %}\n {{ return(values[0]) }}\n {%- else -%}\n {{ return(1) }}\n {%- endif -%}\n\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.statement", "macro.dbt.datediff"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957164, "supported_languages": null}, "macro.dbt.date_spine": {"name": "date_spine", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/date_spine.sql", "original_file_path": "macros/utils/date_spine.sql", "unique_id": "macro.dbt.date_spine", "macro_sql": "{% macro date_spine(datepart, start_date, end_date) %}\n {{ return(adapter.dispatch('date_spine', 'dbt')(datepart, start_date, end_date)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__date_spine"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957172, "supported_languages": null}, "macro.dbt.default__date_spine": {"name": "default__date_spine", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/date_spine.sql", "original_file_path": "macros/utils/date_spine.sql", "unique_id": "macro.dbt.default__date_spine", "macro_sql": "{% macro default__date_spine(datepart, start_date, end_date) %}\n\n\n {# call as follows:\n\n date_spine(\n \"day\",\n \"to_date('01/01/2016', 'mm/dd/yyyy')\",\n \"dbt.dateadd(week, 1, current_date)\"\n ) #}\n\n\n with rawdata as (\n\n {{dbt.generate_series(\n dbt.get_intervals_between(start_date, end_date, datepart)\n )}}\n\n ),\n\n all_periods as (\n\n select (\n {{\n dbt.dateadd(\n datepart,\n \"row_number() over (order by 1) - 1\",\n start_date\n )\n }}\n ) as date_{{datepart}}\n from rawdata\n\n ),\n\n filtered as (\n\n select *\n from all_periods\n where date_{{datepart}} <= {{ end_date }}\n\n )\n\n select * from filtered\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.generate_series", "macro.dbt.get_intervals_between", "macro.dbt.dateadd"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95718, "supported_languages": null}, "macro.dbt.replace": {"name": "replace", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/replace.sql", "original_file_path": "macros/utils/replace.sql", "unique_id": "macro.dbt.replace", "macro_sql": "{% macro replace(field, old_chars, new_chars) -%}\n {{ return(adapter.dispatch('replace', 'dbt') (field, old_chars, new_chars)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__replace"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957191, "supported_languages": null}, "macro.dbt.default__replace": {"name": "default__replace", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/replace.sql", "original_file_path": "macros/utils/replace.sql", "unique_id": "macro.dbt.default__replace", "macro_sql": "{% macro default__replace(field, old_chars, new_chars) %}\n\n replace(\n {{ field }},\n {{ old_chars }},\n {{ new_chars }}\n )\n\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9571981, "supported_languages": null}, "macro.dbt.concat": {"name": "concat", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/concat.sql", "original_file_path": "macros/utils/concat.sql", "unique_id": "macro.dbt.concat", "macro_sql": "{% macro concat(fields) -%}\n {{ return(adapter.dispatch('concat', 'dbt')(fields)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__concat"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957212, "supported_languages": null}, "macro.dbt.default__concat": {"name": "default__concat", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/concat.sql", "original_file_path": "macros/utils/concat.sql", "unique_id": "macro.dbt.default__concat", "macro_sql": "{% macro default__concat(fields) -%}\n {{ fields|join(' || ') }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957218, "supported_languages": null}, "macro.dbt.get_powers_of_two": {"name": "get_powers_of_two", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/generate_series.sql", "original_file_path": "macros/utils/generate_series.sql", "unique_id": "macro.dbt.get_powers_of_two", "macro_sql": "{% macro get_powers_of_two(upper_bound) %}\n {{ return(adapter.dispatch('get_powers_of_two', 'dbt')(upper_bound)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_powers_of_two"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9572291, "supported_languages": null}, "macro.dbt.default__get_powers_of_two": {"name": "default__get_powers_of_two", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/generate_series.sql", "original_file_path": "macros/utils/generate_series.sql", "unique_id": "macro.dbt.default__get_powers_of_two", "macro_sql": "{% macro default__get_powers_of_two(upper_bound) %}\n\n {% if upper_bound <= 0 %}\n {{ exceptions.raise_compiler_error(\"upper bound must be positive\") }}\n {% endif %}\n\n {% for _ in range(1, 100) %}\n {% if upper_bound <= 2 ** loop.index %}{{ return(loop.index) }}{% endif %}\n {% endfor %}\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957238, "supported_languages": null}, "macro.dbt.generate_series": {"name": "generate_series", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/generate_series.sql", "original_file_path": "macros/utils/generate_series.sql", "unique_id": "macro.dbt.generate_series", "macro_sql": "{% macro generate_series(upper_bound) %}\n {{ return(adapter.dispatch('generate_series', 'dbt')(upper_bound)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__generate_series"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957244, "supported_languages": null}, "macro.dbt.default__generate_series": {"name": "default__generate_series", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/generate_series.sql", "original_file_path": "macros/utils/generate_series.sql", "unique_id": "macro.dbt.default__generate_series", "macro_sql": "{% macro default__generate_series(upper_bound) %}\n\n {% set n = dbt.get_powers_of_two(upper_bound) %}\n\n with p as (\n select 0 as generated_number union all select 1\n ), unioned as (\n\n select\n\n {% for i in range(n) %}\n p{{i}}.generated_number * power(2, {{i}})\n {% if not loop.last %} + {% endif %}\n {% endfor %}\n + 1\n as generated_number\n\n from\n\n {% for i in range(n) %}\n p as p{{i}}\n {% if not loop.last %} cross join {% endif %}\n {% endfor %}\n\n )\n\n select *\n from unioned\n where generated_number <= {{upper_bound}}\n order by generated_number\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_powers_of_two"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95725, "supported_languages": null}, "macro.dbt.length": {"name": "length", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/length.sql", "original_file_path": "macros/utils/length.sql", "unique_id": "macro.dbt.length", "macro_sql": "{% macro length(expression) -%}\n {{ return(adapter.dispatch('length', 'dbt') (expression)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__length"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957265, "supported_languages": null}, "macro.dbt.default__length": {"name": "default__length", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/length.sql", "original_file_path": "macros/utils/length.sql", "unique_id": "macro.dbt.default__length", "macro_sql": "{% macro default__length(expression) %}\n\n length(\n {{ expression }}\n )\n\n{%- endmacro -%}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957271, "supported_languages": null}, "macro.dbt.dateadd": {"name": "dateadd", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/dateadd.sql", "original_file_path": "macros/utils/dateadd.sql", "unique_id": "macro.dbt.dateadd", "macro_sql": "{% macro dateadd(datepart, interval, from_date_or_timestamp) %}\n {{ return(adapter.dispatch('dateadd', 'dbt')(datepart, interval, from_date_or_timestamp)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__dateadd"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957282, "supported_languages": null}, "macro.dbt.default__dateadd": {"name": "default__dateadd", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/dateadd.sql", "original_file_path": "macros/utils/dateadd.sql", "unique_id": "macro.dbt.default__dateadd", "macro_sql": "{% macro default__dateadd(datepart, interval, from_date_or_timestamp) %}\n\n dateadd(\n {{ datepart }},\n {{ interval }},\n {{ from_date_or_timestamp }}\n )\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957288, "supported_languages": null}, "macro.dbt.intersect": {"name": "intersect", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/intersect.sql", "original_file_path": "macros/utils/intersect.sql", "unique_id": "macro.dbt.intersect", "macro_sql": "{% macro intersect() %}\n {{ return(adapter.dispatch('intersect', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__intersect"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957299, "supported_languages": null}, "macro.dbt.default__intersect": {"name": "default__intersect", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/intersect.sql", "original_file_path": "macros/utils/intersect.sql", "unique_id": "macro.dbt.default__intersect", "macro_sql": "{% macro default__intersect() %}\n\n intersect\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9573078, "supported_languages": null}, "macro.dbt.escape_single_quotes": {"name": "escape_single_quotes", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/escape_single_quotes.sql", "original_file_path": "macros/utils/escape_single_quotes.sql", "unique_id": "macro.dbt.escape_single_quotes", "macro_sql": "{% macro escape_single_quotes(expression) %}\n {{ return(adapter.dispatch('escape_single_quotes', 'dbt') (expression)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__escape_single_quotes"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957319, "supported_languages": null}, "macro.dbt.default__escape_single_quotes": {"name": "default__escape_single_quotes", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/escape_single_quotes.sql", "original_file_path": "macros/utils/escape_single_quotes.sql", "unique_id": "macro.dbt.default__escape_single_quotes", "macro_sql": "{% macro default__escape_single_quotes(expression) -%}\n{{ expression | replace(\"'\",\"''\") }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9573271, "supported_languages": null}, "macro.dbt.right": {"name": "right", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/right.sql", "original_file_path": "macros/utils/right.sql", "unique_id": "macro.dbt.right", "macro_sql": "{% macro right(string_text, length_expression) -%}\n {{ return(adapter.dispatch('right', 'dbt') (string_text, length_expression)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__right"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95734, "supported_languages": null}, "macro.dbt.default__right": {"name": "default__right", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/right.sql", "original_file_path": "macros/utils/right.sql", "unique_id": "macro.dbt.default__right", "macro_sql": "{% macro default__right(string_text, length_expression) %}\n\n right(\n {{ string_text }},\n {{ length_expression }}\n )\n\n{%- endmacro -%}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957345, "supported_languages": null}, "macro.dbt.listagg": {"name": "listagg", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/listagg.sql", "original_file_path": "macros/utils/listagg.sql", "unique_id": "macro.dbt.listagg", "macro_sql": "{% macro listagg(measure, delimiter_text=\"','\", order_by_clause=none, limit_num=none) -%}\n {{ return(adapter.dispatch('listagg', 'dbt') (measure, delimiter_text, order_by_clause, limit_num)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__listagg"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957356, "supported_languages": null}, "macro.dbt.default__listagg": {"name": "default__listagg", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/listagg.sql", "original_file_path": "macros/utils/listagg.sql", "unique_id": "macro.dbt.default__listagg", "macro_sql": "{% macro default__listagg(measure, delimiter_text, order_by_clause, limit_num) -%}\n\n {% if limit_num -%}\n array_to_string(\n array_slice(\n array_agg(\n {{ measure }}\n ){% if order_by_clause -%}\n within group ({{ order_by_clause }})\n {%- endif %}\n ,0\n ,{{ limit_num }}\n ),\n {{ delimiter_text }}\n )\n {%- else %}\n listagg(\n {{ measure }},\n {{ delimiter_text }}\n )\n {% if order_by_clause -%}\n within group ({{ order_by_clause }})\n {%- endif %}\n {%- endif %}\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957362, "supported_languages": null}, "macro.dbt.datediff": {"name": "datediff", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/datediff.sql", "original_file_path": "macros/utils/datediff.sql", "unique_id": "macro.dbt.datediff", "macro_sql": "{% macro datediff(first_date, second_date, datepart) %}\n {{ return(adapter.dispatch('datediff', 'dbt')(first_date, second_date, datepart)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__datediff"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957373, "supported_languages": null}, "macro.dbt.default__datediff": {"name": "default__datediff", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/datediff.sql", "original_file_path": "macros/utils/datediff.sql", "unique_id": "macro.dbt.default__datediff", "macro_sql": "{% macro default__datediff(first_date, second_date, datepart) -%}\n\n datediff(\n {{ datepart }},\n {{ first_date }},\n {{ second_date }}\n )\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957379, "supported_languages": null}, "macro.dbt.safe_cast": {"name": "safe_cast", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/safe_cast.sql", "original_file_path": "macros/utils/safe_cast.sql", "unique_id": "macro.dbt.safe_cast", "macro_sql": "{% macro safe_cast(field, type) %}\n {{ return(adapter.dispatch('safe_cast', 'dbt') (field, type)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__safe_cast"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957394, "supported_languages": null}, "macro.dbt.default__safe_cast": {"name": "default__safe_cast", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/safe_cast.sql", "original_file_path": "macros/utils/safe_cast.sql", "unique_id": "macro.dbt.default__safe_cast", "macro_sql": "{% macro default__safe_cast(field, type) %}\n {# most databases don't support this function yet\n so we just need to use cast #}\n cast({{field}} as {{type}})\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9574, "supported_languages": null}, "macro.dbt.hash": {"name": "hash", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/hash.sql", "original_file_path": "macros/utils/hash.sql", "unique_id": "macro.dbt.hash", "macro_sql": "{% macro hash(field) -%}\n {{ return(adapter.dispatch('hash', 'dbt') (field)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__hash"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957412, "supported_languages": null}, "macro.dbt.default__hash": {"name": "default__hash", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/hash.sql", "original_file_path": "macros/utils/hash.sql", "unique_id": "macro.dbt.default__hash", "macro_sql": "{% macro default__hash(field) -%}\n md5(cast({{ field }} as {{ api.Column.translate_type('string') }}))\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957418, "supported_languages": null}, "macro.dbt.cast_bool_to_text": {"name": "cast_bool_to_text", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/cast_bool_to_text.sql", "original_file_path": "macros/utils/cast_bool_to_text.sql", "unique_id": "macro.dbt.cast_bool_to_text", "macro_sql": "{% macro cast_bool_to_text(field) %}\n {{ adapter.dispatch('cast_bool_to_text', 'dbt') (field) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__cast_bool_to_text"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957431, "supported_languages": null}, "macro.dbt.default__cast_bool_to_text": {"name": "default__cast_bool_to_text", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/cast_bool_to_text.sql", "original_file_path": "macros/utils/cast_bool_to_text.sql", "unique_id": "macro.dbt.default__cast_bool_to_text", "macro_sql": "{% macro default__cast_bool_to_text(field) %}\n cast({{ field }} as {{ api.Column.translate_type('string') }})\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957438, "supported_languages": null}, "macro.dbt.cast": {"name": "cast", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/cast.sql", "original_file_path": "macros/utils/cast.sql", "unique_id": "macro.dbt.cast", "macro_sql": "{% macro cast(field, type) %}\n {{ return(adapter.dispatch('cast', 'dbt') (field, type)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__cast"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957448, "supported_languages": null}, "macro.dbt.default__cast": {"name": "default__cast", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/cast.sql", "original_file_path": "macros/utils/cast.sql", "unique_id": "macro.dbt.default__cast", "macro_sql": "{% macro default__cast(field, type) %}\n cast({{field}} as {{type}})\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9574552, "supported_languages": null}, "macro.dbt.any_value": {"name": "any_value", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/any_value.sql", "original_file_path": "macros/utils/any_value.sql", "unique_id": "macro.dbt.any_value", "macro_sql": "{% macro any_value(expression) -%}\n {{ return(adapter.dispatch('any_value', 'dbt') (expression)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__any_value"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9574652, "supported_languages": null}, "macro.dbt.default__any_value": {"name": "default__any_value", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/any_value.sql", "original_file_path": "macros/utils/any_value.sql", "unique_id": "macro.dbt.default__any_value", "macro_sql": "{% macro default__any_value(expression) -%}\n\n any_value({{ expression }})\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9574752, "supported_languages": null}, "macro.dbt.position": {"name": "position", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/position.sql", "original_file_path": "macros/utils/position.sql", "unique_id": "macro.dbt.position", "macro_sql": "{% macro position(substring_text, string_text) -%}\n {{ return(adapter.dispatch('position', 'dbt') (substring_text, string_text)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__position"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9574862, "supported_languages": null}, "macro.dbt.default__position": {"name": "default__position", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/position.sql", "original_file_path": "macros/utils/position.sql", "unique_id": "macro.dbt.default__position", "macro_sql": "{% macro default__position(substring_text, string_text) %}\n\n position(\n {{ substring_text }} in {{ string_text }}\n )\n\n{%- endmacro -%}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957491, "supported_languages": null}, "macro.dbt.string_literal": {"name": "string_literal", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/literal.sql", "original_file_path": "macros/utils/literal.sql", "unique_id": "macro.dbt.string_literal", "macro_sql": "{%- macro string_literal(value) -%}\n {{ return(adapter.dispatch('string_literal', 'dbt') (value)) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__string_literal"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957501, "supported_languages": null}, "macro.dbt.default__string_literal": {"name": "default__string_literal", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/literal.sql", "original_file_path": "macros/utils/literal.sql", "unique_id": "macro.dbt.default__string_literal", "macro_sql": "{% macro default__string_literal(value) -%}\n '{{ value }}'\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9575071, "supported_languages": null}, "macro.dbt.type_string": {"name": "type_string", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.type_string", "macro_sql": "\n\n{%- macro type_string() -%}\n {{ return(adapter.dispatch('type_string', 'dbt')()) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__type_string"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95753, "supported_languages": null}, "macro.dbt.default__type_string": {"name": "default__type_string", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.default__type_string", "macro_sql": "{% macro default__type_string() %}\n {{ return(api.Column.translate_type(\"string\")) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957543, "supported_languages": null}, "macro.dbt.type_timestamp": {"name": "type_timestamp", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.type_timestamp", "macro_sql": "\n\n{%- macro type_timestamp() -%}\n {{ return(adapter.dispatch('type_timestamp', 'dbt')()) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__type_timestamp"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957549, "supported_languages": null}, "macro.dbt.default__type_timestamp": {"name": "default__type_timestamp", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.default__type_timestamp", "macro_sql": "{% macro default__type_timestamp() %}\n {{ return(api.Column.translate_type(\"timestamp\")) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957555, "supported_languages": null}, "macro.dbt.type_float": {"name": "type_float", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.type_float", "macro_sql": "\n\n{%- macro type_float() -%}\n {{ return(adapter.dispatch('type_float', 'dbt')()) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__type_float"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957561, "supported_languages": null}, "macro.dbt.default__type_float": {"name": "default__type_float", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.default__type_float", "macro_sql": "{% macro default__type_float() %}\n {{ return(api.Column.translate_type(\"float\")) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957567, "supported_languages": null}, "macro.dbt.type_numeric": {"name": "type_numeric", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.type_numeric", "macro_sql": "\n\n{%- macro type_numeric() -%}\n {{ return(adapter.dispatch('type_numeric', 'dbt')()) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__type_numeric"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957576, "supported_languages": null}, "macro.dbt.default__type_numeric": {"name": "default__type_numeric", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.default__type_numeric", "macro_sql": "{% macro default__type_numeric() %}\n {{ return(api.Column.numeric_type(\"numeric\", 28, 6)) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957581, "supported_languages": null}, "macro.dbt.type_bigint": {"name": "type_bigint", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.type_bigint", "macro_sql": "\n\n{%- macro type_bigint() -%}\n {{ return(adapter.dispatch('type_bigint', 'dbt')()) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__type_bigint"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957587, "supported_languages": null}, "macro.dbt.default__type_bigint": {"name": "default__type_bigint", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.default__type_bigint", "macro_sql": "{% macro default__type_bigint() %}\n {{ return(api.Column.translate_type(\"bigint\")) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957592, "supported_languages": null}, "macro.dbt.type_int": {"name": "type_int", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.type_int", "macro_sql": "\n\n{%- macro type_int() -%}\n {{ return(adapter.dispatch('type_int', 'dbt')()) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__type_int"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95762, "supported_languages": null}, "macro.dbt.default__type_int": {"name": "default__type_int", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.default__type_int", "macro_sql": "{%- macro default__type_int() -%}\n {{ return(api.Column.translate_type(\"integer\")) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957628, "supported_languages": null}, "macro.dbt.type_boolean": {"name": "type_boolean", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.type_boolean", "macro_sql": "\n\n{%- macro type_boolean() -%}\n {{ return(adapter.dispatch('type_boolean', 'dbt')()) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__type_boolean"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957634, "supported_languages": null}, "macro.dbt.default__type_boolean": {"name": "default__type_boolean", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.default__type_boolean", "macro_sql": "{%- macro default__type_boolean() -%}\n {{ return(api.Column.translate_type(\"boolean\")) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957639, "supported_languages": null}, "macro.dbt.array_concat": {"name": "array_concat", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/array_concat.sql", "original_file_path": "macros/utils/array_concat.sql", "unique_id": "macro.dbt.array_concat", "macro_sql": "{% macro array_concat(array_1, array_2) -%}\n {{ return(adapter.dispatch('array_concat', 'dbt')(array_1, array_2)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__array_concat"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95765, "supported_languages": null}, "macro.dbt.default__array_concat": {"name": "default__array_concat", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/array_concat.sql", "original_file_path": "macros/utils/array_concat.sql", "unique_id": "macro.dbt.default__array_concat", "macro_sql": "{% macro default__array_concat(array_1, array_2) -%}\n array_cat({{ array_1 }}, {{ array_2 }})\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9576561, "supported_languages": null}, "macro.dbt.bool_or": {"name": "bool_or", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/bool_or.sql", "original_file_path": "macros/utils/bool_or.sql", "unique_id": "macro.dbt.bool_or", "macro_sql": "{% macro bool_or(expression) -%}\n {{ return(adapter.dispatch('bool_or', 'dbt') (expression)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__bool_or"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957667, "supported_languages": null}, "macro.dbt.default__bool_or": {"name": "default__bool_or", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/bool_or.sql", "original_file_path": "macros/utils/bool_or.sql", "unique_id": "macro.dbt.default__bool_or", "macro_sql": "{% macro default__bool_or(expression) -%}\n\n bool_or({{ expression }})\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9576719, "supported_languages": null}, "macro.dbt.last_day": {"name": "last_day", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/last_day.sql", "original_file_path": "macros/utils/last_day.sql", "unique_id": "macro.dbt.last_day", "macro_sql": "{% macro last_day(date, datepart) %}\n {{ return(adapter.dispatch('last_day', 'dbt') (date, datepart)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__last_day"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957685, "supported_languages": null}, "macro.dbt.default_last_day": {"name": "default_last_day", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/last_day.sql", "original_file_path": "macros/utils/last_day.sql", "unique_id": "macro.dbt.default_last_day", "macro_sql": "\n\n{%- macro default_last_day(date, datepart) -%}\n cast(\n {{dbt.dateadd('day', '-1',\n dbt.dateadd(datepart, '1', dbt.date_trunc(datepart, date))\n )}}\n as date)\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.dateadd", "macro.dbt.date_trunc"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9576929, "supported_languages": null}, "macro.dbt.default__last_day": {"name": "default__last_day", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/last_day.sql", "original_file_path": "macros/utils/last_day.sql", "unique_id": "macro.dbt.default__last_day", "macro_sql": "{% macro default__last_day(date, datepart) -%}\n {{dbt.default_last_day(date, datepart)}}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default_last_day"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9577, "supported_languages": null}, "macro.dbt.split_part": {"name": "split_part", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/split_part.sql", "original_file_path": "macros/utils/split_part.sql", "unique_id": "macro.dbt.split_part", "macro_sql": "{% macro split_part(string_text, delimiter_text, part_number) %}\n {{ return(adapter.dispatch('split_part', 'dbt') (string_text, delimiter_text, part_number)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__split_part"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957712, "supported_languages": null}, "macro.dbt.default__split_part": {"name": "default__split_part", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/split_part.sql", "original_file_path": "macros/utils/split_part.sql", "unique_id": "macro.dbt.default__split_part", "macro_sql": "{% macro default__split_part(string_text, delimiter_text, part_number) %}\n\n split_part(\n {{ string_text }},\n {{ delimiter_text }},\n {{ part_number }}\n )\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9577181, "supported_languages": null}, "macro.dbt._split_part_negative": {"name": "_split_part_negative", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/split_part.sql", "original_file_path": "macros/utils/split_part.sql", "unique_id": "macro.dbt._split_part_negative", "macro_sql": "{% macro _split_part_negative(string_text, delimiter_text, part_number) %}\n\n split_part(\n {{ string_text }},\n {{ delimiter_text }},\n length({{ string_text }})\n - length(\n replace({{ string_text }}, {{ delimiter_text }}, '')\n ) + 2 + {{ part_number }}\n )\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9577239, "supported_languages": null}, "macro.dbt.date_trunc": {"name": "date_trunc", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/date_trunc.sql", "original_file_path": "macros/utils/date_trunc.sql", "unique_id": "macro.dbt.date_trunc", "macro_sql": "{% macro date_trunc(datepart, date) -%}\n {{ return(adapter.dispatch('date_trunc', 'dbt') (datepart, date)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__date_trunc"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957737, "supported_languages": null}, "macro.dbt.default__date_trunc": {"name": "default__date_trunc", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/date_trunc.sql", "original_file_path": "macros/utils/date_trunc.sql", "unique_id": "macro.dbt.default__date_trunc", "macro_sql": "{% macro default__date_trunc(datepart, date) -%}\n date_trunc('{{datepart}}', {{date}})\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9577441, "supported_languages": null}, "macro.dbt.array_construct": {"name": "array_construct", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/array_construct.sql", "original_file_path": "macros/utils/array_construct.sql", "unique_id": "macro.dbt.array_construct", "macro_sql": "{% macro array_construct(inputs=[], data_type=api.Column.translate_type('integer')) -%}\n {{ return(adapter.dispatch('array_construct', 'dbt')(inputs, data_type)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__array_construct"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9577558, "supported_languages": null}, "macro.dbt.default__array_construct": {"name": "default__array_construct", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/array_construct.sql", "original_file_path": "macros/utils/array_construct.sql", "unique_id": "macro.dbt.default__array_construct", "macro_sql": "{% macro default__array_construct(inputs, data_type) -%}\n {% if inputs|length > 0 %}\n array[ {{ inputs|join(' , ') }} ]\n {% else %}\n array[]::{{data_type}}[]\n {% endif %}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957767, "supported_languages": null}, "macro.dbt.array_append": {"name": "array_append", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/array_append.sql", "original_file_path": "macros/utils/array_append.sql", "unique_id": "macro.dbt.array_append", "macro_sql": "{% macro array_append(array, new_element) -%}\n {{ return(adapter.dispatch('array_append', 'dbt')(array, new_element)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__array_append"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95778, "supported_languages": null}, "macro.dbt.default__array_append": {"name": "default__array_append", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/array_append.sql", "original_file_path": "macros/utils/array_append.sql", "unique_id": "macro.dbt.default__array_append", "macro_sql": "{% macro default__array_append(array, new_element) -%}\n array_append({{ array }}, {{ new_element }})\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957788, "supported_languages": null}, "macro.dbt.create_schema": {"name": "create_schema", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/schema.sql", "original_file_path": "macros/adapters/schema.sql", "unique_id": "macro.dbt.create_schema", "macro_sql": "{% macro create_schema(relation) -%}\n {{ adapter.dispatch('create_schema', 'dbt')(relation) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__create_schema"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957799, "supported_languages": null}, "macro.dbt.default__create_schema": {"name": "default__create_schema", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/schema.sql", "original_file_path": "macros/adapters/schema.sql", "unique_id": "macro.dbt.default__create_schema", "macro_sql": "{% macro default__create_schema(relation) -%}\n {%- call statement('create_schema') -%}\n create schema if not exists {{ relation.without_identifier() }}\n {% endcall %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957807, "supported_languages": null}, "macro.dbt.drop_schema": {"name": "drop_schema", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/schema.sql", "original_file_path": "macros/adapters/schema.sql", "unique_id": "macro.dbt.drop_schema", "macro_sql": "{% macro drop_schema(relation) -%}\n {{ adapter.dispatch('drop_schema', 'dbt')(relation) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__drop_schema"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957813, "supported_languages": null}, "macro.dbt.default__drop_schema": {"name": "default__drop_schema", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/schema.sql", "original_file_path": "macros/adapters/schema.sql", "unique_id": "macro.dbt.default__drop_schema", "macro_sql": "{% macro default__drop_schema(relation) -%}\n {%- call statement('drop_schema') -%}\n drop schema if exists {{ relation.without_identifier() }} cascade\n {% endcall %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957818, "supported_languages": null}, "macro.dbt.current_timestamp": {"name": "current_timestamp", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/timestamps.sql", "original_file_path": "macros/adapters/timestamps.sql", "unique_id": "macro.dbt.current_timestamp", "macro_sql": "{%- macro current_timestamp() -%}\n {{ adapter.dispatch('current_timestamp', 'dbt')() }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt_postgres.postgres__current_timestamp"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9578319, "supported_languages": null}, "macro.dbt.default__current_timestamp": {"name": "default__current_timestamp", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/timestamps.sql", "original_file_path": "macros/adapters/timestamps.sql", "unique_id": "macro.dbt.default__current_timestamp", "macro_sql": "{% macro default__current_timestamp() -%}\n {{ exceptions.raise_not_implemented(\n 'current_timestamp macro not implemented for adapter ' + adapter.type()) }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957838, "supported_languages": null}, "macro.dbt.snapshot_get_time": {"name": "snapshot_get_time", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/timestamps.sql", "original_file_path": "macros/adapters/timestamps.sql", "unique_id": "macro.dbt.snapshot_get_time", "macro_sql": "\n\n{%- macro snapshot_get_time() -%}\n {{ adapter.dispatch('snapshot_get_time', 'dbt')() }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt_postgres.postgres__snapshot_get_time"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957844, "supported_languages": null}, "macro.dbt.default__snapshot_get_time": {"name": "default__snapshot_get_time", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/timestamps.sql", "original_file_path": "macros/adapters/timestamps.sql", "unique_id": "macro.dbt.default__snapshot_get_time", "macro_sql": "{% macro default__snapshot_get_time() %}\n {{ current_timestamp() }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.current_timestamp"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957852, "supported_languages": null}, "macro.dbt.current_timestamp_backcompat": {"name": "current_timestamp_backcompat", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/timestamps.sql", "original_file_path": "macros/adapters/timestamps.sql", "unique_id": "macro.dbt.current_timestamp_backcompat", "macro_sql": "{% macro current_timestamp_backcompat() %}\n {{ return(adapter.dispatch('current_timestamp_backcompat', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__current_timestamp_backcompat"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9578571, "supported_languages": null}, "macro.dbt.default__current_timestamp_backcompat": {"name": "default__current_timestamp_backcompat", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/timestamps.sql", "original_file_path": "macros/adapters/timestamps.sql", "unique_id": "macro.dbt.default__current_timestamp_backcompat", "macro_sql": "{% macro default__current_timestamp_backcompat() %}\n current_timestamp::timestamp\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957862, "supported_languages": null}, "macro.dbt.current_timestamp_in_utc_backcompat": {"name": "current_timestamp_in_utc_backcompat", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/timestamps.sql", "original_file_path": "macros/adapters/timestamps.sql", "unique_id": "macro.dbt.current_timestamp_in_utc_backcompat", "macro_sql": "{% macro current_timestamp_in_utc_backcompat() %}\n {{ return(adapter.dispatch('current_timestamp_in_utc_backcompat', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__current_timestamp_in_utc_backcompat"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957868, "supported_languages": null}, "macro.dbt.default__current_timestamp_in_utc_backcompat": {"name": "default__current_timestamp_in_utc_backcompat", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/timestamps.sql", "original_file_path": "macros/adapters/timestamps.sql", "unique_id": "macro.dbt.default__current_timestamp_in_utc_backcompat", "macro_sql": "{% macro default__current_timestamp_in_utc_backcompat() %}\n {{ return(adapter.dispatch('current_timestamp_backcompat', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.current_timestamp_backcompat", "macro.dbt_postgres.postgres__current_timestamp_backcompat"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9578738, "supported_languages": null}, "macro.dbt.get_create_index_sql": {"name": "get_create_index_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/indexes.sql", "original_file_path": "macros/adapters/indexes.sql", "unique_id": "macro.dbt.get_create_index_sql", "macro_sql": "{% macro get_create_index_sql(relation, index_dict) -%}\n {{ return(adapter.dispatch('get_create_index_sql', 'dbt')(relation, index_dict)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_create_index_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957889, "supported_languages": null}, "macro.dbt.default__get_create_index_sql": {"name": "default__get_create_index_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/indexes.sql", "original_file_path": "macros/adapters/indexes.sql", "unique_id": "macro.dbt.default__get_create_index_sql", "macro_sql": "{% macro default__get_create_index_sql(relation, index_dict) -%}\n {% do return(None) %}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957897, "supported_languages": null}, "macro.dbt.create_indexes": {"name": "create_indexes", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/indexes.sql", "original_file_path": "macros/adapters/indexes.sql", "unique_id": "macro.dbt.create_indexes", "macro_sql": "{% macro create_indexes(relation) -%}\n {{ adapter.dispatch('create_indexes', 'dbt')(relation) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__create_indexes"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9579039, "supported_languages": null}, "macro.dbt.default__create_indexes": {"name": "default__create_indexes", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/indexes.sql", "original_file_path": "macros/adapters/indexes.sql", "unique_id": "macro.dbt.default__create_indexes", "macro_sql": "{% macro default__create_indexes(relation) -%}\n {%- set _indexes = config.get('indexes', default=[]) -%}\n\n {% for _index_dict in _indexes %}\n {% set create_index_sql = get_create_index_sql(relation, _index_dict) %}\n {% if create_index_sql %}\n {% do run_query(create_index_sql) %}\n {% endif %}\n {% endfor %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_create_index_sql", "macro.dbt.run_query"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95791, "supported_languages": null}, "macro.dbt.get_drop_index_sql": {"name": "get_drop_index_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/indexes.sql", "original_file_path": "macros/adapters/indexes.sql", "unique_id": "macro.dbt.get_drop_index_sql", "macro_sql": "{% macro get_drop_index_sql(relation, index_name) -%}\n {{ adapter.dispatch('get_drop_index_sql', 'dbt')(relation, index_name) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_drop_index_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957916, "supported_languages": null}, "macro.dbt.default__get_drop_index_sql": {"name": "default__get_drop_index_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/indexes.sql", "original_file_path": "macros/adapters/indexes.sql", "unique_id": "macro.dbt.default__get_drop_index_sql", "macro_sql": "{% macro default__get_drop_index_sql(relation, index_name) -%}\n {{ exceptions.raise_compiler_error(\"`get_drop_index_sql has not been implemented for this adapter.\") }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957922, "supported_languages": null}, "macro.dbt.get_show_indexes_sql": {"name": "get_show_indexes_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/indexes.sql", "original_file_path": "macros/adapters/indexes.sql", "unique_id": "macro.dbt.get_show_indexes_sql", "macro_sql": "{% macro get_show_indexes_sql(relation) -%}\n {{ adapter.dispatch('get_show_indexes_sql', 'dbt')(relation) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_show_indexes_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957928, "supported_languages": null}, "macro.dbt.default__get_show_indexes_sql": {"name": "default__get_show_indexes_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/indexes.sql", "original_file_path": "macros/adapters/indexes.sql", "unique_id": "macro.dbt.default__get_show_indexes_sql", "macro_sql": "{% macro default__get_show_indexes_sql(relation) -%}\n {{ exceptions.raise_compiler_error(\"`get_show_indexes_sql has not been implemented for this adapter.\") }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957933, "supported_languages": null}, "macro.dbt.make_intermediate_relation": {"name": "make_intermediate_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.make_intermediate_relation", "macro_sql": "{% macro make_intermediate_relation(base_relation, suffix='__dbt_tmp') %}\n {{ return(adapter.dispatch('make_intermediate_relation', 'dbt')(base_relation, suffix)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__make_intermediate_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957949, "supported_languages": null}, "macro.dbt.default__make_intermediate_relation": {"name": "default__make_intermediate_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.default__make_intermediate_relation", "macro_sql": "{% macro default__make_intermediate_relation(base_relation, suffix) %}\n {{ return(default__make_temp_relation(base_relation, suffix)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__make_temp_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9579551, "supported_languages": null}, "macro.dbt.make_temp_relation": {"name": "make_temp_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.make_temp_relation", "macro_sql": "{% macro make_temp_relation(base_relation, suffix='__dbt_tmp') %}\n {{ return(adapter.dispatch('make_temp_relation', 'dbt')(base_relation, suffix)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__make_temp_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9579608, "supported_languages": null}, "macro.dbt.default__make_temp_relation": {"name": "default__make_temp_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.default__make_temp_relation", "macro_sql": "{% macro default__make_temp_relation(base_relation, suffix) %}\n {%- set temp_identifier = base_relation.identifier ~ suffix -%}\n {%- set temp_relation = base_relation.incorporate(\n path={\"identifier\": temp_identifier}) -%}\n\n {{ return(temp_relation) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9579709, "supported_languages": null}, "macro.dbt.make_backup_relation": {"name": "make_backup_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.make_backup_relation", "macro_sql": "{% macro make_backup_relation(base_relation, backup_relation_type, suffix='__dbt_backup') %}\n {{ return(adapter.dispatch('make_backup_relation', 'dbt')(base_relation, backup_relation_type, suffix)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__make_backup_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957977, "supported_languages": null}, "macro.dbt.default__make_backup_relation": {"name": "default__make_backup_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.default__make_backup_relation", "macro_sql": "{% macro default__make_backup_relation(base_relation, backup_relation_type, suffix) %}\n {%- set backup_identifier = base_relation.identifier ~ suffix -%}\n {%- set backup_relation = base_relation.incorporate(\n path={\"identifier\": backup_identifier},\n type=backup_relation_type\n ) -%}\n {{ return(backup_relation) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957982, "supported_languages": null}, "macro.dbt.truncate_relation": {"name": "truncate_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.truncate_relation", "macro_sql": "{% macro truncate_relation(relation) -%}\n {{ return(adapter.dispatch('truncate_relation', 'dbt')(relation)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__truncate_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9579918, "supported_languages": null}, "macro.dbt.default__truncate_relation": {"name": "default__truncate_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.default__truncate_relation", "macro_sql": "{% macro default__truncate_relation(relation) -%}\n {% call statement('truncate_relation') -%}\n truncate table {{ relation }}\n {%- endcall %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.957998, "supported_languages": null}, "macro.dbt.get_or_create_relation": {"name": "get_or_create_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.get_or_create_relation", "macro_sql": "{% macro get_or_create_relation(database, schema, identifier, type) -%}\n {{ return(adapter.dispatch('get_or_create_relation', 'dbt')(database, schema, identifier, type)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_or_create_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958005, "supported_languages": null}, "macro.dbt.default__get_or_create_relation": {"name": "default__get_or_create_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.default__get_or_create_relation", "macro_sql": "{% macro default__get_or_create_relation(database, schema, identifier, type) %}\n {%- set target_relation = adapter.get_relation(database=database, schema=schema, identifier=identifier) %}\n\n {% if target_relation %}\n {% do return([true, target_relation]) %}\n {% endif %}\n\n {%- set new_relation = api.Relation.create(\n database=database,\n schema=schema,\n identifier=identifier,\n type=type\n ) -%}\n {% do return([false, new_relation]) %}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9580119, "supported_languages": null}, "macro.dbt.load_cached_relation": {"name": "load_cached_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.load_cached_relation", "macro_sql": "{% macro load_cached_relation(relation) %}\n {% do return(adapter.get_relation(\n database=relation.database,\n schema=relation.schema,\n identifier=relation.identifier\n )) -%}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958019, "supported_languages": null}, "macro.dbt.load_relation": {"name": "load_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.load_relation", "macro_sql": "{% macro load_relation(relation) %}\n {{ return(load_cached_relation(relation)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.load_cached_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958025, "supported_languages": null}, "macro.dbt.collect_freshness": {"name": "collect_freshness", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/freshness.sql", "original_file_path": "macros/adapters/freshness.sql", "unique_id": "macro.dbt.collect_freshness", "macro_sql": "{% macro collect_freshness(source, loaded_at_field, filter) %}\n {{ return(adapter.dispatch('collect_freshness', 'dbt')(source, loaded_at_field, filter))}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__collect_freshness"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958039, "supported_languages": null}, "macro.dbt.default__collect_freshness": {"name": "default__collect_freshness", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/freshness.sql", "original_file_path": "macros/adapters/freshness.sql", "unique_id": "macro.dbt.default__collect_freshness", "macro_sql": "{% macro default__collect_freshness(source, loaded_at_field, filter) %}\n {% call statement('collect_freshness', fetch_result=True, auto_begin=False) -%}\n select\n max({{ loaded_at_field }}) as max_loaded_at,\n {{ current_timestamp() }} as snapshotted_at\n from {{ source }}\n {% if filter %}\n where {{ filter }}\n {% endif %}\n {% endcall %}\n {{ return(load_result('collect_freshness')) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement", "macro.dbt.current_timestamp"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958045, "supported_languages": null}, "macro.dbt.validate_sql": {"name": "validate_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/validate_sql.sql", "original_file_path": "macros/adapters/validate_sql.sql", "unique_id": "macro.dbt.validate_sql", "macro_sql": "{% macro validate_sql(sql) -%}\n {{ return(adapter.dispatch('validate_sql', 'dbt')(sql)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__validate_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958066, "supported_languages": null}, "macro.dbt.default__validate_sql": {"name": "default__validate_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/validate_sql.sql", "original_file_path": "macros/adapters/validate_sql.sql", "unique_id": "macro.dbt.default__validate_sql", "macro_sql": "{% macro default__validate_sql(sql) -%}\n {% call statement('validate_sql') -%}\n explain {{ sql }}\n {% endcall %}\n {{ return(load_result('validate_sql')) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958072, "supported_languages": null}, "macro.dbt.copy_grants": {"name": "copy_grants", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.copy_grants", "macro_sql": "{% macro copy_grants() %}\n {{ return(adapter.dispatch('copy_grants', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__copy_grants"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95809, "supported_languages": null}, "macro.dbt.default__copy_grants": {"name": "default__copy_grants", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.default__copy_grants", "macro_sql": "{% macro default__copy_grants() %}\n {{ return(True) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958095, "supported_languages": null}, "macro.dbt.support_multiple_grantees_per_dcl_statement": {"name": "support_multiple_grantees_per_dcl_statement", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.support_multiple_grantees_per_dcl_statement", "macro_sql": "{% macro support_multiple_grantees_per_dcl_statement() %}\n {{ return(adapter.dispatch('support_multiple_grantees_per_dcl_statement', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__support_multiple_grantees_per_dcl_statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958102, "supported_languages": null}, "macro.dbt.default__support_multiple_grantees_per_dcl_statement": {"name": "default__support_multiple_grantees_per_dcl_statement", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.default__support_multiple_grantees_per_dcl_statement", "macro_sql": "\n\n{%- macro default__support_multiple_grantees_per_dcl_statement() -%}\n {{ return(True) }}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958108, "supported_languages": null}, "macro.dbt.should_revoke": {"name": "should_revoke", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.should_revoke", "macro_sql": "{% macro should_revoke(existing_relation, full_refresh_mode=True) %}\n\n {% if not existing_relation %}\n {#-- The table doesn't already exist, so no grants to copy over --#}\n {{ return(False) }}\n {% elif full_refresh_mode %}\n {#-- The object is being REPLACED -- whether grants are copied over depends on the value of user config --#}\n {{ return(copy_grants()) }}\n {% else %}\n {#-- The table is being merged/upserted/inserted -- grants will be carried over --#}\n {{ return(True) }}\n {% endif %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.copy_grants"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958116, "supported_languages": null}, "macro.dbt.get_show_grant_sql": {"name": "get_show_grant_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.get_show_grant_sql", "macro_sql": "{% macro get_show_grant_sql(relation) %}\n {{ return(adapter.dispatch(\"get_show_grant_sql\", \"dbt\")(relation)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_show_grant_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958121, "supported_languages": null}, "macro.dbt.default__get_show_grant_sql": {"name": "default__get_show_grant_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.default__get_show_grant_sql", "macro_sql": "{% macro default__get_show_grant_sql(relation) %}\n show grants on {{ relation }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958127, "supported_languages": null}, "macro.dbt.get_grant_sql": {"name": "get_grant_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.get_grant_sql", "macro_sql": "{% macro get_grant_sql(relation, privilege, grantees) %}\n {{ return(adapter.dispatch('get_grant_sql', 'dbt')(relation, privilege, grantees)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_grant_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958132, "supported_languages": null}, "macro.dbt.default__get_grant_sql": {"name": "default__get_grant_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.default__get_grant_sql", "macro_sql": "\n\n{%- macro default__get_grant_sql(relation, privilege, grantees) -%}\n grant {{ privilege }} on {{ relation }} to {{ grantees | join(', ') }}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9581382, "supported_languages": null}, "macro.dbt.get_revoke_sql": {"name": "get_revoke_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.get_revoke_sql", "macro_sql": "{% macro get_revoke_sql(relation, privilege, grantees) %}\n {{ return(adapter.dispatch('get_revoke_sql', 'dbt')(relation, privilege, grantees)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_revoke_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958146, "supported_languages": null}, "macro.dbt.default__get_revoke_sql": {"name": "default__get_revoke_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.default__get_revoke_sql", "macro_sql": "\n\n{%- macro default__get_revoke_sql(relation, privilege, grantees) -%}\n revoke {{ privilege }} on {{ relation }} from {{ grantees | join(', ') }}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958153, "supported_languages": null}, "macro.dbt.get_dcl_statement_list": {"name": "get_dcl_statement_list", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.get_dcl_statement_list", "macro_sql": "{% macro get_dcl_statement_list(relation, grant_config, get_dcl_macro) %}\n {{ return(adapter.dispatch('get_dcl_statement_list', 'dbt')(relation, grant_config, get_dcl_macro)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_dcl_statement_list"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9581592, "supported_languages": null}, "macro.dbt.default__get_dcl_statement_list": {"name": "default__get_dcl_statement_list", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.default__get_dcl_statement_list", "macro_sql": "\n\n{%- macro default__get_dcl_statement_list(relation, grant_config, get_dcl_macro) -%}\n {#\n -- Unpack grant_config into specific privileges and the set of users who need them granted/revoked.\n -- Depending on whether this database supports multiple grantees per statement, pass in the list of\n -- all grantees per privilege, or (if not) template one statement per privilege-grantee pair.\n -- `get_dcl_macro` will be either `get_grant_sql` or `get_revoke_sql`\n #}\n {%- set dcl_statements = [] -%}\n {%- for privilege, grantees in grant_config.items() %}\n {%- if support_multiple_grantees_per_dcl_statement() and grantees -%}\n {%- set dcl = get_dcl_macro(relation, privilege, grantees) -%}\n {%- do dcl_statements.append(dcl) -%}\n {%- else -%}\n {%- for grantee in grantees -%}\n {% set dcl = get_dcl_macro(relation, privilege, [grantee]) %}\n {%- do dcl_statements.append(dcl) -%}\n {% endfor -%}\n {%- endif -%}\n {%- endfor -%}\n {{ return(dcl_statements) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.support_multiple_grantees_per_dcl_statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958167, "supported_languages": null}, "macro.dbt.call_dcl_statements": {"name": "call_dcl_statements", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.call_dcl_statements", "macro_sql": "{% macro call_dcl_statements(dcl_statement_list) %}\n {{ return(adapter.dispatch(\"call_dcl_statements\", \"dbt\")(dcl_statement_list)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__call_dcl_statements"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9581718, "supported_languages": null}, "macro.dbt.default__call_dcl_statements": {"name": "default__call_dcl_statements", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.default__call_dcl_statements", "macro_sql": "{% macro default__call_dcl_statements(dcl_statement_list) %}\n {#\n -- By default, supply all grant + revoke statements in a single semicolon-separated block,\n -- so that they're all processed together.\n\n -- Some databases do not support this. Those adapters will need to override this macro\n -- to run each statement individually.\n #}\n {% call statement('grants') %}\n {% for dcl_statement in dcl_statement_list %}\n {{ dcl_statement }};\n {% endfor %}\n {% endcall %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958198, "supported_languages": null}, "macro.dbt.apply_grants": {"name": "apply_grants", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.apply_grants", "macro_sql": "{% macro apply_grants(relation, grant_config, should_revoke) %}\n {{ return(adapter.dispatch(\"apply_grants\", \"dbt\")(relation, grant_config, should_revoke)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__apply_grants"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9582038, "supported_languages": null}, "macro.dbt.default__apply_grants": {"name": "default__apply_grants", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.default__apply_grants", "macro_sql": "{% macro default__apply_grants(relation, grant_config, should_revoke=True) %}\n {#-- If grant_config is {} or None, this is a no-op --#}\n {% if grant_config %}\n {% if should_revoke %}\n {#-- We think previous grants may have carried over --#}\n {#-- Show current grants and calculate diffs --#}\n {% set current_grants_table = run_query(get_show_grant_sql(relation)) %}\n {% set current_grants_dict = adapter.standardize_grants_dict(current_grants_table) %}\n {% set needs_granting = diff_of_two_dicts(grant_config, current_grants_dict) %}\n {% set needs_revoking = diff_of_two_dicts(current_grants_dict, grant_config) %}\n {% if not (needs_granting or needs_revoking) %}\n {{ log('On ' ~ relation ~': All grants are in place, no revocation or granting needed.')}}\n {% endif %}\n {% else %}\n {#-- We don't think there's any chance of previous grants having carried over. --#}\n {#-- Jump straight to granting what the user has configured. --#}\n {% set needs_revoking = {} %}\n {% set needs_granting = grant_config %}\n {% endif %}\n {% if needs_granting or needs_revoking %}\n {% set revoke_statement_list = get_dcl_statement_list(relation, needs_revoking, get_revoke_sql) %}\n {% set grant_statement_list = get_dcl_statement_list(relation, needs_granting, get_grant_sql) %}\n {% set dcl_statement_list = revoke_statement_list + grant_statement_list %}\n {% if dcl_statement_list %}\n {{ call_dcl_statements(dcl_statement_list) }}\n {% endif %}\n {% endif %}\n {% endif %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.run_query", "macro.dbt.get_show_grant_sql", "macro.dbt.get_dcl_statement_list", "macro.dbt.call_dcl_statements"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9582121, "supported_languages": null}, "macro.dbt.get_show_sql": {"name": "get_show_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/show.sql", "original_file_path": "macros/adapters/show.sql", "unique_id": "macro.dbt.get_show_sql", "macro_sql": "{% macro get_show_sql(compiled_code, sql_header, limit) -%}\n {%- if sql_header -%}\n {{ sql_header }}\n {%- endif -%}\n {%- if limit is not none -%}\n {{ get_limit_subquery_sql(compiled_code, limit) }}\n {%- else -%}\n {{ compiled_code }}\n {%- endif -%}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_limit_subquery_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958223, "supported_languages": null}, "macro.dbt.get_limit_subquery_sql": {"name": "get_limit_subquery_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/show.sql", "original_file_path": "macros/adapters/show.sql", "unique_id": "macro.dbt.get_limit_subquery_sql", "macro_sql": "{% macro get_limit_subquery_sql(sql, limit) %}\n {{ adapter.dispatch('get_limit_subquery_sql', 'dbt')(sql, limit) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_limit_subquery_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9582322, "supported_languages": null}, "macro.dbt.default__get_limit_subquery_sql": {"name": "default__get_limit_subquery_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/show.sql", "original_file_path": "macros/adapters/show.sql", "unique_id": "macro.dbt.default__get_limit_subquery_sql", "macro_sql": "{% macro default__get_limit_subquery_sql(sql, limit) %}\n select *\n from (\n {{ sql }}\n ) as model_limit_subq\n limit {{ limit }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958237, "supported_languages": null}, "macro.dbt.alter_column_comment": {"name": "alter_column_comment", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/persist_docs.sql", "original_file_path": "macros/adapters/persist_docs.sql", "unique_id": "macro.dbt.alter_column_comment", "macro_sql": "{% macro alter_column_comment(relation, column_dict) -%}\n {{ return(adapter.dispatch('alter_column_comment', 'dbt')(relation, column_dict)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__alter_column_comment"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9582489, "supported_languages": null}, "macro.dbt.default__alter_column_comment": {"name": "default__alter_column_comment", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/persist_docs.sql", "original_file_path": "macros/adapters/persist_docs.sql", "unique_id": "macro.dbt.default__alter_column_comment", "macro_sql": "{% macro default__alter_column_comment(relation, column_dict) -%}\n {{ exceptions.raise_not_implemented(\n 'alter_column_comment macro not implemented for adapter '+adapter.type()) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958255, "supported_languages": null}, "macro.dbt.alter_relation_comment": {"name": "alter_relation_comment", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/persist_docs.sql", "original_file_path": "macros/adapters/persist_docs.sql", "unique_id": "macro.dbt.alter_relation_comment", "macro_sql": "{% macro alter_relation_comment(relation, relation_comment) -%}\n {{ return(adapter.dispatch('alter_relation_comment', 'dbt')(relation, relation_comment)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__alter_relation_comment"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958264, "supported_languages": null}, "macro.dbt.default__alter_relation_comment": {"name": "default__alter_relation_comment", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/persist_docs.sql", "original_file_path": "macros/adapters/persist_docs.sql", "unique_id": "macro.dbt.default__alter_relation_comment", "macro_sql": "{% macro default__alter_relation_comment(relation, relation_comment) -%}\n {{ exceptions.raise_not_implemented(\n 'alter_relation_comment macro not implemented for adapter '+adapter.type()) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958269, "supported_languages": null}, "macro.dbt.persist_docs": {"name": "persist_docs", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/persist_docs.sql", "original_file_path": "macros/adapters/persist_docs.sql", "unique_id": "macro.dbt.persist_docs", "macro_sql": "{% macro persist_docs(relation, model, for_relation=true, for_columns=true) -%}\n {{ return(adapter.dispatch('persist_docs', 'dbt')(relation, model, for_relation, for_columns)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__persist_docs"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9582741, "supported_languages": null}, "macro.dbt.default__persist_docs": {"name": "default__persist_docs", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/persist_docs.sql", "original_file_path": "macros/adapters/persist_docs.sql", "unique_id": "macro.dbt.default__persist_docs", "macro_sql": "{% macro default__persist_docs(relation, model, for_relation, for_columns) -%}\n {% if for_relation and config.persist_relation_docs() and model.description %}\n {% do run_query(alter_relation_comment(relation, model.description)) %}\n {% endif %}\n\n {% if for_columns and config.persist_column_docs() and model.columns %}\n {% do run_query(alter_column_comment(relation, model.columns)) %}\n {% endif %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.run_query", "macro.dbt.alter_relation_comment", "macro.dbt.alter_column_comment"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958279, "supported_languages": null}, "macro.dbt.get_catalog_relations": {"name": "get_catalog_relations", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.get_catalog_relations", "macro_sql": "{% macro get_catalog_relations(information_schema, relations) -%}\n {{ return(adapter.dispatch('get_catalog_relations', 'dbt')(information_schema, relations)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_catalog_relations"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9582999, "supported_languages": null}, "macro.dbt.default__get_catalog_relations": {"name": "default__get_catalog_relations", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.default__get_catalog_relations", "macro_sql": "{% macro default__get_catalog_relations(information_schema, relations) -%}\n {% set typename = adapter.type() %}\n {% set msg -%}\n get_catalog_relations not implemented for {{ typename }}\n {%- endset %}\n\n {{ exceptions.raise_compiler_error(msg) }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958306, "supported_languages": null}, "macro.dbt.get_catalog": {"name": "get_catalog", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.get_catalog", "macro_sql": "{% macro get_catalog(information_schema, schemas) -%}\n {{ return(adapter.dispatch('get_catalog', 'dbt')(information_schema, schemas)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_catalog"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958313, "supported_languages": null}, "macro.dbt.default__get_catalog": {"name": "default__get_catalog", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.default__get_catalog", "macro_sql": "{% macro default__get_catalog(information_schema, schemas) -%}\n\n {% set typename = adapter.type() %}\n {% set msg -%}\n get_catalog not implemented for {{ typename }}\n {%- endset %}\n\n {{ exceptions.raise_compiler_error(msg) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9583218, "supported_languages": null}, "macro.dbt.information_schema_name": {"name": "information_schema_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.information_schema_name", "macro_sql": "{% macro information_schema_name(database) %}\n {{ return(adapter.dispatch('information_schema_name', 'dbt')(database)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__information_schema_name"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958328, "supported_languages": null}, "macro.dbt.default__information_schema_name": {"name": "default__information_schema_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.default__information_schema_name", "macro_sql": "{% macro default__information_schema_name(database) -%}\n {%- if database -%}\n {{ database }}.INFORMATION_SCHEMA\n {%- else -%}\n INFORMATION_SCHEMA\n {%- endif -%}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958336, "supported_languages": null}, "macro.dbt.list_schemas": {"name": "list_schemas", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.list_schemas", "macro_sql": "{% macro list_schemas(database) -%}\n {{ return(adapter.dispatch('list_schemas', 'dbt')(database)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__list_schemas"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958343, "supported_languages": null}, "macro.dbt.default__list_schemas": {"name": "default__list_schemas", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.default__list_schemas", "macro_sql": "{% macro default__list_schemas(database) -%}\n {% set sql %}\n select distinct schema_name\n from {{ information_schema_name(database) }}.SCHEMATA\n where catalog_name ilike '{{ database }}'\n {% endset %}\n {{ return(run_query(sql)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.information_schema_name", "macro.dbt.run_query"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95836, "supported_languages": null}, "macro.dbt.check_schema_exists": {"name": "check_schema_exists", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.check_schema_exists", "macro_sql": "{% macro check_schema_exists(information_schema, schema) -%}\n {{ return(adapter.dispatch('check_schema_exists', 'dbt')(information_schema, schema)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__check_schema_exists"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958365, "supported_languages": null}, "macro.dbt.default__check_schema_exists": {"name": "default__check_schema_exists", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.default__check_schema_exists", "macro_sql": "{% macro default__check_schema_exists(information_schema, schema) -%}\n {% set sql -%}\n select count(*)\n from {{ information_schema.replace(information_schema_view='SCHEMATA') }}\n where catalog_name='{{ information_schema.database }}'\n and schema_name='{{ schema }}'\n {%- endset %}\n {{ return(run_query(sql)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.replace", "macro.dbt.run_query"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95837, "supported_languages": null}, "macro.dbt.list_relations_without_caching": {"name": "list_relations_without_caching", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.list_relations_without_caching", "macro_sql": "{% macro list_relations_without_caching(schema_relation) %}\n {{ return(adapter.dispatch('list_relations_without_caching', 'dbt')(schema_relation)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__list_relations_without_caching"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958377, "supported_languages": null}, "macro.dbt.default__list_relations_without_caching": {"name": "default__list_relations_without_caching", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.default__list_relations_without_caching", "macro_sql": "{% macro default__list_relations_without_caching(schema_relation) %}\n {{ exceptions.raise_not_implemented(\n 'list_relations_without_caching macro not implemented for adapter '+adapter.type()) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9583821, "supported_languages": null}, "macro.dbt.get_relations": {"name": "get_relations", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.get_relations", "macro_sql": "{% macro get_relations() %}\n {{ return(adapter.dispatch('get_relations', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_relations"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95839, "supported_languages": null}, "macro.dbt.default__get_relations": {"name": "default__get_relations", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.default__get_relations", "macro_sql": "{% macro default__get_relations() %}\n {{ exceptions.raise_not_implemented(\n 'get_relations macro not implemented for adapter '+adapter.type()) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958397, "supported_languages": null}, "macro.dbt.get_relation_last_modified": {"name": "get_relation_last_modified", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.get_relation_last_modified", "macro_sql": "{% macro get_relation_last_modified(information_schema, relations) %}\n {{ return(adapter.dispatch('get_relation_last_modified', 'dbt')(information_schema, relations)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_relation_last_modified"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9584022, "supported_languages": null}, "macro.dbt.default__get_relation_last_modified": {"name": "default__get_relation_last_modified", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.default__get_relation_last_modified", "macro_sql": "{% macro default__get_relation_last_modified(information_schema, relations) %}\n {{ exceptions.raise_not_implemented(\n 'get_relation_last_modified macro not implemented for adapter ' + adapter.type()) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958407, "supported_languages": null}, "macro.dbt.get_columns_in_relation": {"name": "get_columns_in_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.get_columns_in_relation", "macro_sql": "{% macro get_columns_in_relation(relation) -%}\n {{ return(adapter.dispatch('get_columns_in_relation', 'dbt')(relation)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_columns_in_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958425, "supported_languages": null}, "macro.dbt.default__get_columns_in_relation": {"name": "default__get_columns_in_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.default__get_columns_in_relation", "macro_sql": "{% macro default__get_columns_in_relation(relation) -%}\n {{ exceptions.raise_not_implemented(\n 'get_columns_in_relation macro not implemented for adapter '+adapter.type()) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95843, "supported_languages": null}, "macro.dbt.sql_convert_columns_in_relation": {"name": "sql_convert_columns_in_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.sql_convert_columns_in_relation", "macro_sql": "{% macro sql_convert_columns_in_relation(table) -%}\n {% set columns = [] %}\n {% for row in table %}\n {% do columns.append(api.Column(*row)) %}\n {% endfor %}\n {{ return(columns) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9584372, "supported_languages": null}, "macro.dbt.get_empty_subquery_sql": {"name": "get_empty_subquery_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.get_empty_subquery_sql", "macro_sql": "{% macro get_empty_subquery_sql(select_sql, select_sql_header=none) -%}\n {{ return(adapter.dispatch('get_empty_subquery_sql', 'dbt')(select_sql, select_sql_header)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_empty_subquery_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958444, "supported_languages": null}, "macro.dbt.default__get_empty_subquery_sql": {"name": "default__get_empty_subquery_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.default__get_empty_subquery_sql", "macro_sql": "{% macro default__get_empty_subquery_sql(select_sql, select_sql_header=none) %}\n {%- if select_sql_header is not none -%}\n {{ select_sql_header }}\n {%- endif -%}\n select * from (\n {{ select_sql }}\n ) as __dbt_sbq\n where false\n limit 0\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958451, "supported_languages": null}, "macro.dbt.get_empty_schema_sql": {"name": "get_empty_schema_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.get_empty_schema_sql", "macro_sql": "{% macro get_empty_schema_sql(columns) -%}\n {{ return(adapter.dispatch('get_empty_schema_sql', 'dbt')(columns)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_empty_schema_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9584582, "supported_languages": null}, "macro.dbt.default__get_empty_schema_sql": {"name": "default__get_empty_schema_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.default__get_empty_schema_sql", "macro_sql": "{% macro default__get_empty_schema_sql(columns) %}\n {%- set col_err = [] -%}\n {%- set col_naked_numeric = [] -%}\n select\n {% for i in columns %}\n {%- set col = columns[i] -%}\n {%- if col['data_type'] is not defined -%}\n {%- do col_err.append(col['name']) -%}\n {#-- If this column's type is just 'numeric' then it is missing precision/scale, raise a warning --#}\n {%- elif col['data_type'].strip().lower() in ('numeric', 'decimal', 'number') -%}\n {%- do col_naked_numeric.append(col['name']) -%}\n {%- endif -%}\n {% set col_name = adapter.quote(col['name']) if col.get('quote') else col['name'] %}\n {{ cast('null', col['data_type']) }} as {{ col_name }}{{ \", \" if not loop.last }}\n {%- endfor -%}\n {%- if (col_err | length) > 0 -%}\n {{ exceptions.column_type_missing(column_names=col_err) }}\n {%- elif (col_naked_numeric | length) > 0 -%}\n {{ exceptions.warn(\"Detected columns with numeric type and unspecified precision/scale, this can lead to unintended rounding: \" ~ col_naked_numeric ~ \"`\") }}\n {%- endif -%}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.cast"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958466, "supported_languages": null}, "macro.dbt.get_column_schema_from_query": {"name": "get_column_schema_from_query", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.get_column_schema_from_query", "macro_sql": "{% macro get_column_schema_from_query(select_sql, select_sql_header=none) -%}\n {% set columns = [] %}\n {# -- Using an 'empty subquery' here to get the same schema as the given select_sql statement, without necessitating a data scan.#}\n {% set sql = get_empty_subquery_sql(select_sql, select_sql_header) %}\n {% set column_schema = adapter.get_column_schema_from_query(sql) %}\n {{ return(column_schema) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_empty_subquery_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958473, "supported_languages": null}, "macro.dbt.get_columns_in_query": {"name": "get_columns_in_query", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.get_columns_in_query", "macro_sql": "{% macro get_columns_in_query(select_sql) -%}\n {{ return(adapter.dispatch('get_columns_in_query', 'dbt')(select_sql)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_columns_in_query"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958478, "supported_languages": null}, "macro.dbt.default__get_columns_in_query": {"name": "default__get_columns_in_query", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.default__get_columns_in_query", "macro_sql": "{% macro default__get_columns_in_query(select_sql) %}\n {% call statement('get_columns_in_query', fetch_result=True, auto_begin=False) -%}\n {{ get_empty_subquery_sql(select_sql) }}\n {% endcall %}\n {{ return(load_result('get_columns_in_query').table.columns | map(attribute='name') | list) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement", "macro.dbt.get_empty_subquery_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9584892, "supported_languages": null}, "macro.dbt.alter_column_type": {"name": "alter_column_type", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.alter_column_type", "macro_sql": "{% macro alter_column_type(relation, column_name, new_column_type) -%}\n {{ return(adapter.dispatch('alter_column_type', 'dbt')(relation, column_name, new_column_type)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__alter_column_type"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958498, "supported_languages": null}, "macro.dbt.default__alter_column_type": {"name": "default__alter_column_type", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.default__alter_column_type", "macro_sql": "{% macro default__alter_column_type(relation, column_name, new_column_type) -%}\n {#\n 1. Create a new column (w/ temp name and correct type)\n 2. Copy data over to it\n 3. Drop the existing column (cascade!)\n 4. Rename the new column to existing column\n #}\n {%- set tmp_column = column_name + \"__dbt_alter\" -%}\n\n {% call statement('alter_column_type') %}\n alter table {{ relation }} add column {{ adapter.quote(tmp_column) }} {{ new_column_type }};\n update {{ relation }} set {{ adapter.quote(tmp_column) }} = {{ adapter.quote(column_name) }};\n alter table {{ relation }} drop column {{ adapter.quote(column_name) }} cascade;\n alter table {{ relation }} rename column {{ adapter.quote(tmp_column) }} to {{ adapter.quote(column_name) }}\n {% endcall %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958503, "supported_languages": null}, "macro.dbt.alter_relation_add_remove_columns": {"name": "alter_relation_add_remove_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.alter_relation_add_remove_columns", "macro_sql": "{% macro alter_relation_add_remove_columns(relation, add_columns = none, remove_columns = none) -%}\n {{ return(adapter.dispatch('alter_relation_add_remove_columns', 'dbt')(relation, add_columns, remove_columns)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__alter_relation_add_remove_columns"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9585102, "supported_languages": null}, "macro.dbt.default__alter_relation_add_remove_columns": {"name": "default__alter_relation_add_remove_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.default__alter_relation_add_remove_columns", "macro_sql": "{% macro default__alter_relation_add_remove_columns(relation, add_columns, remove_columns) %}\n\n {% if add_columns is none %}\n {% set add_columns = [] %}\n {% endif %}\n {% if remove_columns is none %}\n {% set remove_columns = [] %}\n {% endif %}\n\n {% set sql -%}\n\n alter {{ relation.type }} {{ relation }}\n\n {% for column in add_columns %}\n add column {{ column.name }} {{ column.data_type }}{{ ',' if not loop.last }}\n {% endfor %}{{ ',' if add_columns and remove_columns }}\n\n {% for column in remove_columns %}\n drop column {{ column.name }}{{ ',' if not loop.last }}\n {% endfor %}\n\n {%- endset -%}\n\n {% do run_query(sql) %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.run_query"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958515, "supported_languages": null}, "macro.dbt.get_fixture_sql": {"name": "get_fixture_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/unit_test_sql/get_fixture_sql.sql", "original_file_path": "macros/unit_test_sql/get_fixture_sql.sql", "unique_id": "macro.dbt.get_fixture_sql", "macro_sql": "{% macro get_fixture_sql(rows, column_name_to_data_types) %}\n-- Fixture for {{ model.name }}\n{% set default_row = {} %}\n\n{%- if not column_name_to_data_types -%}\n{#-- Use defer_relation IFF it is available in the manifest and 'this' is missing from the database --#}\n{%- set this_or_defer_relation = defer_relation if (defer_relation and not load_relation(this)) else this -%}\n{%- set columns_in_relation = adapter.get_columns_in_relation(this_or_defer_relation) -%}\n\n{%- set column_name_to_data_types = {} -%}\n{%- for column in columns_in_relation -%}\n{#-- This needs to be a case-insensitive comparison --#}\n{%- do column_name_to_data_types.update({column.name|lower: column.data_type}) -%}\n{%- endfor -%}\n{%- endif -%}\n\n{%- if not column_name_to_data_types -%}\n {{ exceptions.raise_compiler_error(\"Not able to get columns for unit test '\" ~ model.name ~ \"' from relation \" ~ this) }}\n{%- endif -%}\n\n{%- for column_name, column_type in column_name_to_data_types.items() -%}\n {%- do default_row.update({column_name: (safe_cast(\"null\", column_type) | trim )}) -%}\n{%- endfor -%}\n\n\n{%- for row in rows -%}\n{%- set formatted_row = format_row(row, column_name_to_data_types) -%}\n{%- set default_row_copy = default_row.copy() -%}\n{%- do default_row_copy.update(formatted_row) -%}\nselect\n{%- for column_name, column_value in default_row_copy.items() %} {{ column_value }} as {{ column_name }}{% if not loop.last -%}, {%- endif %}\n{%- endfor %}\n{%- if not loop.last %}\nunion all\n{% endif %}\n{%- endfor -%}\n\n{%- if (rows | length) == 0 -%}\n select\n {%- for column_name, column_value in default_row.items() %} {{ column_value }} as {{ column_name }}{% if not loop.last -%},{%- endif %}\n {%- endfor %}\n limit 0\n{%- endif -%}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.load_relation", "macro.dbt.safe_cast", "macro.dbt.format_row"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958533, "supported_languages": null}, "macro.dbt.get_expected_sql": {"name": "get_expected_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/unit_test_sql/get_fixture_sql.sql", "original_file_path": "macros/unit_test_sql/get_fixture_sql.sql", "unique_id": "macro.dbt.get_expected_sql", "macro_sql": "{% macro get_expected_sql(rows, column_name_to_data_types) %}\n\n{%- if (rows | length) == 0 -%}\n select * from dbt_internal_unit_test_actual\n limit 0\n{%- else -%}\n{%- for row in rows -%}\n{%- set formatted_row = format_row(row, column_name_to_data_types) -%}\nselect\n{%- for column_name, column_value in formatted_row.items() %} {{ column_value }} as {{ column_name }}{% if not loop.last -%}, {%- endif %}\n{%- endfor %}\n{%- if not loop.last %}\nunion all\n{% endif %}\n{%- endfor -%}\n{%- endif -%}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.format_row"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958542, "supported_languages": null}, "macro.dbt.format_row": {"name": "format_row", "resource_type": "macro", "package_name": "dbt", "path": "macros/unit_test_sql/get_fixture_sql.sql", "original_file_path": "macros/unit_test_sql/get_fixture_sql.sql", "unique_id": "macro.dbt.format_row", "macro_sql": "\n\n{%- macro format_row(row, column_name_to_data_types) -%}\n {#-- generate case-insensitive formatted row --#}\n {% set formatted_row = {} %}\n {%- for column_name, column_value in row.items() -%}\n {% set column_name = column_name|lower %}\n\n {%- if column_name not in column_name_to_data_types %}\n {#-- if user-provided row contains column name that relation does not contain, raise an error --#}\n {% set fixture_name = \"expected output\" if model.resource_type == 'unit_test' else (\"'\" ~ model.name ~ \"'\") %}\n {{ exceptions.raise_compiler_error(\n \"Invalid column name: '\" ~ column_name ~ \"' in unit test fixture for \" ~ fixture_name ~ \".\"\n \"\\nAccepted columns for \" ~ fixture_name ~ \" are: \" ~ (column_name_to_data_types.keys()|list)\n ) }}\n {%- endif -%}\n\n {%- set column_type = column_name_to_data_types[column_name] %}\n\n {#-- sanitize column_value: wrap yaml strings in quotes, apply cast --#}\n {%- set column_value_clean = column_value -%}\n {%- if column_value is string -%}\n {%- set column_value_clean = dbt.string_literal(dbt.escape_single_quotes(column_value)) -%}\n {%- elif column_value is none -%}\n {%- set column_value_clean = 'null' -%}\n {%- endif -%}\n\n {%- set row_update = {column_name: safe_cast(column_value_clean, column_type) } -%}\n {%- do formatted_row.update(row_update) -%}\n {%- endfor -%}\n {{ return(formatted_row) }}\n{%- endmacro -%}", "depends_on": {"macros": ["macro.dbt.string_literal", "macro.dbt.escape_single_quotes", "macro.dbt.safe_cast"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9585521, "supported_languages": null}, "macro.dbt.resolve_model_name": {"name": "resolve_model_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/python_model/python.sql", "original_file_path": "macros/python_model/python.sql", "unique_id": "macro.dbt.resolve_model_name", "macro_sql": "{% macro resolve_model_name(input_model_name) %}\n {{ return(adapter.dispatch('resolve_model_name', 'dbt')(input_model_name)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__resolve_model_name"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958566, "supported_languages": null}, "macro.dbt.default__resolve_model_name": {"name": "default__resolve_model_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/python_model/python.sql", "original_file_path": "macros/python_model/python.sql", "unique_id": "macro.dbt.default__resolve_model_name", "macro_sql": "\n\n{%- macro default__resolve_model_name(input_model_name) -%}\n {{ input_model_name | string | replace('\"', '\\\"') }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958571, "supported_languages": null}, "macro.dbt.build_ref_function": {"name": "build_ref_function", "resource_type": "macro", "package_name": "dbt", "path": "macros/python_model/python.sql", "original_file_path": "macros/python_model/python.sql", "unique_id": "macro.dbt.build_ref_function", "macro_sql": "{% macro build_ref_function(model) %}\n\n {%- set ref_dict = {} -%}\n {%- for _ref in model.refs -%}\n {% set _ref_args = [_ref.get('package'), _ref['name']] if _ref.get('package') else [_ref['name'],] %}\n {%- set resolved = ref(*_ref_args, v=_ref.get('version')) -%}\n {%- if _ref.get('version') -%}\n {% do _ref_args.extend([\"v\" ~ _ref['version']]) %}\n {%- endif -%}\n {%- do ref_dict.update({_ref_args | join('.'): resolve_model_name(resolved)}) -%}\n {%- endfor -%}\n\ndef ref(*args, **kwargs):\n refs = {{ ref_dict | tojson }}\n key = '.'.join(args)\n version = kwargs.get(\"v\") or kwargs.get(\"version\")\n if version:\n key += f\".v{version}\"\n dbt_load_df_function = kwargs.get(\"dbt_load_df_function\")\n return dbt_load_df_function(refs[key])\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.resolve_model_name"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9585762, "supported_languages": null}, "macro.dbt.build_source_function": {"name": "build_source_function", "resource_type": "macro", "package_name": "dbt", "path": "macros/python_model/python.sql", "original_file_path": "macros/python_model/python.sql", "unique_id": "macro.dbt.build_source_function", "macro_sql": "{% macro build_source_function(model) %}\n\n {%- set source_dict = {} -%}\n {%- for _source in model.sources -%}\n {%- set resolved = source(*_source) -%}\n {%- do source_dict.update({_source | join('.'): resolve_model_name(resolved)}) -%}\n {%- endfor -%}\n\ndef source(*args, dbt_load_df_function):\n sources = {{ source_dict | tojson }}\n key = '.'.join(args)\n return dbt_load_df_function(sources[key])\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.resolve_model_name"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958584, "supported_languages": null}, "macro.dbt.build_config_dict": {"name": "build_config_dict", "resource_type": "macro", "package_name": "dbt", "path": "macros/python_model/python.sql", "original_file_path": "macros/python_model/python.sql", "unique_id": "macro.dbt.build_config_dict", "macro_sql": "{% macro build_config_dict(model) %}\n {%- set config_dict = {} -%}\n {% set config_dbt_used = zip(model.config.config_keys_used, model.config.config_keys_defaults) | list %}\n {%- for key, default in config_dbt_used -%}\n {# weird type testing with enum, would be much easier to write this logic in Python! #}\n {%- if key == \"language\" -%}\n {%- set value = \"python\" -%}\n {%- endif -%}\n {%- set value = model.config.get(key, default) -%}\n {%- do config_dict.update({key: value}) -%}\n {%- endfor -%}\nconfig_dict = {{ config_dict }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9585888, "supported_languages": null}, "macro.dbt.py_script_postfix": {"name": "py_script_postfix", "resource_type": "macro", "package_name": "dbt", "path": "macros/python_model/python.sql", "original_file_path": "macros/python_model/python.sql", "unique_id": "macro.dbt.py_script_postfix", "macro_sql": "{% macro py_script_postfix(model) %}\n# This part is user provided model code\n# you will need to copy the next section to run the code\n# COMMAND ----------\n# this part is dbt logic for get ref work, do not modify\n\n{{ build_ref_function(model ) }}\n{{ build_source_function(model ) }}\n{{ build_config_dict(model) }}\n\nclass config:\n def __init__(self, *args, **kwargs):\n pass\n\n @staticmethod\n def get(key, default=None):\n return config_dict.get(key, default)\n\nclass this:\n \"\"\"dbt.this() or dbt.this.identifier\"\"\"\n database = \"{{ this.database }}\"\n schema = \"{{ this.schema }}\"\n identifier = \"{{ this.identifier }}\"\n {% set this_relation_name = resolve_model_name(this) %}\n def __repr__(self):\n return '{{ this_relation_name }}'\n\n\nclass dbtObj:\n def __init__(self, load_df_function) -> None:\n self.source = lambda *args: source(*args, dbt_load_df_function=load_df_function)\n self.ref = lambda *args, **kwargs: ref(*args, **kwargs, dbt_load_df_function=load_df_function)\n self.config = config\n self.this = this()\n self.is_incremental = {{ is_incremental() }}\n\n# COMMAND ----------\n{{py_script_comment()}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.build_ref_function", "macro.dbt.build_source_function", "macro.dbt.build_config_dict", "macro.dbt.resolve_model_name", "macro.dbt.is_incremental", "macro.dbt.py_script_comment"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958596, "supported_languages": null}, "macro.dbt.py_script_comment": {"name": "py_script_comment", "resource_type": "macro", "package_name": "dbt", "path": "macros/python_model/python.sql", "original_file_path": "macros/python_model/python.sql", "unique_id": "macro.dbt.py_script_comment", "macro_sql": "{%macro py_script_comment()%}\n{%endmacro%}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.958603, "supported_languages": null}, "macro.dbt.test_unique": {"name": "test_unique", "resource_type": "macro", "package_name": "dbt", "path": "tests/generic/builtin.sql", "original_file_path": "tests/generic/builtin.sql", "unique_id": "macro.dbt.test_unique", "macro_sql": "{% test unique(model, column_name) %}\n {% set macro = adapter.dispatch('test_unique', 'dbt') %}\n {{ macro(model, column_name) }}\n{% endtest %}", "depends_on": {"macros": ["macro.dbt.default__test_unique"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.95939, "supported_languages": null}, "macro.dbt.test_not_null": {"name": "test_not_null", "resource_type": "macro", "package_name": "dbt", "path": "tests/generic/builtin.sql", "original_file_path": "tests/generic/builtin.sql", "unique_id": "macro.dbt.test_not_null", "macro_sql": "{% test not_null(model, column_name) %}\n {% set macro = adapter.dispatch('test_not_null', 'dbt') %}\n {{ macro(model, column_name) }}\n{% endtest %}", "depends_on": {"macros": ["macro.dbt.default__test_not_null"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.959782, "supported_languages": null}, "macro.dbt.test_accepted_values": {"name": "test_accepted_values", "resource_type": "macro", "package_name": "dbt", "path": "tests/generic/builtin.sql", "original_file_path": "tests/generic/builtin.sql", "unique_id": "macro.dbt.test_accepted_values", "macro_sql": "{% test accepted_values(model, column_name, values, quote=True) %}\n {% set macro = adapter.dispatch('test_accepted_values', 'dbt') %}\n {{ macro(model, column_name, values, quote) }}\n{% endtest %}", "depends_on": {"macros": ["macro.dbt.default__test_accepted_values"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.960213, "supported_languages": null}, "macro.dbt.test_relationships": {"name": "test_relationships", "resource_type": "macro", "package_name": "dbt", "path": "tests/generic/builtin.sql", "original_file_path": "tests/generic/builtin.sql", "unique_id": "macro.dbt.test_relationships", "macro_sql": "{% test relationships(model, column_name, to, field) %}\n {% set macro = adapter.dispatch('test_relationships', 'dbt') %}\n {{ macro(model, column_name, to, field) }}\n{% endtest %}", "depends_on": {"macros": ["macro.dbt.default__test_relationships"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1714648414.9606311, "supported_languages": null}}, "docs": {"doc.test.somedoc": {"name": "somedoc", "resource_type": "doc", "package_name": "test", "path": "somedoc.md", "original_file_path": "models/somedoc.md", "unique_id": "doc.test.somedoc", "block_contents": "Testing, testing"}, "doc.dbt.__overview__": {"name": "__overview__", "resource_type": "doc", "package_name": "dbt", "path": "overview.md", "original_file_path": "docs/overview.md", "unique_id": "doc.dbt.__overview__", "block_contents": "### Welcome!\n\nWelcome to the auto-generated documentation for your dbt project!\n\n### Navigation\n\nYou can use the `Project` and `Database` navigation tabs on the left side of the window to explore the models\nin your project.\n\n#### Project Tab\nThe `Project` tab mirrors the directory structure of your dbt project. In this tab, you can see all of the\nmodels defined in your dbt project, as well as models imported from dbt packages.\n\n#### Database Tab\nThe `Database` tab also exposes your models, but in a format that looks more like a database explorer. This view\nshows relations (tables and views) grouped into database schemas. Note that ephemeral models are _not_ shown\nin this interface, as they do not exist in the database.\n\n### Graph Exploration\nYou can click the blue icon on the bottom-right corner of the page to view the lineage graph of your models.\n\nOn model pages, you'll see the immediate parents and children of the model you're exploring. By clicking the `Expand`\nbutton at the top-right of this lineage pane, you'll be able to see all of the models that are used to build,\nor are built from, the model you're exploring.\n\nOnce expanded, you'll be able to use the `--select` and `--exclude` model selection syntax to filter the\nmodels in the graph. For more information on model selection, check out the [dbt docs](https://docs.getdbt.com/docs/model-selection-syntax).\n\nNote that you can also right-click on models to interactively filter and explore the graph.\n\n---\n\n### More information\n\n- [What is dbt](https://docs.getdbt.com/docs/introduction)?\n- Read the [dbt viewpoint](https://docs.getdbt.com/docs/viewpoint)\n- [Installation](https://docs.getdbt.com/docs/installation)\n- Join the [dbt Community](https://www.getdbt.com/community/) for questions and discussion"}}, "exposures": {"exposure.test.simple_exposure": {"name": "simple_exposure", "resource_type": "exposure", "package_name": "test", "path": "schema.yml", "original_file_path": "models/schema.yml", "unique_id": "exposure.test.simple_exposure", "fqn": ["test", "simple_exposure"], "type": "dashboard", "owner": {"email": "something@example.com", "name": null}, "description": "", "label": null, "maturity": null, "meta": {}, "tags": [], "config": {"enabled": true}, "unrendered_config": {}, "url": null, "depends_on": {"macros": [], "nodes": ["source.test.my_source.my_table", "model.test.my_model"]}, "refs": [{"name": "my_model", "package": null, "version": null}], "sources": [["my_source", "my_table"]], "metrics": [], "created_at": 1714648416.054376}}, "metrics": {"metric.test.blue_customers_post_2010": {"name": "blue_customers_post_2010", "resource_type": "metric", "package_name": "test", "path": "schema.yml", "original_file_path": "models/schema.yml", "unique_id": "metric.test.blue_customers_post_2010", "fqn": ["test", "blue_customers_post_2010"], "description": "", "label": "Blue Customers since 2010", "type": "simple", "type_params": {"measure": {"name": "customers", "filter": {"where_filters": [{"where_sql_template": "{{ Dimension('id__favorite_color') }} = 'blue'"}]}, "alias": null, "join_to_timespine": false, "fill_nulls_with": null}, "input_measures": [{"name": "customers", "filter": {"where_filters": [{"where_sql_template": "{{ Dimension('id__favorite_color') }} = 'blue'"}]}, "alias": null, "join_to_timespine": false, "fill_nulls_with": null}], "numerator": null, "denominator": null, "expr": null, "window": null, "grain_to_date": null, "metrics": [], "conversion_type_params": null}, "filter": {"where_filters": [{"where_sql_template": "{{ TimeDimension('id__created_at', 'day') }} > '2010-01-01'"}]}, "metadata": null, "meta": {}, "tags": [], "config": {"enabled": true, "group": null, "meta": {}}, "unrendered_config": {}, "sources": [], "depends_on": {"macros": [], "nodes": ["semantic_model.test.semantic_people"]}, "refs": [], "metrics": [], "created_at": 1714648416.109668, "group": null}, "metric.test.customers": {"name": "customers", "resource_type": "metric", "package_name": "test", "path": "schema.yml", "original_file_path": "models/schema.yml", "unique_id": "metric.test.customers", "fqn": ["test", "customers"], "description": "", "label": "Customers Metric", "type": "simple", "type_params": {"measure": {"name": "customers", "filter": null, "alias": null, "join_to_timespine": false, "fill_nulls_with": null}, "input_measures": [{"name": "customers", "filter": null, "alias": null, "join_to_timespine": false, "fill_nulls_with": null}], "numerator": null, "denominator": null, "expr": null, "window": null, "grain_to_date": null, "metrics": [], "conversion_type_params": null}, "filter": null, "metadata": null, "meta": {}, "tags": [], "config": {"enabled": true, "group": null, "meta": {}}, "unrendered_config": {}, "sources": [], "depends_on": {"macros": [], "nodes": ["semantic_model.test.semantic_people"]}, "refs": [], "metrics": [], "created_at": 1714648416.110243, "group": null}, "metric.test.ratio_of_blue_customers_to_red_customers": {"name": "ratio_of_blue_customers_to_red_customers", "resource_type": "metric", "package_name": "test", "path": "schema.yml", "original_file_path": "models/schema.yml", "unique_id": "metric.test.ratio_of_blue_customers_to_red_customers", "fqn": ["test", "ratio_of_blue_customers_to_red_customers"], "description": "", "label": "Very Important Customer Color Ratio", "type": "ratio", "type_params": {"measure": null, "input_measures": [{"name": "customers", "filter": null, "alias": null, "join_to_timespine": false, "fill_nulls_with": null}], "numerator": {"name": "customers", "filter": {"where_filters": [{"where_sql_template": "{{ Dimension('id__favorite_color')}} = 'blue'"}]}, "alias": null, "offset_window": null, "offset_to_grain": null}, "denominator": {"name": "customers", "filter": {"where_filters": [{"where_sql_template": "{{ Dimension('id__favorite_color')}} = 'red'"}]}, "alias": null, "offset_window": null, "offset_to_grain": null}, "expr": null, "window": null, "grain_to_date": null, "metrics": [], "conversion_type_params": null}, "filter": null, "metadata": null, "meta": {}, "tags": [], "config": {"enabled": true, "group": null, "meta": {}}, "unrendered_config": {}, "sources": [], "depends_on": {"macros": [], "nodes": ["metric.test.customers"]}, "refs": [], "metrics": [], "created_at": 1714648416.1137412, "group": null}, "metric.test.doubled_blue_customers": {"name": "doubled_blue_customers", "resource_type": "metric", "package_name": "test", "path": "schema.yml", "original_file_path": "models/schema.yml", "unique_id": "metric.test.doubled_blue_customers", "fqn": ["test", "doubled_blue_customers"], "description": "", "label": "Inflated blue customer numbers", "type": "derived", "type_params": {"measure": null, "input_measures": [{"name": "customers", "filter": null, "alias": null, "join_to_timespine": false, "fill_nulls_with": null}], "numerator": null, "denominator": null, "expr": "customers * 2", "window": null, "grain_to_date": null, "metrics": [{"name": "customers", "filter": {"where_filters": [{"where_sql_template": "{{ Dimension('id__favorite_color')}} = 'blue'"}]}, "alias": null, "offset_window": null, "offset_to_grain": null}], "conversion_type_params": null}, "filter": null, "metadata": null, "meta": {}, "tags": [], "config": {"enabled": true, "group": null, "meta": {}}, "unrendered_config": {}, "sources": [], "depends_on": {"macros": [], "nodes": ["metric.test.customers"]}, "refs": [], "metrics": [], "created_at": 1714648416.114889, "group": null}}, "groups": {}, "selectors": {}, "disabled": {"model.test.disabled_model": [{"database": "dbt", "schema": "test17146484148326086409_test_previous_version_state", "name": "disabled_model", "resource_type": "model", "package_name": "test", "path": "disabled_model.sql", "original_file_path": "models/disabled_model.sql", "unique_id": "model.test.disabled_model", "fqn": ["test", "disabled_model"], "alias": "disabled_model", "checksum": {"name": "sha256", "checksum": "597106d23ce34e3cd2430588e5c1cf474ebdd138fc47e09b925a4ab258a27acc"}, "config": {"enabled": false, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "view", "incremental_strategy": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": null, "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}, "access": "protected"}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {"enabled": false}, "created_at": 1714648415.360441, "config_call_dict": {"enabled": false}, "relation_name": "\"dbt\".\"test17146484148326086409_test_previous_version_state\".\"disabled_model\"", "raw_code": "{{ config(enabled=False) }}\nselect 2 as id", "language": "sql", "refs": [], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": []}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}, "access": "protected", "constraints": [], "version": null, "latest_version": null, "deprecation_date": null, "defer_relation": null}], "snapshot.test.disabled_snapshot_seed": [{"database": "dbt", "schema": "test17146484148326086409_test_previous_version_state", "name": "disabled_snapshot_seed", "resource_type": "snapshot", "package_name": "test", "path": "disabled_snapshot_seed.sql", "original_file_path": "snapshots/disabled_snapshot_seed.sql", "unique_id": "snapshot.test.disabled_snapshot_seed", "fqn": ["test", "disabled_snapshot_seed", "disabled_snapshot_seed"], "alias": "disabled_snapshot_seed", "checksum": {"name": "sha256", "checksum": "fe76c9dd437341c9e82a0f2a8baf3148f961b768eaa0a4410cd27d3c071bd617"}, "config": {"enabled": false, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "snapshot", "incremental_strategy": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": "id", "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}, "strategy": "check", "target_schema": "test17146484148326086409_test_previous_version_state", "target_database": null, "updated_at": null, "check_cols": "all"}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {"unique_key": "id", "strategy": "check", "check_cols": "all", "target_schema": "test17146484148326086409_test_previous_version_state", "enabled": false}, "created_at": 1714648415.499299, "config_call_dict": {"unique_key": "id", "strategy": "check", "check_cols": "all", "target_schema": "test17146484148326086409_test_previous_version_state", "enabled": false}, "relation_name": "\"dbt\".\"test17146484148326086409_test_previous_version_state\".\"disabled_snapshot_seed\"", "raw_code": "\n{{\n config(\n unique_key='id',\n strategy='check',\n check_cols='all',\n target_schema=schema,\n enabled=False,\n )\n}}\nselect * from {{ ref('my_seed') }}\n", "language": "sql", "refs": [{"name": "my_seed", "package": null, "version": null}], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": []}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}, "defer_relation": null}], "analysis.test.disabled_al": [{"database": "dbt", "schema": "test17146484148326086409_test_previous_version_state", "name": "disabled_al", "resource_type": "analysis", "package_name": "test", "path": "analysis/disabled_al.sql", "original_file_path": "analyses/disabled_al.sql", "unique_id": "analysis.test.disabled_al", "fqn": ["test", "analysis", "disabled_al"], "alias": "disabled_al", "checksum": {"name": "sha256", "checksum": "32d36ad6cff0786eb562440ba60ef6c9b9a7f4c282dfb7a52eaf19d36370f0e1"}, "config": {"enabled": false, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "view", "incremental_strategy": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": null, "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {"enabled": false}, "created_at": 1714648415.5861099, "config_call_dict": {"enabled": false}, "relation_name": null, "raw_code": "{{ config(enabled=False) }}\nselect 9 as id", "language": "sql", "refs": [], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": []}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}}], "test.test.disabled_just_my": [{"database": "dbt", "schema": "test17146484148326086409_test_previous_version_state_dbt_test__audit", "name": "disabled_just_my", "resource_type": "test", "package_name": "test", "path": "disabled_just_my.sql", "original_file_path": "tests/disabled_just_my.sql", "unique_id": "test.test.disabled_just_my", "fqn": ["test", "disabled_just_my"], "alias": "disabled_just_my", "checksum": {"name": "sha256", "checksum": "4f2268fd89a3b4ef899264ada6d7aa33603671cbc5d5acead7dc2eadf1add985"}, "config": {"enabled": false, "alias": null, "schema": "dbt_test__audit", "database": null, "tags": [], "meta": {}, "group": null, "materialized": "test", "severity": "ERROR", "store_failures": null, "store_failures_as": null, "where": null, "limit": null, "fail_calc": "count(*)", "warn_if": "!= 0", "error_if": "!= 0"}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {"enabled": false}, "created_at": 1714648415.6666071, "config_call_dict": {"enabled": false}, "relation_name": null, "raw_code": "{{ config(enabled=False) }}\n\nselect * from {{ ref('my_model') }}\nwhere false", "language": "sql", "refs": [{"name": "my_model", "package": null, "version": null}], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": []}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}}], "test.test.disabled_check_nothing_my_model_.f2c6a72d37": [{"database": "dbt", "schema": "test17146484148326086409_test_previous_version_state_dbt_test__audit", "name": "disabled_check_nothing_my_model_", "resource_type": "test", "package_name": "test", "path": "disabled_check_nothing_my_model_.sql", "original_file_path": "models/schema.yml", "unique_id": "test.test.disabled_check_nothing_my_model_.f2c6a72d37", "fqn": ["test", "disabled_check_nothing_my_model_"], "alias": "disabled_check_nothing_my_model_", "checksum": {"name": "none", "checksum": ""}, "config": {"enabled": false, "alias": null, "schema": "dbt_test__audit", "database": null, "tags": [], "meta": {}, "group": null, "materialized": "test", "severity": "ERROR", "store_failures": null, "store_failures_as": null, "where": null, "limit": null, "fail_calc": "count(*)", "warn_if": "!= 0", "error_if": "!= 0"}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {"enabled": false}, "created_at": 1714648415.9626381, "config_call_dict": {"enabled": false}, "relation_name": null, "raw_code": "{{ test_disabled_check_nothing(**_dbt_generic_test_kwargs) }}", "language": "sql", "refs": [{"name": "my_model", "package": null, "version": null}], "sources": [], "metrics": [], "depends_on": {"macros": ["macro.test.test_disabled_check_nothing", "macro.dbt.get_where_subquery"], "nodes": []}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}, "column_name": null, "file_key_name": "models.my_model", "attached_node": "model.test.my_model", "test_metadata": {"name": "disabled_check_nothing", "kwargs": {"model": "{{ get_where_subquery(ref('my_model')) }}"}, "namespace": null}}], "exposure.test.disabled_exposure": [{"name": "disabled_exposure", "resource_type": "exposure", "package_name": "test", "path": "schema.yml", "original_file_path": "models/schema.yml", "unique_id": "exposure.test.disabled_exposure", "fqn": ["test", "disabled_exposure"], "type": "dashboard", "owner": {"email": "something@example.com", "name": null}, "description": "", "label": null, "maturity": null, "meta": {}, "tags": [], "config": {"enabled": false}, "unrendered_config": {"enabled": false}, "url": null, "depends_on": {"macros": [], "nodes": []}, "refs": [{"name": "my_model", "package": null, "version": null}], "sources": [], "metrics": [], "created_at": 1714648416.0555809}], "metric.test.disabled_metric": [{"name": "disabled_metric", "resource_type": "metric", "package_name": "test", "path": "schema.yml", "original_file_path": "models/schema.yml", "unique_id": "metric.test.disabled_metric", "fqn": ["test", "disabled_metric"], "description": "", "label": "Count records", "type": "simple", "type_params": {"measure": {"name": "customers", "filter": null, "alias": null, "join_to_timespine": false, "fill_nulls_with": null}, "input_measures": [], "numerator": null, "denominator": null, "expr": null, "window": null, "grain_to_date": null, "metrics": [], "conversion_type_params": null}, "filter": {"where_filters": [{"where_sql_template": "{{ Dimension('id__favorite_color') }} = 'blue'"}]}, "metadata": null, "meta": {}, "tags": [], "config": {"enabled": false, "group": null, "meta": {}}, "unrendered_config": {"enabled": false}, "sources": [], "depends_on": {"macros": [], "nodes": []}, "refs": [], "metrics": [], "created_at": 1714648416.110999, "group": null}], "seed.test.disabled_seed": [{"database": "dbt", "schema": "test17146484148326086409_test_previous_version_state", "name": "disabled_seed", "resource_type": "seed", "package_name": "test", "path": "disabled_seed.csv", "original_file_path": "seeds/disabled_seed.csv", "unique_id": "seed.test.disabled_seed", "fqn": ["test", "disabled_seed"], "alias": "disabled_seed", "checksum": {"name": "sha256", "checksum": "31fddd8ec40c6aba6a3a8e7d83fedea2fd0a56c47b64ea3df1847ec1b018e2d1"}, "config": {"enabled": false, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "seed", "incremental_strategy": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": null, "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}, "delimiter": ",", "quote_columns": null}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": "test://models/schema.yml", "build_path": null, "unrendered_config": {"enabled": false}, "created_at": 1714648415.984149, "config_call_dict": {}, "relation_name": "\"dbt\".\"test17146484148326086409_test_previous_version_state\".\"disabled_seed\"", "raw_code": "", "root_path": "/private/var/folders/7h/hj5_fw9j291c58hwfdvy5xbm0000gp/T/pytest-of-jerco/pytest-50/project0", "depends_on": {"macros": []}, "defer_relation": null}], "source.test.my_source.disabled_table": [{"database": "dbt", "schema": "my_source", "name": "disabled_table", "resource_type": "source", "package_name": "test", "path": "models/schema.yml", "original_file_path": "models/schema.yml", "unique_id": "source.test.my_source.disabled_table", "fqn": ["test", "my_source", "disabled_table"], "source_name": "my_source", "source_description": "My source", "loader": "a_loader", "identifier": "disabled_table", "quoting": {"database": null, "schema": null, "identifier": null, "column": null}, "loaded_at_field": null, "freshness": {"warn_after": {"count": null, "period": null}, "error_after": {"count": null, "period": null}, "filter": null}, "external": null, "description": "Disabled table", "columns": {}, "meta": {}, "source_meta": {}, "tags": [], "config": {"enabled": false}, "patch_path": null, "unrendered_config": {"enabled": false}, "relation_name": "\"dbt\".\"my_source\".\"disabled_table\"", "created_at": 1714648416.1610181}]}, "parent_map": {"model.test.my_model": [], "model.test.metricflow_time_spine": [], "snapshot.test.snapshot_seed": ["seed.test.my_seed"], "analysis.test.a": [], "test.test.just_my": ["model.test.my_model"], "seed.test.my_seed": [], "test.test.not_null_my_model_id.43e0e9183a": ["model.test.my_model"], "test.test.check_nothing_my_model_.d5a5e66110": ["model.test.my_model"], "source.test.my_source.my_table": [], "exposure.test.simple_exposure": ["model.test.my_model", "source.test.my_source.my_table"], "metric.test.blue_customers_post_2010": ["semantic_model.test.semantic_people"], "metric.test.customers": ["semantic_model.test.semantic_people"], "metric.test.ratio_of_blue_customers_to_red_customers": ["metric.test.customers"], "metric.test.doubled_blue_customers": ["metric.test.customers"], "semantic_model.test.semantic_people": ["model.test.my_model"]}, "child_map": {"model.test.my_model": ["exposure.test.simple_exposure", "semantic_model.test.semantic_people", "test.test.check_nothing_my_model_.d5a5e66110", "test.test.just_my", "test.test.not_null_my_model_id.43e0e9183a"], "model.test.metricflow_time_spine": [], "snapshot.test.snapshot_seed": [], "analysis.test.a": [], "test.test.just_my": [], "seed.test.my_seed": ["snapshot.test.snapshot_seed"], "test.test.not_null_my_model_id.43e0e9183a": [], "test.test.check_nothing_my_model_.d5a5e66110": [], "source.test.my_source.my_table": ["exposure.test.simple_exposure"], "exposure.test.simple_exposure": [], "metric.test.blue_customers_post_2010": [], "metric.test.customers": ["metric.test.doubled_blue_customers", "metric.test.ratio_of_blue_customers_to_red_customers"], "metric.test.ratio_of_blue_customers_to_red_customers": [], "metric.test.doubled_blue_customers": [], "semantic_model.test.semantic_people": ["metric.test.blue_customers_post_2010", "metric.test.customers"]}, "group_map": {}, "saved_queries": {}, "semantic_models": {"semantic_model.test.semantic_people": {"name": "semantic_people", "resource_type": "semantic_model", "package_name": "test", "path": "schema.yml", "original_file_path": "models/schema.yml", "unique_id": "semantic_model.test.semantic_people", "fqn": ["test", "semantic_people"], "model": "ref('my_model')", "node_relation": {"alias": "my_model", "schema_name": "test17146484148326086409_test_previous_version_state", "database": "dbt", "relation_name": "\"dbt\".\"test17146484148326086409_test_previous_version_state\".\"my_model\""}, "description": null, "label": null, "defaults": {"agg_time_dimension": "created_at"}, "entities": [{"name": "id", "type": "primary", "description": null, "label": null, "role": null, "expr": null}], "measures": [{"name": "years_tenure", "agg": "sum", "description": null, "label": null, "create_metric": false, "expr": "tenure", "agg_params": null, "non_additive_dimension": null, "agg_time_dimension": null}, {"name": "people", "agg": "count", "description": null, "label": null, "create_metric": false, "expr": "id", "agg_params": null, "non_additive_dimension": null, "agg_time_dimension": null}, {"name": "customers", "agg": "count", "description": null, "label": null, "create_metric": false, "expr": "id", "agg_params": null, "non_additive_dimension": null, "agg_time_dimension": null}], "dimensions": [{"name": "favorite_color", "type": "categorical", "description": null, "label": null, "is_partition": false, "type_params": null, "expr": null, "metadata": null}, {"name": "created_at", "type": "time", "description": null, "label": null, "is_partition": false, "type_params": {"time_granularity": "day", "validity_params": null}, "expr": null, "metadata": null}], "metadata": null, "depends_on": {"macros": [], "nodes": ["model.test.my_model"]}, "refs": [{"name": "my_model", "package": null, "version": null}], "created_at": 1714648416.155906, "config": {"enabled": true, "group": null, "meta": {}}, "unrendered_config": {}, "primary_entity": null, "group": null}}, "unit_tests": {}} +{"metadata": {"dbt_schema_version": "https://schemas.getdbt.com/dbt/manifest/v12.json", "dbt_version": "1.9.0b2", "generated_at": "2024-10-17T20:45:45.544925Z", "invocation_id": "24bf5cff-2d7a-40fa-8d57-9b387e83159b", "env": {}, "project_name": "test", "project_id": "098f6bcd4621d373cade4e832627b4f6", "user_id": null, "send_anonymous_usage_stats": false, "adapter_type": "postgres"}, "nodes": {"model.test.my_model": {"database": "dbt", "schema": "test17291979431657508683_test_previous_version_state", "name": "my_model", "resource_type": "model", "package_name": "test", "path": "my_model.sql", "original_file_path": "models/my_model.sql", "unique_id": "model.test.my_model", "fqn": ["test", "my_model"], "alias": "my_model", "checksum": {"name": "sha256", "checksum": "3ea0f972fa1b56aa2dc2f56ee784b6a5796312f9a813d59ae70fd8855f10d16d"}, "config": {"enabled": true, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "view", "incremental_strategy": null, "batch_size": null, "lookback": 1, "begin": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": null, "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}, "event_time": null, "access": "protected"}, "tags": [], "description": "Example model", "columns": {"id": {"name": "id", "description": "", "meta": {}, "data_type": null, "constraints": [], "quote": null, "tags": [], "granularity": null}}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": "test://models/schema.yml", "build_path": null, "unrendered_config": {}, "created_at": 1729197944.086334, "relation_name": "\"dbt\".\"test17291979431657508683_test_previous_version_state\".\"my_model\"", "raw_code": "select 1 as id", "language": "sql", "refs": [], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": []}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}, "access": "protected", "constraints": [], "version": null, "latest_version": null, "deprecation_date": null, "primary_key": [], "time_spine": null}, "model.test.metricflow_time_spine": {"database": "dbt", "schema": "test17291979431657508683_test_previous_version_state", "name": "metricflow_time_spine", "resource_type": "model", "package_name": "test", "path": "metricflow_time_spine.sql", "original_file_path": "models/metricflow_time_spine.sql", "unique_id": "model.test.metricflow_time_spine", "fqn": ["test", "metricflow_time_spine"], "alias": "metricflow_time_spine", "checksum": {"name": "sha256", "checksum": "954d9b349821edb5558a373119a7d91eeac9e620aaa96cd112c0d14bab729fdb"}, "config": {"enabled": true, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "view", "incremental_strategy": null, "batch_size": null, "lookback": 1, "begin": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": null, "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}, "event_time": null, "access": "protected"}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {}, "created_at": 1729197943.716785, "relation_name": "\"dbt\".\"test17291979431657508683_test_previous_version_state\".\"metricflow_time_spine\"", "raw_code": "SELECT to_date('02/20/2023', 'mm/dd/yyyy') as date_day", "language": "sql", "refs": [], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": []}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}, "access": "protected", "constraints": [], "version": null, "latest_version": null, "deprecation_date": null, "primary_key": [], "time_spine": null}, "snapshot.test.snapshot_seed": {"database": "dbt", "schema": "test17291979431657508683_test_previous_version_state", "name": "snapshot_seed", "resource_type": "snapshot", "package_name": "test", "path": "snapshot_seed.sql", "original_file_path": "snapshots/snapshot_seed.sql", "unique_id": "snapshot.test.snapshot_seed", "fqn": ["test", "snapshot_seed", "snapshot_seed"], "alias": "snapshot_seed", "checksum": {"name": "sha256", "checksum": "5fc998f39655f8fe52443a919e749b6e23883ef90202b040412baac13c6bfe18"}, "config": {"enabled": true, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "snapshot", "incremental_strategy": null, "batch_size": null, "lookback": 1, "begin": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": "id", "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}, "event_time": null, "strategy": "check", "target_schema": "test17291979431657508683_test_previous_version_state", "target_database": null, "updated_at": null, "check_cols": "all", "snapshot_meta_column_names": {"dbt_valid_to": null, "dbt_valid_from": null, "dbt_scd_id": null, "dbt_updated_at": null}, "dbt_valid_to_current": null}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {"unique_key": "id", "strategy": "check", "check_cols": "all", "target_schema": "test17291979431657508683_test_previous_version_state"}, "created_at": 1729197943.81848, "relation_name": "\"dbt\".\"test17291979431657508683_test_previous_version_state\".\"snapshot_seed\"", "raw_code": "\n{{\n config(\n unique_key='id',\n strategy='check',\n check_cols='all',\n target_schema=schema,\n )\n}}\nselect * from {{ ref('my_seed') }}\n", "language": "sql", "refs": [{"name": "my_seed", "package": null, "version": null}], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": ["seed.test.my_seed"]}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}}, "analysis.test.a": {"database": "dbt", "schema": "test17291979431657508683_test_previous_version_state", "name": "a", "resource_type": "analysis", "package_name": "test", "path": "analysis/a.sql", "original_file_path": "analyses/a.sql", "unique_id": "analysis.test.a", "fqn": ["test", "analysis", "a"], "alias": "a", "checksum": {"name": "sha256", "checksum": "a389c282f569f0bbdc2a8a4f174dea746c28582fdaf2048d31d9226af9feab23"}, "config": {"enabled": true, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "view", "incremental_strategy": null, "batch_size": null, "lookback": 1, "begin": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": null, "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}, "event_time": null}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {}, "created_at": 1729197943.901566, "relation_name": null, "raw_code": "select 4 as id", "language": "sql", "refs": [], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": []}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}}, "test.test.just_my": {"database": "dbt", "schema": "test17291979431657508683_test_previous_version_state_dbt_test__audit", "name": "just_my", "resource_type": "test", "package_name": "test", "path": "just_my.sql", "original_file_path": "tests/just_my.sql", "unique_id": "test.test.just_my", "fqn": ["test", "just_my"], "alias": "just_my", "checksum": {"name": "sha256", "checksum": "744889a2e2d9ce380619265e1217d7ccf6e6ca896c048d42ebe0f9cfb74d7156"}, "config": {"enabled": true, "alias": null, "schema": "dbt_test__audit", "database": null, "tags": ["data_test_tag"], "meta": {}, "group": null, "materialized": "test", "severity": "ERROR", "store_failures": null, "store_failures_as": null, "where": null, "limit": null, "fail_calc": "count(*)", "warn_if": "!= 0", "error_if": "!= 0"}, "tags": ["data_test_tag"], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {"tags": ["data_test_tag"]}, "created_at": 1729197943.960447, "relation_name": null, "raw_code": "{{ config(tags = ['data_test_tag']) }}\n\nselect * from {{ ref('my_model') }}\nwhere false", "language": "sql", "refs": [{"name": "my_model", "package": null, "version": null}], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": ["model.test.my_model"]}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}}, "seed.test.my_seed": {"database": "dbt", "schema": "test17291979431657508683_test_previous_version_state", "name": "my_seed", "resource_type": "seed", "package_name": "test", "path": "my_seed.csv", "original_file_path": "seeds/my_seed.csv", "unique_id": "seed.test.my_seed", "fqn": ["test", "my_seed"], "alias": "my_seed", "checksum": {"name": "sha256", "checksum": "f7ede83f36165ac6b7a047aa2c3f212dff385bfa9f35f395108cd06fc8e96943"}, "config": {"enabled": true, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "seed", "incremental_strategy": null, "batch_size": null, "lookback": 1, "begin": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": null, "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}, "event_time": null, "delimiter": ",", "quote_columns": null}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {}, "created_at": 1729197944.040256, "relation_name": "\"dbt\".\"test17291979431657508683_test_previous_version_state\".\"my_seed\"", "raw_code": "", "root_path": "/private/var/folders/79/5290gpvn3lx5jdryk4844rm80000gn/T/pytest-of-quigleymalcolm/pytest-289/project0", "depends_on": {"macros": []}}, "test.test.not_null_my_model_id.43e0e9183a": {"database": "dbt", "schema": "test17291979431657508683_test_previous_version_state_dbt_test__audit", "name": "not_null_my_model_id", "resource_type": "test", "package_name": "test", "path": "not_null_my_model_id.sql", "original_file_path": "models/schema.yml", "unique_id": "test.test.not_null_my_model_id.43e0e9183a", "fqn": ["test", "not_null_my_model_id"], "alias": "not_null_my_model_id", "checksum": {"name": "none", "checksum": ""}, "config": {"enabled": true, "alias": null, "schema": "dbt_test__audit", "database": null, "tags": [], "meta": {}, "group": null, "materialized": "test", "severity": "ERROR", "store_failures": null, "store_failures_as": null, "where": null, "limit": null, "fail_calc": "count(*)", "warn_if": "!= 0", "error_if": "!= 0"}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {}, "created_at": 1729197944.1260731, "relation_name": null, "raw_code": "{{ test_not_null(**_dbt_generic_test_kwargs) }}", "language": "sql", "refs": [{"name": "my_model", "package": null, "version": null}], "sources": [], "metrics": [], "depends_on": {"macros": ["macro.dbt.test_not_null"], "nodes": ["model.test.my_model"]}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}, "column_name": "id", "file_key_name": "models.my_model", "attached_node": "model.test.my_model", "test_metadata": {"name": "not_null", "kwargs": {"column_name": "id", "model": "{{ get_where_subquery(ref('my_model')) }}"}, "namespace": null}}, "test.test.check_nothing_my_model_.d5a5e66110": {"database": "dbt", "schema": "test17291979431657508683_test_previous_version_state_dbt_test__audit", "name": "check_nothing_my_model_", "resource_type": "test", "package_name": "test", "path": "check_nothing_my_model_.sql", "original_file_path": "models/schema.yml", "unique_id": "test.test.check_nothing_my_model_.d5a5e66110", "fqn": ["test", "check_nothing_my_model_"], "alias": "check_nothing_my_model_", "checksum": {"name": "none", "checksum": ""}, "config": {"enabled": true, "alias": null, "schema": "dbt_test__audit", "database": null, "tags": [], "meta": {}, "group": null, "materialized": "test", "severity": "ERROR", "store_failures": null, "store_failures_as": null, "where": null, "limit": null, "fail_calc": "count(*)", "warn_if": "!= 0", "error_if": "!= 0"}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {}, "created_at": 1729197944.12699, "relation_name": null, "raw_code": "{{ test_check_nothing(**_dbt_generic_test_kwargs) }}", "language": "sql", "refs": [{"name": "my_model", "package": null, "version": null}], "sources": [], "metrics": [], "depends_on": {"macros": ["macro.test.test_check_nothing", "macro.dbt.get_where_subquery"], "nodes": ["model.test.my_model"]}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}, "column_name": null, "file_key_name": "models.my_model", "attached_node": "model.test.my_model", "test_metadata": {"name": "check_nothing", "kwargs": {"model": "{{ get_where_subquery(ref('my_model')) }}"}, "namespace": null}}}, "sources": {"source.test.my_source.my_table": {"database": "dbt", "schema": "my_source", "name": "my_table", "resource_type": "source", "package_name": "test", "path": "models/schema.yml", "original_file_path": "models/schema.yml", "unique_id": "source.test.my_source.my_table", "fqn": ["test", "my_source", "my_table"], "source_name": "my_source", "source_description": "My source", "loader": "a_loader", "identifier": "my_seed", "quoting": {"database": null, "schema": null, "identifier": null, "column": null}, "loaded_at_field": null, "freshness": {"warn_after": {"count": null, "period": null}, "error_after": {"count": null, "period": null}, "filter": null}, "external": null, "description": "My table", "columns": {}, "meta": {}, "source_meta": {}, "tags": [], "config": {"enabled": true, "event_time": null}, "patch_path": null, "unrendered_config": {}, "relation_name": "\"dbt\".\"my_source\".\"my_seed\"", "created_at": 1729197944.289959, "unrendered_database": null, "unrendered_schema": null}}, "macros": {"macro.test.test_check_nothing": {"name": "test_check_nothing", "resource_type": "macro", "package_name": "test", "path": "macros/dummy_test.sql", "original_file_path": "macros/dummy_test.sql", "unique_id": "macro.test.test_check_nothing", "macro_sql": "{% test check_nothing(model) %}\n-- a silly test to make sure that table-level tests show up in the manifest\n-- without a column_name field\n\nselect 0\n\n{% endtest %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.38802, "supported_languages": null}, "macro.test.test_disabled_check_nothing": {"name": "test_disabled_check_nothing", "resource_type": "macro", "package_name": "test", "path": "macros/disabled_dummy_test.sql", "original_file_path": "macros/disabled_dummy_test.sql", "unique_id": "macro.test.test_disabled_check_nothing", "macro_sql": "{% test disabled_check_nothing(model) %}\n-- a silly test to make sure that table-level tests show up in the manifest\n-- without a column_name field\n\n{{ config(enabled=False) }}\nselect 0\n\n{% endtest %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388277, "supported_languages": null}, "macro.test.do_nothing": {"name": "do_nothing", "resource_type": "macro", "package_name": "test", "path": "macros/do_nothing.sql", "original_file_path": "macros/do_nothing.sql", "unique_id": "macro.test.do_nothing", "macro_sql": "{% macro do_nothing(foo2, bar2) %}\n select\n '{{ foo2 }}' as foo2,\n '{{ bar2 }}' as bar2\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3884742, "supported_languages": null}, "macro.dbt_postgres.postgres__current_timestamp": {"name": "postgres__current_timestamp", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/timestamps.sql", "original_file_path": "macros/timestamps.sql", "unique_id": "macro.dbt_postgres.postgres__current_timestamp", "macro_sql": "{% macro postgres__current_timestamp() -%}\n now()\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388491, "supported_languages": null}, "macro.dbt_postgres.postgres__snapshot_string_as_time": {"name": "postgres__snapshot_string_as_time", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/timestamps.sql", "original_file_path": "macros/timestamps.sql", "unique_id": "macro.dbt_postgres.postgres__snapshot_string_as_time", "macro_sql": "{% macro postgres__snapshot_string_as_time(timestamp) -%}\n {%- set result = \"'\" ~ timestamp ~ \"'::timestamp without time zone\" -%}\n {{ return(result) }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388496, "supported_languages": null}, "macro.dbt_postgres.postgres__snapshot_get_time": {"name": "postgres__snapshot_get_time", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/timestamps.sql", "original_file_path": "macros/timestamps.sql", "unique_id": "macro.dbt_postgres.postgres__snapshot_get_time", "macro_sql": "{% macro postgres__snapshot_get_time() -%}\n {{ current_timestamp() }}::timestamp without time zone\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.current_timestamp"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3885021, "supported_languages": null}, "macro.dbt_postgres.postgres__current_timestamp_backcompat": {"name": "postgres__current_timestamp_backcompat", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/timestamps.sql", "original_file_path": "macros/timestamps.sql", "unique_id": "macro.dbt_postgres.postgres__current_timestamp_backcompat", "macro_sql": "{% macro postgres__current_timestamp_backcompat() %}\n current_timestamp::{{ type_timestamp() }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.type_timestamp"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388509, "supported_languages": null}, "macro.dbt_postgres.postgres__current_timestamp_in_utc_backcompat": {"name": "postgres__current_timestamp_in_utc_backcompat", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/timestamps.sql", "original_file_path": "macros/timestamps.sql", "unique_id": "macro.dbt_postgres.postgres__current_timestamp_in_utc_backcompat", "macro_sql": "{% macro postgres__current_timestamp_in_utc_backcompat() %}\n (current_timestamp at time zone 'utc')::{{ type_timestamp() }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.type_timestamp"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388515, "supported_languages": null}, "macro.dbt_postgres.postgres__get_catalog_relations": {"name": "postgres__get_catalog_relations", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/catalog.sql", "original_file_path": "macros/catalog.sql", "unique_id": "macro.dbt_postgres.postgres__get_catalog_relations", "macro_sql": "{% macro postgres__get_catalog_relations(information_schema, relations) -%}\n {%- call statement('catalog', fetch_result=True) -%}\n\n {#\n If the user has multiple databases set and the first one is wrong, this will fail.\n But we won't fail in the case where there are multiple quoting-difference-only dbs, which is better.\n #}\n {% set database = information_schema.database %}\n {{ adapter.verify_database(database) }}\n\n select\n '{{ database }}' as table_database,\n sch.nspname as table_schema,\n tbl.relname as table_name,\n case tbl.relkind\n when 'v' then 'VIEW'\n when 'm' then 'MATERIALIZED VIEW'\n else 'BASE TABLE'\n end as table_type,\n tbl_desc.description as table_comment,\n col.attname as column_name,\n col.attnum as column_index,\n pg_catalog.format_type(col.atttypid, col.atttypmod) as column_type,\n col_desc.description as column_comment,\n pg_get_userbyid(tbl.relowner) as table_owner\n\n from pg_catalog.pg_namespace sch\n join pg_catalog.pg_class tbl on tbl.relnamespace = sch.oid\n join pg_catalog.pg_attribute col on col.attrelid = tbl.oid\n left outer join pg_catalog.pg_description tbl_desc on (tbl_desc.objoid = tbl.oid and tbl_desc.objsubid = 0)\n left outer join pg_catalog.pg_description col_desc on (col_desc.objoid = tbl.oid and col_desc.objsubid = col.attnum)\n where (\n {%- for relation in relations -%}\n {%- if relation.identifier -%}\n (upper(sch.nspname) = upper('{{ relation.schema }}') and\n upper(tbl.relname) = upper('{{ relation.identifier }}'))\n {%- else-%}\n upper(sch.nspname) = upper('{{ relation.schema }}')\n {%- endif -%}\n {%- if not loop.last %} or {% endif -%}\n {%- endfor -%}\n )\n and not pg_is_other_temp_schema(sch.oid) -- not a temporary schema belonging to another session\n and tbl.relpersistence in ('p', 'u') -- [p]ermanent table or [u]nlogged table. Exclude [t]emporary tables\n and tbl.relkind in ('r', 'v', 'f', 'p', 'm') -- o[r]dinary table, [v]iew, [f]oreign table, [p]artitioned table, [m]aterialized view. Other values are [i]ndex, [S]equence, [c]omposite type, [t]OAST table\n and col.attnum > 0 -- negative numbers are used for system columns such as oid\n and not col.attisdropped -- column as not been dropped\n\n order by\n sch.nspname,\n tbl.relname,\n col.attnum\n\n {%- endcall -%}\n\n {{ return(load_result('catalog').table) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388556, "supported_languages": null}, "macro.dbt_postgres.postgres__get_catalog": {"name": "postgres__get_catalog", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/catalog.sql", "original_file_path": "macros/catalog.sql", "unique_id": "macro.dbt_postgres.postgres__get_catalog", "macro_sql": "{% macro postgres__get_catalog(information_schema, schemas) -%}\n {%- set relations = [] -%}\n {%- for schema in schemas -%}\n {%- set dummy = relations.append({'schema': schema}) -%}\n {%- endfor -%}\n {{ return(postgres__get_catalog_relations(information_schema, relations)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_catalog_relations"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388562, "supported_languages": null}, "macro.dbt_postgres.postgres__get_relations": {"name": "postgres__get_relations", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations.sql", "original_file_path": "macros/relations.sql", "unique_id": "macro.dbt_postgres.postgres__get_relations", "macro_sql": "{% macro postgres__get_relations() -%}\n\n {#\n -- in pg_depend, objid is the dependent, refobjid is the referenced object\n -- > a pg_depend entry indicates that the referenced object cannot be\n -- > dropped without also dropping the dependent object.\n #}\n\n {%- call statement('relations', fetch_result=True) -%}\n with relation as (\n select\n pg_rewrite.ev_class as class,\n pg_rewrite.oid as id\n from pg_rewrite\n ),\n class as (\n select\n oid as id,\n relname as name,\n relnamespace as schema,\n relkind as kind\n from pg_class\n ),\n dependency as (\n select distinct\n pg_depend.objid as id,\n pg_depend.refobjid as ref\n from pg_depend\n ),\n schema as (\n select\n pg_namespace.oid as id,\n pg_namespace.nspname as name\n from pg_namespace\n where nspname != 'information_schema' and nspname not like 'pg\\_%'\n ),\n referenced as (\n select\n relation.id AS id,\n referenced_class.name ,\n referenced_class.schema ,\n referenced_class.kind\n from relation\n join class as referenced_class on relation.class=referenced_class.id\n where referenced_class.kind in ('r', 'v', 'm')\n ),\n relationships as (\n select\n referenced.name as referenced_name,\n referenced.schema as referenced_schema_id,\n dependent_class.name as dependent_name,\n dependent_class.schema as dependent_schema_id,\n referenced.kind as kind\n from referenced\n join dependency on referenced.id=dependency.id\n join class as dependent_class on dependency.ref=dependent_class.id\n where\n (referenced.name != dependent_class.name or\n referenced.schema != dependent_class.schema)\n )\n\n select\n referenced_schema.name as referenced_schema,\n relationships.referenced_name as referenced_name,\n dependent_schema.name as dependent_schema,\n relationships.dependent_name as dependent_name\n from relationships\n join schema as dependent_schema on relationships.dependent_schema_id=dependent_schema.id\n join schema as referenced_schema on relationships.referenced_schema_id=referenced_schema.id\n group by referenced_schema, referenced_name, dependent_schema, dependent_name\n order by referenced_schema, referenced_name, dependent_schema, dependent_name;\n\n {%- endcall -%}\n\n {{ return(load_result('relations').table) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3885708, "supported_languages": null}, "macro.dbt_postgres.postgres_get_relations": {"name": "postgres_get_relations", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations.sql", "original_file_path": "macros/relations.sql", "unique_id": "macro.dbt_postgres.postgres_get_relations", "macro_sql": "{% macro postgres_get_relations() %}\n {{ return(postgres__get_relations()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_relations"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388577, "supported_languages": null}, "macro.dbt_postgres.postgres__create_table_as": {"name": "postgres__create_table_as", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__create_table_as", "macro_sql": "{% macro postgres__create_table_as(temporary, relation, sql) -%}\n {%- set unlogged = config.get('unlogged', default=false) -%}\n {%- set sql_header = config.get('sql_header', none) -%}\n\n {{ sql_header if sql_header is not none }}\n\n create {% if temporary -%}\n temporary\n {%- elif unlogged -%}\n unlogged\n {%- endif %} table {{ relation }}\n {% set contract_config = config.get('contract') %}\n {% if contract_config.enforced %}\n {{ get_assert_columns_equivalent(sql) }}\n {% endif -%}\n {% if contract_config.enforced and (not temporary) -%}\n {{ get_table_columns_and_constraints() }} ;\n insert into {{ relation }} (\n {{ adapter.dispatch('get_column_names', 'dbt')() }}\n )\n {%- set sql = get_select_subquery(sql) %}\n {% else %}\n as\n {% endif %}\n (\n {{ sql }}\n );\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.get_assert_columns_equivalent", "macro.dbt.get_table_columns_and_constraints", "macro.dbt.default__get_column_names", "macro.dbt.get_select_subquery"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388591, "supported_languages": null}, "macro.dbt_postgres.postgres__get_create_index_sql": {"name": "postgres__get_create_index_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__get_create_index_sql", "macro_sql": "{% macro postgres__get_create_index_sql(relation, index_dict) -%}\n {%- set index_config = adapter.parse_index(index_dict) -%}\n {%- set comma_separated_columns = \", \".join(index_config.columns) -%}\n {%- set index_name = index_config.render(relation) -%}\n\n create {% if index_config.unique -%}\n unique\n {%- endif %} index if not exists\n \"{{ index_name }}\"\n on {{ relation }} {% if index_config.type -%}\n using {{ index_config.type }}\n {%- endif %}\n ({{ comma_separated_columns }})\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388598, "supported_languages": null}, "macro.dbt_postgres.postgres__create_schema": {"name": "postgres__create_schema", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__create_schema", "macro_sql": "{% macro postgres__create_schema(relation) -%}\n {% if relation.database -%}\n {{ adapter.verify_database(relation.database) }}\n {%- endif -%}\n {%- call statement('create_schema') -%}\n create schema if not exists {{ relation.without_identifier().include(database=False) }}\n {%- endcall -%}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388602, "supported_languages": null}, "macro.dbt_postgres.postgres__drop_schema": {"name": "postgres__drop_schema", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__drop_schema", "macro_sql": "{% macro postgres__drop_schema(relation) -%}\n {% if relation.database -%}\n {{ adapter.verify_database(relation.database) }}\n {%- endif -%}\n {%- call statement('drop_schema') -%}\n drop schema if exists {{ relation.without_identifier().include(database=False) }} cascade\n {%- endcall -%}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3886101, "supported_languages": null}, "macro.dbt_postgres.postgres__get_columns_in_relation": {"name": "postgres__get_columns_in_relation", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__get_columns_in_relation", "macro_sql": "{% macro postgres__get_columns_in_relation(relation) -%}\n {% call statement('get_columns_in_relation', fetch_result=True) %}\n select\n column_name,\n data_type,\n character_maximum_length,\n numeric_precision,\n numeric_scale\n\n from {{ relation.information_schema('columns') }}\n where table_name = '{{ relation.identifier }}'\n {% if relation.schema %}\n and table_schema = '{{ relation.schema }}'\n {% endif %}\n order by ordinal_position\n\n {% endcall %}\n {% set table = load_result('get_columns_in_relation').table %}\n {{ return(sql_convert_columns_in_relation(table)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement", "macro.dbt.sql_convert_columns_in_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3886158, "supported_languages": null}, "macro.dbt_postgres.postgres__list_relations_without_caching": {"name": "postgres__list_relations_without_caching", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__list_relations_without_caching", "macro_sql": "{% macro postgres__list_relations_without_caching(schema_relation) %}\n {% call statement('list_relations_without_caching', fetch_result=True) -%}\n select\n '{{ schema_relation.database }}' as database,\n tablename as name,\n schemaname as schema,\n 'table' as type\n from pg_tables\n where schemaname ilike '{{ schema_relation.schema }}'\n union all\n select\n '{{ schema_relation.database }}' as database,\n viewname as name,\n schemaname as schema,\n 'view' as type\n from pg_views\n where schemaname ilike '{{ schema_relation.schema }}'\n union all\n select\n '{{ schema_relation.database }}' as database,\n matviewname as name,\n schemaname as schema,\n 'materialized_view' as type\n from pg_matviews\n where schemaname ilike '{{ schema_relation.schema }}'\n {% endcall %}\n {{ return(load_result('list_relations_without_caching').table) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3886201, "supported_languages": null}, "macro.dbt_postgres.postgres__information_schema_name": {"name": "postgres__information_schema_name", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__information_schema_name", "macro_sql": "{% macro postgres__information_schema_name(database) -%}\n {% if database_name -%}\n {{ adapter.verify_database(database_name) }}\n {%- endif -%}\n information_schema\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388625, "supported_languages": null}, "macro.dbt_postgres.postgres__list_schemas": {"name": "postgres__list_schemas", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__list_schemas", "macro_sql": "{% macro postgres__list_schemas(database) %}\n {% if database -%}\n {{ adapter.verify_database(database) }}\n {%- endif -%}\n {% call statement('list_schemas', fetch_result=True, auto_begin=False) %}\n select distinct nspname from pg_namespace\n {% endcall %}\n {{ return(load_result('list_schemas').table) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.38863, "supported_languages": null}, "macro.dbt_postgres.postgres__check_schema_exists": {"name": "postgres__check_schema_exists", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__check_schema_exists", "macro_sql": "{% macro postgres__check_schema_exists(information_schema, schema) -%}\n {% if information_schema.database -%}\n {{ adapter.verify_database(information_schema.database) }}\n {%- endif -%}\n {% call statement('check_schema_exists', fetch_result=True, auto_begin=False) %}\n select count(*) from pg_namespace where nspname = '{{ schema }}'\n {% endcall %}\n {{ return(load_result('check_schema_exists').table) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388635, "supported_languages": null}, "macro.dbt_postgres.postgres__make_relation_with_suffix": {"name": "postgres__make_relation_with_suffix", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__make_relation_with_suffix", "macro_sql": "{% macro postgres__make_relation_with_suffix(base_relation, suffix, dstring) %}\n {% if dstring %}\n {% set dt = modules.datetime.datetime.now() %}\n {% set dtstring = dt.strftime(\"%H%M%S%f\") %}\n {% set suffix = suffix ~ dtstring %}\n {% endif %}\n {% set suffix_length = suffix|length %}\n {% set relation_max_name_length = base_relation.relation_max_name_length() %}\n {% if suffix_length > relation_max_name_length %}\n {% do exceptions.raise_compiler_error('Relation suffix is too long (' ~ suffix_length ~ ' characters). Maximum length is ' ~ relation_max_name_length ~ ' characters.') %}\n {% endif %}\n {% set identifier = base_relation.identifier[:relation_max_name_length - suffix_length] ~ suffix %}\n\n {{ return(base_relation.incorporate(path={\"identifier\": identifier })) }}\n\n {% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388639, "supported_languages": null}, "macro.dbt_postgres.postgres__make_intermediate_relation": {"name": "postgres__make_intermediate_relation", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__make_intermediate_relation", "macro_sql": "{% macro postgres__make_intermediate_relation(base_relation, suffix) %}\n {{ return(postgres__make_relation_with_suffix(base_relation, suffix, dstring=False)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__make_relation_with_suffix"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388646, "supported_languages": null}, "macro.dbt_postgres.postgres__make_temp_relation": {"name": "postgres__make_temp_relation", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__make_temp_relation", "macro_sql": "{% macro postgres__make_temp_relation(base_relation, suffix) %}\n {% set temp_relation = postgres__make_relation_with_suffix(base_relation, suffix, dstring=True) %}\n {{ return(temp_relation.incorporate(path={\"schema\": none,\n \"database\": none})) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__make_relation_with_suffix"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.38865, "supported_languages": null}, "macro.dbt_postgres.postgres__make_backup_relation": {"name": "postgres__make_backup_relation", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__make_backup_relation", "macro_sql": "{% macro postgres__make_backup_relation(base_relation, backup_relation_type, suffix) %}\n {% set backup_relation = postgres__make_relation_with_suffix(base_relation, suffix, dstring=False) %}\n {{ return(backup_relation.incorporate(type=backup_relation_type)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__make_relation_with_suffix"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3886561, "supported_languages": null}, "macro.dbt_postgres.postgres_escape_comment": {"name": "postgres_escape_comment", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres_escape_comment", "macro_sql": "{% macro postgres_escape_comment(comment) -%}\n {% if comment is not string %}\n {% do exceptions.raise_compiler_error('cannot escape a non-string: ' ~ comment) %}\n {% endif %}\n {%- set magic = '$dbt_comment_literal_block$' -%}\n {%- if magic in comment -%}\n {%- do exceptions.raise_compiler_error('The string ' ~ magic ~ ' is not allowed in comments.') -%}\n {%- endif -%}\n {{ magic }}{{ comment }}{{ magic }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388662, "supported_languages": null}, "macro.dbt_postgres.postgres__alter_relation_comment": {"name": "postgres__alter_relation_comment", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__alter_relation_comment", "macro_sql": "{% macro postgres__alter_relation_comment(relation, comment) %}\n {% set escaped_comment = postgres_escape_comment(comment) %}\n {% if relation.type == 'materialized_view' -%}\n {% set relation_type = \"materialized view\" %}\n {%- else -%}\n {%- set relation_type = relation.type -%}\n {%- endif -%}\n comment on {{ relation_type }} {{ relation }} is {{ escaped_comment }};\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres_escape_comment"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3886678, "supported_languages": null}, "macro.dbt_postgres.postgres__alter_column_comment": {"name": "postgres__alter_column_comment", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__alter_column_comment", "macro_sql": "{% macro postgres__alter_column_comment(relation, column_dict) %}\n {% set existing_columns = adapter.get_columns_in_relation(relation) | map(attribute=\"name\") | list %}\n {% for column_name in column_dict if (column_name in existing_columns) %}\n {% set comment = column_dict[column_name]['description'] %}\n {% set escaped_comment = postgres_escape_comment(comment) %}\n comment on column {{ relation }}.{{ adapter.quote(column_name) if column_dict[column_name]['quote'] else column_name }} is {{ escaped_comment }};\n {% endfor %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres_escape_comment"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388672, "supported_languages": null}, "macro.dbt_postgres.postgres__get_show_grant_sql": {"name": "postgres__get_show_grant_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__get_show_grant_sql", "macro_sql": "\n\n{%- macro postgres__get_show_grant_sql(relation) -%}\n select grantee, privilege_type\n from {{ relation.information_schema('role_table_grants') }}\n where grantor = current_role\n and grantee != current_role\n and table_schema = '{{ relation.schema }}'\n and table_name = '{{ relation.identifier }}'\n{%- endmacro -%}\n\n", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388682, "supported_languages": null}, "macro.dbt_postgres.postgres__copy_grants": {"name": "postgres__copy_grants", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__copy_grants", "macro_sql": "{% macro postgres__copy_grants() %}\n {{ return(False) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3886871, "supported_languages": null}, "macro.dbt_postgres.postgres__get_show_indexes_sql": {"name": "postgres__get_show_indexes_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__get_show_indexes_sql", "macro_sql": "{% macro postgres__get_show_indexes_sql(relation) %}\n select\n i.relname as name,\n m.amname as method,\n ix.indisunique as \"unique\",\n array_to_string(array_agg(a.attname), ',') as column_names\n from pg_index ix\n join pg_class i\n on i.oid = ix.indexrelid\n join pg_am m\n on m.oid=i.relam\n join pg_class t\n on t.oid = ix.indrelid\n join pg_namespace n\n on n.oid = t.relnamespace\n join pg_attribute a\n on a.attrelid = t.oid\n and a.attnum = ANY(ix.indkey)\n where t.relname = '{{ relation.identifier }}'\n and n.nspname = '{{ relation.schema }}'\n and t.relkind in ('r', 'm')\n group by 1, 2, 3\n order by 1, 2, 3\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388691, "supported_languages": null}, "macro.dbt_postgres.postgres__get_drop_index_sql": {"name": "postgres__get_drop_index_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/adapters.sql", "original_file_path": "macros/adapters.sql", "unique_id": "macro.dbt_postgres.postgres__get_drop_index_sql", "macro_sql": "\n\n\n{%- macro postgres__get_drop_index_sql(relation, index_name) -%}\n drop index if exists \"{{ relation.schema }}\".\"{{ index_name }}\"\n{%- endmacro -%}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388695, "supported_languages": null}, "macro.dbt_postgres.postgres__get_incremental_default_sql": {"name": "postgres__get_incremental_default_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/materializations/incremental_strategies.sql", "original_file_path": "macros/materializations/incremental_strategies.sql", "unique_id": "macro.dbt_postgres.postgres__get_incremental_default_sql", "macro_sql": "{% macro postgres__get_incremental_default_sql(arg_dict) %}\n\n {% if arg_dict[\"unique_key\"] %}\n {% do return(get_incremental_delete_insert_sql(arg_dict)) %}\n {% else %}\n {% do return(get_incremental_append_sql(arg_dict)) %}\n {% endif %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_incremental_delete_insert_sql", "macro.dbt.get_incremental_append_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3887072, "supported_languages": null}, "macro.dbt_postgres.postgres__get_incremental_microbatch_sql": {"name": "postgres__get_incremental_microbatch_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/materializations/incremental_strategies.sql", "original_file_path": "macros/materializations/incremental_strategies.sql", "unique_id": "macro.dbt_postgres.postgres__get_incremental_microbatch_sql", "macro_sql": "{% macro postgres__get_incremental_microbatch_sql(arg_dict) %}\n\n {% if arg_dict[\"unique_key\"] %}\n {% do return(adapter.dispatch('get_incremental_merge_sql', 'dbt')(arg_dict)) %}\n {% else %}\n {{ exceptions.raise_compiler_error(\"dbt-postgres 'microbatch' requires a `unique_key` config\") }}\n {% endif %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_incremental_merge_sql", "macro.dbt.default__get_incremental_merge_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388712, "supported_languages": null}, "macro.dbt_postgres.postgres__snapshot_merge_sql": {"name": "postgres__snapshot_merge_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/materializations/snapshot_merge.sql", "original_file_path": "macros/materializations/snapshot_merge.sql", "unique_id": "macro.dbt_postgres.postgres__snapshot_merge_sql", "macro_sql": "{% macro postgres__snapshot_merge_sql(target, source, insert_cols) -%}\n {%- set insert_cols_csv = insert_cols | join(', ') -%}\n\n {%- set columns = config.get(\"snapshot_table_column_names\") or get_snapshot_table_column_names() -%}\n\n update {{ target }}\n set {{ columns.dbt_valid_to }} = DBT_INTERNAL_SOURCE.{{ columns.dbt_valid_to }}\n from {{ source }} as DBT_INTERNAL_SOURCE\n where DBT_INTERNAL_SOURCE.{{ columns.dbt_scd_id }}::text = {{ target }}.{{ columns.dbt_scd_id }}::text\n and DBT_INTERNAL_SOURCE.dbt_change_type::text in ('update'::text, 'delete'::text)\n {% if config.get(\"dbt_valid_to_current\") %}\n and ({{ target }}.{{ columns.dbt_valid_to }} = {{ config.get('dbt_valid_to_current') }} or {{ target }}.{{ columns.dbt_valid_to }} is null);\n {% else %}\n and {{ target }}.{{ columns.dbt_valid_to }} is null;\n {% endif %}\n\n\n insert into {{ target }} ({{ insert_cols_csv }})\n select {% for column in insert_cols -%}\n DBT_INTERNAL_SOURCE.{{ column }} {%- if not loop.last %}, {%- endif %}\n {%- endfor %}\n from {{ source }} as DBT_INTERNAL_SOURCE\n where DBT_INTERNAL_SOURCE.dbt_change_type::text = 'insert'::text;\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_snapshot_table_column_names"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388725, "supported_languages": null}, "macro.dbt_postgres.postgres__drop_materialized_view": {"name": "postgres__drop_materialized_view", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/materialized_view/drop.sql", "original_file_path": "macros/relations/materialized_view/drop.sql", "unique_id": "macro.dbt_postgres.postgres__drop_materialized_view", "macro_sql": "{% macro postgres__drop_materialized_view(relation) -%}\n drop materialized view if exists {{ relation }} cascade\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388733, "supported_languages": null}, "macro.dbt_postgres.postgres__describe_materialized_view": {"name": "postgres__describe_materialized_view", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/materialized_view/describe.sql", "original_file_path": "macros/relations/materialized_view/describe.sql", "unique_id": "macro.dbt_postgres.postgres__describe_materialized_view", "macro_sql": "{% macro postgres__describe_materialized_view(relation) %}\n -- for now just get the indexes, we don't need the name or the query yet\n {% set _indexes = run_query(get_show_indexes_sql(relation)) %}\n {% do return({'indexes': _indexes}) %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.run_query", "macro.dbt.get_show_indexes_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388741, "supported_languages": null}, "macro.dbt_postgres.postgres__refresh_materialized_view": {"name": "postgres__refresh_materialized_view", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/materialized_view/refresh.sql", "original_file_path": "macros/relations/materialized_view/refresh.sql", "unique_id": "macro.dbt_postgres.postgres__refresh_materialized_view", "macro_sql": "{% macro postgres__refresh_materialized_view(relation) %}\n refresh materialized view {{ relation }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388756, "supported_languages": null}, "macro.dbt_postgres.postgres__get_rename_materialized_view_sql": {"name": "postgres__get_rename_materialized_view_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/materialized_view/rename.sql", "original_file_path": "macros/relations/materialized_view/rename.sql", "unique_id": "macro.dbt_postgres.postgres__get_rename_materialized_view_sql", "macro_sql": "{% macro postgres__get_rename_materialized_view_sql(relation, new_name) %}\n alter materialized view {{ relation }} rename to {{ new_name }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388764, "supported_languages": null}, "macro.dbt_postgres.postgres__get_alter_materialized_view_as_sql": {"name": "postgres__get_alter_materialized_view_as_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/materialized_view/alter.sql", "original_file_path": "macros/relations/materialized_view/alter.sql", "unique_id": "macro.dbt_postgres.postgres__get_alter_materialized_view_as_sql", "macro_sql": "{% macro postgres__get_alter_materialized_view_as_sql(\n relation,\n configuration_changes,\n sql,\n existing_relation,\n backup_relation,\n intermediate_relation\n) %}\n\n -- apply a full refresh immediately if needed\n {% if configuration_changes.requires_full_refresh %}\n\n {{ get_replace_sql(existing_relation, relation, sql) }}\n\n -- otherwise apply individual changes as needed\n {% else %}\n\n {{ postgres__update_indexes_on_materialized_view(relation, configuration_changes.indexes) }}\n\n {%- endif -%}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_replace_sql", "macro.dbt_postgres.postgres__update_indexes_on_materialized_view"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388772, "supported_languages": null}, "macro.dbt_postgres.postgres__update_indexes_on_materialized_view": {"name": "postgres__update_indexes_on_materialized_view", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/materialized_view/alter.sql", "original_file_path": "macros/relations/materialized_view/alter.sql", "unique_id": "macro.dbt_postgres.postgres__update_indexes_on_materialized_view", "macro_sql": "\n\n\n{%- macro postgres__update_indexes_on_materialized_view(relation, index_changes) -%}\n {{- log(\"Applying UPDATE INDEXES to: \" ~ relation) -}}\n\n {%- for _index_change in index_changes -%}\n {%- set _index = _index_change.context -%}\n\n {%- if _index_change.action == \"drop\" -%}\n\n {{ postgres__get_drop_index_sql(relation, _index.name) }}\n\n {%- elif _index_change.action == \"create\" -%}\n\n {{ postgres__get_create_index_sql(relation, _index.as_node_config) }}\n\n {%- endif -%}\n\t{{ ';' if not loop.last else \"\" }}\n\n {%- endfor -%}\n\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_drop_index_sql", "macro.dbt_postgres.postgres__get_create_index_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388781, "supported_languages": null}, "macro.dbt_postgres.postgres__get_materialized_view_configuration_changes": {"name": "postgres__get_materialized_view_configuration_changes", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/materialized_view/alter.sql", "original_file_path": "macros/relations/materialized_view/alter.sql", "unique_id": "macro.dbt_postgres.postgres__get_materialized_view_configuration_changes", "macro_sql": "{% macro postgres__get_materialized_view_configuration_changes(existing_relation, new_config) %}\n {% set _existing_materialized_view = postgres__describe_materialized_view(existing_relation) %}\n {% set _configuration_changes = existing_relation.get_materialized_view_config_change_collection(_existing_materialized_view, new_config.model) %}\n {% do return(_configuration_changes) %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__describe_materialized_view"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388789, "supported_languages": null}, "macro.dbt_postgres.postgres__get_create_materialized_view_as_sql": {"name": "postgres__get_create_materialized_view_as_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/materialized_view/create.sql", "original_file_path": "macros/relations/materialized_view/create.sql", "unique_id": "macro.dbt_postgres.postgres__get_create_materialized_view_as_sql", "macro_sql": "{% macro postgres__get_create_materialized_view_as_sql(relation, sql) %}\n create materialized view if not exists {{ relation }} as {{ sql }};\n\n {% for _index_dict in config.get('indexes', []) -%}\n {{- get_create_index_sql(relation, _index_dict) -}}{{ ';' if not loop.last else \"\" }}\n {%- endfor -%}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_create_index_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388797, "supported_languages": null}, "macro.dbt_postgres.postgres__drop_table": {"name": "postgres__drop_table", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/table/drop.sql", "original_file_path": "macros/relations/table/drop.sql", "unique_id": "macro.dbt_postgres.postgres__drop_table", "macro_sql": "{% macro postgres__drop_table(relation) -%}\n drop table if exists {{ relation }} cascade\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3888068, "supported_languages": null}, "macro.dbt_postgres.postgres__get_replace_table_sql": {"name": "postgres__get_replace_table_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/table/replace.sql", "original_file_path": "macros/relations/table/replace.sql", "unique_id": "macro.dbt_postgres.postgres__get_replace_table_sql", "macro_sql": "{% macro postgres__get_replace_table_sql(relation, sql) -%}\n\n {%- set sql_header = config.get('sql_header', none) -%}\n {{ sql_header if sql_header is not none }}\n\n create or replace table {{ relation }}\n {% set contract_config = config.get('contract') %}\n {% if contract_config.enforced %}\n {{ get_assert_columns_equivalent(sql) }}\n {{ get_table_columns_and_constraints() }}\n {%- set sql = get_select_subquery(sql) %}\n {% endif %}\n as (\n {{ sql }}\n );\n\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.get_assert_columns_equivalent", "macro.dbt.get_table_columns_and_constraints", "macro.dbt.get_select_subquery"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3888152, "supported_languages": null}, "macro.dbt_postgres.postgres__get_rename_table_sql": {"name": "postgres__get_rename_table_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/table/rename.sql", "original_file_path": "macros/relations/table/rename.sql", "unique_id": "macro.dbt_postgres.postgres__get_rename_table_sql", "macro_sql": "{% macro postgres__get_rename_table_sql(relation, new_name) %}\n alter table {{ relation }} rename to {{ new_name }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3888252, "supported_languages": null}, "macro.dbt_postgres.postgres__drop_view": {"name": "postgres__drop_view", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/view/drop.sql", "original_file_path": "macros/relations/view/drop.sql", "unique_id": "macro.dbt_postgres.postgres__drop_view", "macro_sql": "{% macro postgres__drop_view(relation) -%}\n drop view if exists {{ relation }} cascade\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388834, "supported_languages": null}, "macro.dbt_postgres.postgres__get_replace_view_sql": {"name": "postgres__get_replace_view_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/view/replace.sql", "original_file_path": "macros/relations/view/replace.sql", "unique_id": "macro.dbt_postgres.postgres__get_replace_view_sql", "macro_sql": "{% macro postgres__get_replace_view_sql(relation, sql) -%}\n\n {%- set sql_header = config.get('sql_header', none) -%}\n {{ sql_header if sql_header is not none }}\n\n create or replace view {{ relation }}\n {% set contract_config = config.get('contract') %}\n {% if contract_config.enforced %}\n {{ get_assert_columns_equivalent(sql) }}\n {%- endif %}\n as (\n {{ sql }}\n );\n\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.get_assert_columns_equivalent"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388845, "supported_languages": null}, "macro.dbt_postgres.postgres__get_rename_view_sql": {"name": "postgres__get_rename_view_sql", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/relations/view/rename.sql", "original_file_path": "macros/relations/view/rename.sql", "unique_id": "macro.dbt_postgres.postgres__get_rename_view_sql", "macro_sql": "{% macro postgres__get_rename_view_sql(relation, new_name) %}\n alter view {{ relation }} rename to {{ new_name }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.38886, "supported_languages": null}, "macro.dbt_postgres.postgres__dateadd": {"name": "postgres__dateadd", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/utils/dateadd.sql", "original_file_path": "macros/utils/dateadd.sql", "unique_id": "macro.dbt_postgres.postgres__dateadd", "macro_sql": "{% macro postgres__dateadd(datepart, interval, from_date_or_timestamp) %}\n\n {{ from_date_or_timestamp }} + ((interval '1 {{ datepart }}') * ({{ interval }}))\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388871, "supported_languages": null}, "macro.dbt_postgres.postgres__listagg": {"name": "postgres__listagg", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/utils/listagg.sql", "original_file_path": "macros/utils/listagg.sql", "unique_id": "macro.dbt_postgres.postgres__listagg", "macro_sql": "{% macro postgres__listagg(measure, delimiter_text, order_by_clause, limit_num) -%}\n\n {% if limit_num -%}\n array_to_string(\n (array_agg(\n {{ measure }}\n {% if order_by_clause -%}\n {{ order_by_clause }}\n {%- endif %}\n ))[1:{{ limit_num }}],\n {{ delimiter_text }}\n )\n {%- else %}\n string_agg(\n {{ measure }},\n {{ delimiter_text }}\n {% if order_by_clause -%}\n {{ order_by_clause }}\n {%- endif %}\n )\n {%- endif %}\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388879, "supported_languages": null}, "macro.dbt_postgres.postgres__datediff": {"name": "postgres__datediff", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/utils/datediff.sql", "original_file_path": "macros/utils/datediff.sql", "unique_id": "macro.dbt_postgres.postgres__datediff", "macro_sql": "{% macro postgres__datediff(first_date, second_date, datepart) -%}\n\n {% if datepart == 'year' %}\n (date_part('year', ({{second_date}})::date) - date_part('year', ({{first_date}})::date))\n {% elif datepart == 'quarter' %}\n ({{ datediff(first_date, second_date, 'year') }} * 4 + date_part('quarter', ({{second_date}})::date) - date_part('quarter', ({{first_date}})::date))\n {% elif datepart == 'month' %}\n ({{ datediff(first_date, second_date, 'year') }} * 12 + date_part('month', ({{second_date}})::date) - date_part('month', ({{first_date}})::date))\n {% elif datepart == 'day' %}\n (({{second_date}})::date - ({{first_date}})::date)\n {% elif datepart == 'week' %}\n ({{ datediff(first_date, second_date, 'day') }} / 7 + case\n when date_part('dow', ({{first_date}})::timestamp) <= date_part('dow', ({{second_date}})::timestamp) then\n case when {{first_date}} <= {{second_date}} then 0 else -1 end\n else\n case when {{first_date}} <= {{second_date}} then 1 else 0 end\n end)\n {% elif datepart == 'hour' %}\n ({{ datediff(first_date, second_date, 'day') }} * 24 + date_part('hour', ({{second_date}})::timestamp) - date_part('hour', ({{first_date}})::timestamp))\n {% elif datepart == 'minute' %}\n ({{ datediff(first_date, second_date, 'hour') }} * 60 + date_part('minute', ({{second_date}})::timestamp) - date_part('minute', ({{first_date}})::timestamp))\n {% elif datepart == 'second' %}\n ({{ datediff(first_date, second_date, 'minute') }} * 60 + floor(date_part('second', ({{second_date}})::timestamp)) - floor(date_part('second', ({{first_date}})::timestamp)))\n {% elif datepart == 'millisecond' %}\n ({{ datediff(first_date, second_date, 'minute') }} * 60000 + floor(date_part('millisecond', ({{second_date}})::timestamp)) - floor(date_part('millisecond', ({{first_date}})::timestamp)))\n {% elif datepart == 'microsecond' %}\n ({{ datediff(first_date, second_date, 'minute') }} * 60000000 + floor(date_part('microsecond', ({{second_date}})::timestamp)) - floor(date_part('microsecond', ({{first_date}})::timestamp)))\n {% else %}\n {{ exceptions.raise_compiler_error(\"Unsupported datepart for macro datediff in postgres: {!r}\".format(datepart)) }}\n {% endif %}\n\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.datediff"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.38889, "supported_languages": null}, "macro.dbt_postgres.postgres__any_value": {"name": "postgres__any_value", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/utils/any_value.sql", "original_file_path": "macros/utils/any_value.sql", "unique_id": "macro.dbt_postgres.postgres__any_value", "macro_sql": "{% macro postgres__any_value(expression) -%}\n\n min({{ expression }})\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3888981, "supported_languages": null}, "macro.dbt_postgres.postgres__last_day": {"name": "postgres__last_day", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/utils/last_day.sql", "original_file_path": "macros/utils/last_day.sql", "unique_id": "macro.dbt_postgres.postgres__last_day", "macro_sql": "{% macro postgres__last_day(date, datepart) -%}\n\n {%- if datepart == 'quarter' -%}\n -- postgres dateadd does not support quarter interval.\n cast(\n {{dbt.dateadd('day', '-1',\n dbt.dateadd('month', '3', dbt.date_trunc(datepart, date))\n )}}\n as date)\n {%- else -%}\n {{dbt.default_last_day(date, datepart)}}\n {%- endif -%}\n\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.dateadd", "macro.dbt.date_trunc", "macro.dbt.default_last_day"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388906, "supported_languages": null}, "macro.dbt_postgres.postgres__split_part": {"name": "postgres__split_part", "resource_type": "macro", "package_name": "dbt_postgres", "path": "macros/utils/split_part.sql", "original_file_path": "macros/utils/split_part.sql", "unique_id": "macro.dbt_postgres.postgres__split_part", "macro_sql": "{% macro postgres__split_part(string_text, delimiter_text, part_number) %}\n\n {% if part_number >= 0 %}\n {{ dbt.default__split_part(string_text, delimiter_text, part_number) }}\n {% else %}\n {{ dbt._split_part_negative(string_text, delimiter_text, part_number) }}\n {% endif %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__split_part", "macro.dbt._split_part_negative"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388918, "supported_languages": null}, "macro.dbt.run_hooks": {"name": "run_hooks", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/hooks.sql", "original_file_path": "macros/materializations/hooks.sql", "unique_id": "macro.dbt.run_hooks", "macro_sql": "{% macro run_hooks(hooks, inside_transaction=True) %}\n {% for hook in hooks | selectattr('transaction', 'equalto', inside_transaction) %}\n {% if not inside_transaction and loop.first %}\n {% call statement(auto_begin=inside_transaction) %}\n commit;\n {% endcall %}\n {% endif %}\n {% set rendered = render(hook.get('sql')) | trim %}\n {% if (rendered | length) > 0 %}\n {% call statement(auto_begin=inside_transaction) %}\n {{ rendered }}\n {% endcall %}\n {% endif %}\n {% endfor %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388928, "supported_languages": null}, "macro.dbt.make_hook_config": {"name": "make_hook_config", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/hooks.sql", "original_file_path": "macros/materializations/hooks.sql", "unique_id": "macro.dbt.make_hook_config", "macro_sql": "{% macro make_hook_config(sql, inside_transaction) %}\n {{ tojson({\"sql\": sql, \"transaction\": inside_transaction}) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388934, "supported_languages": null}, "macro.dbt.before_begin": {"name": "before_begin", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/hooks.sql", "original_file_path": "macros/materializations/hooks.sql", "unique_id": "macro.dbt.before_begin", "macro_sql": "{% macro before_begin(sql) %}\n {{ make_hook_config(sql, inside_transaction=False) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.make_hook_config"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388938, "supported_languages": null}, "macro.dbt.in_transaction": {"name": "in_transaction", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/hooks.sql", "original_file_path": "macros/materializations/hooks.sql", "unique_id": "macro.dbt.in_transaction", "macro_sql": "{% macro in_transaction(sql) %}\n {{ make_hook_config(sql, inside_transaction=True) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.make_hook_config"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3889449, "supported_languages": null}, "macro.dbt.after_commit": {"name": "after_commit", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/hooks.sql", "original_file_path": "macros/materializations/hooks.sql", "unique_id": "macro.dbt.after_commit", "macro_sql": "{% macro after_commit(sql) %}\n {{ make_hook_config(sql, inside_transaction=False) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.make_hook_config"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.38895, "supported_languages": null}, "macro.dbt.set_sql_header": {"name": "set_sql_header", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/configs.sql", "original_file_path": "macros/materializations/configs.sql", "unique_id": "macro.dbt.set_sql_header", "macro_sql": "{% macro set_sql_header(config) -%}\n {{ config.set('sql_header', caller()) }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388958, "supported_languages": null}, "macro.dbt.should_full_refresh": {"name": "should_full_refresh", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/configs.sql", "original_file_path": "macros/materializations/configs.sql", "unique_id": "macro.dbt.should_full_refresh", "macro_sql": "{% macro should_full_refresh() %}\n {% set config_full_refresh = config.get('full_refresh') %}\n {% if config_full_refresh is none %}\n {% set config_full_refresh = flags.FULL_REFRESH %}\n {% endif %}\n {% do return(config_full_refresh) %}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388963, "supported_languages": null}, "macro.dbt.should_store_failures": {"name": "should_store_failures", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/configs.sql", "original_file_path": "macros/materializations/configs.sql", "unique_id": "macro.dbt.should_store_failures", "macro_sql": "{% macro should_store_failures() %}\n {% set config_store_failures = config.get('store_failures') %}\n {% if config_store_failures is none %}\n {% set config_store_failures = flags.STORE_FAILURES %}\n {% endif %}\n {% do return(config_store_failures) %}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388971, "supported_languages": null}, "macro.dbt.snapshot_merge_sql": {"name": "snapshot_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/snapshot_merge.sql", "original_file_path": "macros/materializations/snapshots/snapshot_merge.sql", "unique_id": "macro.dbt.snapshot_merge_sql", "macro_sql": "{% macro snapshot_merge_sql(target, source, insert_cols) -%}\n {{ adapter.dispatch('snapshot_merge_sql', 'dbt')(target, source, insert_cols) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__snapshot_merge_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.388981, "supported_languages": null}, "macro.dbt.default__snapshot_merge_sql": {"name": "default__snapshot_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/snapshot_merge.sql", "original_file_path": "macros/materializations/snapshots/snapshot_merge.sql", "unique_id": "macro.dbt.default__snapshot_merge_sql", "macro_sql": "{% macro default__snapshot_merge_sql(target, source, insert_cols) -%}\n {%- set insert_cols_csv = insert_cols | join(', ') -%}\n\n {%- set columns = config.get(\"snapshot_table_column_names\") or get_snapshot_table_column_names() -%}\n\n merge into {{ target.render() }} as DBT_INTERNAL_DEST\n using {{ source }} as DBT_INTERNAL_SOURCE\n on DBT_INTERNAL_SOURCE.{{ columns.dbt_scd_id }} = DBT_INTERNAL_DEST.{{ columns.dbt_scd_id }}\n\n when matched\n {% if config.get(\"dbt_valid_to_current\") %}\n and (DBT_INTERNAL_DEST.{{ columns.dbt_valid_to }} = {{ config.get('dbt_valid_to_current') }} or\n DBT_INTERNAL_DEST.{{ columns.dbt_valid_to }} is null)\n {% else %}\n and DBT_INTERNAL_DEST.{{ columns.dbt_valid_to }} is null\n {% endif %}\n and DBT_INTERNAL_SOURCE.dbt_change_type in ('update', 'delete')\n then update\n set {{ columns.dbt_valid_to }} = DBT_INTERNAL_SOURCE.{{ columns.dbt_valid_to }}\n\n when not matched\n and DBT_INTERNAL_SOURCE.dbt_change_type = 'insert'\n then insert ({{ insert_cols_csv }})\n values ({{ insert_cols_csv }})\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_snapshot_table_column_names"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.38899, "supported_languages": null}, "macro.dbt.strategy_dispatch": {"name": "strategy_dispatch", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/strategies.sql", "original_file_path": "macros/materializations/snapshots/strategies.sql", "unique_id": "macro.dbt.strategy_dispatch", "macro_sql": "{% macro strategy_dispatch(name) -%}\n{% set original_name = name %}\n {% if '.' in name %}\n {% set package_name, name = name.split(\".\", 1) %}\n {% else %}\n {% set package_name = none %}\n {% endif %}\n\n {% if package_name is none %}\n {% set package_context = context %}\n {% elif package_name in context %}\n {% set package_context = context[package_name] %}\n {% else %}\n {% set error_msg %}\n Could not find package '{{package_name}}', called with '{{original_name}}'\n {% endset %}\n {{ exceptions.raise_compiler_error(error_msg | trim) }}\n {% endif %}\n\n {%- set search_name = 'snapshot_' ~ name ~ '_strategy' -%}\n\n {% if search_name not in package_context %}\n {% set error_msg %}\n The specified strategy macro '{{name}}' was not found in package '{{ package_name }}'\n {% endset %}\n {{ exceptions.raise_compiler_error(error_msg | trim) }}\n {% endif %}\n {{ return(package_context[search_name]) }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389009, "supported_languages": null}, "macro.dbt.snapshot_hash_arguments": {"name": "snapshot_hash_arguments", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/strategies.sql", "original_file_path": "macros/materializations/snapshots/strategies.sql", "unique_id": "macro.dbt.snapshot_hash_arguments", "macro_sql": "{% macro snapshot_hash_arguments(args) -%}\n {{ adapter.dispatch('snapshot_hash_arguments', 'dbt')(args) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__snapshot_hash_arguments"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389017, "supported_languages": null}, "macro.dbt.default__snapshot_hash_arguments": {"name": "default__snapshot_hash_arguments", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/strategies.sql", "original_file_path": "macros/materializations/snapshots/strategies.sql", "unique_id": "macro.dbt.default__snapshot_hash_arguments", "macro_sql": "{% macro default__snapshot_hash_arguments(args) -%}\n md5({%- for arg in args -%}\n coalesce(cast({{ arg }} as varchar ), '')\n {% if not loop.last %} || '|' || {% endif %}\n {%- endfor -%})\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389024, "supported_languages": null}, "macro.dbt.snapshot_timestamp_strategy": {"name": "snapshot_timestamp_strategy", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/strategies.sql", "original_file_path": "macros/materializations/snapshots/strategies.sql", "unique_id": "macro.dbt.snapshot_timestamp_strategy", "macro_sql": "{% macro snapshot_timestamp_strategy(node, snapshotted_rel, current_rel, model_config, target_exists) %}\n {# The model_config parameter is no longer used, but is passed in anyway for compatibility. #}\n {% set primary_key = config.get('unique_key') %}\n {% set updated_at = config.get('updated_at') %}\n {% set invalidate_hard_deletes = config.get('invalidate_hard_deletes') or false %}\n {% set columns = config.get(\"snapshot_table_column_names\") or get_snapshot_table_column_names() %}\n\n {#/*\n The snapshot relation might not have an {{ updated_at }} value if the\n snapshot strategy is changed from `check` to `timestamp`. We\n should use a dbt-created column for the comparison in the snapshot\n table instead of assuming that the user-supplied {{ updated_at }}\n will be present in the historical data.\n\n See https://github.com/dbt-labs/dbt-core/issues/2350\n */ #}\n {% set row_changed_expr -%}\n ({{ snapshotted_rel }}.{{ columns.dbt_valid_from }} < {{ current_rel }}.{{ updated_at }})\n {%- endset %}\n\n {% set scd_id_expr = snapshot_hash_arguments([primary_key, updated_at]) %}\n\n {% do return({\n \"unique_key\": primary_key,\n \"updated_at\": updated_at,\n \"row_changed\": row_changed_expr,\n \"scd_id\": scd_id_expr,\n \"invalidate_hard_deletes\": invalidate_hard_deletes\n }) %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_snapshot_table_column_names", "macro.dbt.snapshot_hash_arguments"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389028, "supported_languages": null}, "macro.dbt.snapshot_string_as_time": {"name": "snapshot_string_as_time", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/strategies.sql", "original_file_path": "macros/materializations/snapshots/strategies.sql", "unique_id": "macro.dbt.snapshot_string_as_time", "macro_sql": "{% macro snapshot_string_as_time(timestamp) -%}\n {{ adapter.dispatch('snapshot_string_as_time', 'dbt')(timestamp) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__snapshot_string_as_time"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389034, "supported_languages": null}, "macro.dbt.default__snapshot_string_as_time": {"name": "default__snapshot_string_as_time", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/strategies.sql", "original_file_path": "macros/materializations/snapshots/strategies.sql", "unique_id": "macro.dbt.default__snapshot_string_as_time", "macro_sql": "{% macro default__snapshot_string_as_time(timestamp) %}\n {% do exceptions.raise_not_implemented(\n 'snapshot_string_as_time macro not implemented for adapter '+adapter.type()\n ) %}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389039, "supported_languages": null}, "macro.dbt.snapshot_check_all_get_existing_columns": {"name": "snapshot_check_all_get_existing_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/strategies.sql", "original_file_path": "macros/materializations/snapshots/strategies.sql", "unique_id": "macro.dbt.snapshot_check_all_get_existing_columns", "macro_sql": "{% macro snapshot_check_all_get_existing_columns(node, target_exists, check_cols_config) -%}\n {%- if not target_exists -%}\n {#-- no table yet -> return whatever the query does --#}\n {{ return((false, query_columns)) }}\n {%- endif -%}\n\n {#-- handle any schema changes --#}\n {%- set target_relation = adapter.get_relation(database=node.database, schema=node.schema, identifier=node.alias) -%}\n\n {% if check_cols_config == 'all' %}\n {%- set query_columns = get_columns_in_query(node['compiled_code']) -%}\n\n {% elif check_cols_config is iterable and (check_cols_config | length) > 0 %}\n {#-- query for proper casing/quoting, to support comparison below --#}\n {%- set select_check_cols_from_target -%}\n {#-- N.B. The whitespace below is necessary to avoid edge case issue with comments --#}\n {#-- See: https://github.com/dbt-labs/dbt-core/issues/6781 --#}\n select {{ check_cols_config | join(', ') }} from (\n {{ node['compiled_code'] }}\n ) subq\n {%- endset -%}\n {% set query_columns = get_columns_in_query(select_check_cols_from_target) %}\n\n {% else %}\n {% do exceptions.raise_compiler_error(\"Invalid value for 'check_cols': \" ~ check_cols_config) %}\n {% endif %}\n\n {%- set existing_cols = adapter.get_columns_in_relation(target_relation) | map(attribute = 'name') | list -%}\n {%- set ns = namespace() -%} {#-- handle for-loop scoping with a namespace --#}\n {%- set ns.column_added = false -%}\n\n {%- set intersection = [] -%}\n {%- for col in query_columns -%}\n {%- if col in existing_cols -%}\n {%- do intersection.append(adapter.quote(col)) -%}\n {%- else -%}\n {% set ns.column_added = true %}\n {%- endif -%}\n {%- endfor -%}\n {{ return((ns.column_added, intersection)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.get_columns_in_query"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3890471, "supported_languages": null}, "macro.dbt.snapshot_check_strategy": {"name": "snapshot_check_strategy", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/strategies.sql", "original_file_path": "macros/materializations/snapshots/strategies.sql", "unique_id": "macro.dbt.snapshot_check_strategy", "macro_sql": "{% macro snapshot_check_strategy(node, snapshotted_rel, current_rel, model_config, target_exists) %}\n {# The model_config parameter is no longer used, but is passed in anyway for compatibility. #}\n {% set check_cols_config = config.get('check_cols') %}\n {% set primary_key = config.get('unique_key') %}\n {% set invalidate_hard_deletes = config.get('invalidate_hard_deletes') or false %}\n {% set updated_at = config.get('updated_at') or snapshot_get_time() %}\n\n {% set column_added = false %}\n\n {% set column_added, check_cols = snapshot_check_all_get_existing_columns(node, target_exists, check_cols_config) %}\n\n {%- set row_changed_expr -%}\n (\n {%- if column_added -%}\n {{ get_true_sql() }}\n {%- else -%}\n {%- for col in check_cols -%}\n {{ snapshotted_rel }}.{{ col }} != {{ current_rel }}.{{ col }}\n or\n (\n (({{ snapshotted_rel }}.{{ col }} is null) and not ({{ current_rel }}.{{ col }} is null))\n or\n ((not {{ snapshotted_rel }}.{{ col }} is null) and ({{ current_rel }}.{{ col }} is null))\n )\n {%- if not loop.last %} or {% endif -%}\n {%- endfor -%}\n {%- endif -%}\n )\n {%- endset %}\n\n {% set scd_id_expr = snapshot_hash_arguments([primary_key, updated_at]) %}\n\n {% do return({\n \"unique_key\": primary_key,\n \"updated_at\": updated_at,\n \"row_changed\": row_changed_expr,\n \"scd_id\": scd_id_expr,\n \"invalidate_hard_deletes\": invalidate_hard_deletes\n }) %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.snapshot_get_time", "macro.dbt.snapshot_check_all_get_existing_columns", "macro.dbt.get_true_sql", "macro.dbt.snapshot_hash_arguments"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3890612, "supported_languages": null}, "macro.dbt.create_columns": {"name": "create_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.create_columns", "macro_sql": "{% macro create_columns(relation, columns) %}\n {{ adapter.dispatch('create_columns', 'dbt')(relation, columns) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__create_columns"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389078, "supported_languages": null}, "macro.dbt.default__create_columns": {"name": "default__create_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.default__create_columns", "macro_sql": "{% macro default__create_columns(relation, columns) %}\n {% for column in columns %}\n {% call statement() %}\n alter table {{ relation.render() }} add column \"{{ column.name }}\" {{ column.data_type }};\n {% endcall %}\n {% endfor %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3890822, "supported_languages": null}, "macro.dbt.post_snapshot": {"name": "post_snapshot", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.post_snapshot", "macro_sql": "{% macro post_snapshot(staging_relation) %}\n {{ adapter.dispatch('post_snapshot', 'dbt')(staging_relation) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__post_snapshot"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.38909, "supported_languages": null}, "macro.dbt.default__post_snapshot": {"name": "default__post_snapshot", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.default__post_snapshot", "macro_sql": "{% macro default__post_snapshot(staging_relation) %}\n {# no-op #}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3890948, "supported_languages": null}, "macro.dbt.get_true_sql": {"name": "get_true_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.get_true_sql", "macro_sql": "{% macro get_true_sql() %}\n {{ adapter.dispatch('get_true_sql', 'dbt')() }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_true_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3891, "supported_languages": null}, "macro.dbt.default__get_true_sql": {"name": "default__get_true_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.default__get_true_sql", "macro_sql": "{% macro default__get_true_sql() %}\n {{ return('TRUE') }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3891032, "supported_languages": null}, "macro.dbt.snapshot_staging_table": {"name": "snapshot_staging_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.snapshot_staging_table", "macro_sql": "{% macro snapshot_staging_table(strategy, source_sql, target_relation) -%}\n {{ adapter.dispatch('snapshot_staging_table', 'dbt')(strategy, source_sql, target_relation) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__snapshot_staging_table"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389107, "supported_languages": null}, "macro.dbt.get_snapshot_table_column_names": {"name": "get_snapshot_table_column_names", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.get_snapshot_table_column_names", "macro_sql": "{% macro get_snapshot_table_column_names() %}\n {{ return({'dbt_valid_to': 'dbt_valid_to', 'dbt_valid_from': 'dbt_valid_from', 'dbt_scd_id': 'dbt_scd_id', 'dbt_updated_at': 'dbt_updated_at'}) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3891149, "supported_languages": null}, "macro.dbt.default__snapshot_staging_table": {"name": "default__snapshot_staging_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.default__snapshot_staging_table", "macro_sql": "{% macro default__snapshot_staging_table(strategy, source_sql, target_relation) -%}\n {% set columns = config.get('snapshot_table_column_names') or get_snapshot_table_column_names() %}\n\n with snapshot_query as (\n\n {{ source_sql }}\n\n ),\n\n snapshotted_data as (\n\n select *,\n {{ strategy.unique_key }} as dbt_unique_key\n\n from {{ target_relation }}\n where\n {% if config.get('dbt_valid_to_current') %}\n {# Check for either dbt_valid_to_current OR null, in order to correctly update records with nulls #}\n ( {{ columns.dbt_valid_to }} = {{ config.get('dbt_valid_to_current') }} or {{ columns.dbt_valid_to }} is null)\n {% else %}\n {{ columns.dbt_valid_to }} is null\n {% endif %}\n\n ),\n\n insertions_source_data as (\n\n select\n *,\n {{ strategy.unique_key }} as dbt_unique_key,\n {{ strategy.updated_at }} as {{ columns.dbt_updated_at }},\n {{ strategy.updated_at }} as {{ columns.dbt_valid_from }},\n {{ get_dbt_valid_to_current(strategy, columns) }},\n {{ strategy.scd_id }} as {{ columns.dbt_scd_id }}\n\n from snapshot_query\n ),\n\n updates_source_data as (\n\n select\n *,\n {{ strategy.unique_key }} as dbt_unique_key,\n {{ strategy.updated_at }} as {{ columns.dbt_updated_at }},\n {{ strategy.updated_at }} as {{ columns.dbt_valid_from }},\n {{ strategy.updated_at }} as {{ columns.dbt_valid_to }}\n\n from snapshot_query\n ),\n\n {%- if strategy.invalidate_hard_deletes %}\n\n deletes_source_data as (\n\n select\n *,\n {{ strategy.unique_key }} as dbt_unique_key\n from snapshot_query\n ),\n {% endif %}\n\n insertions as (\n\n select\n 'insert' as dbt_change_type,\n source_data.*\n\n from insertions_source_data as source_data\n left outer join snapshotted_data on snapshotted_data.dbt_unique_key = source_data.dbt_unique_key\n where snapshotted_data.dbt_unique_key is null\n or (\n snapshotted_data.dbt_unique_key is not null\n and (\n {{ strategy.row_changed }}\n )\n )\n\n ),\n\n updates as (\n\n select\n 'update' as dbt_change_type,\n source_data.*,\n snapshotted_data.{{ columns.dbt_scd_id }}\n\n from updates_source_data as source_data\n join snapshotted_data on snapshotted_data.dbt_unique_key = source_data.dbt_unique_key\n where (\n {{ strategy.row_changed }}\n )\n )\n\n {%- if strategy.invalidate_hard_deletes -%}\n ,\n\n deletes as (\n\n select\n 'delete' as dbt_change_type,\n source_data.*,\n {{ snapshot_get_time() }} as {{ columns.dbt_valid_from }},\n {{ snapshot_get_time() }} as {{ columns.dbt_updated_at }},\n {{ snapshot_get_time() }} as {{ columns.dbt_valid_to }},\n snapshotted_data.{{ columns.dbt_scd_id }}\n\n from snapshotted_data\n left join deletes_source_data as source_data on snapshotted_data.dbt_unique_key = source_data.dbt_unique_key\n where source_data.dbt_unique_key is null\n )\n {%- endif %}\n\n select * from insertions\n union all\n select * from updates\n {%- if strategy.invalidate_hard_deletes %}\n union all\n select * from deletes\n {%- endif %}\n\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.get_snapshot_table_column_names", "macro.dbt.get_dbt_valid_to_current", "macro.dbt.snapshot_get_time"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389138, "supported_languages": null}, "macro.dbt.build_snapshot_table": {"name": "build_snapshot_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.build_snapshot_table", "macro_sql": "{% macro build_snapshot_table(strategy, sql) -%}\n {{ adapter.dispatch('build_snapshot_table', 'dbt')(strategy, sql) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__build_snapshot_table"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3891442, "supported_languages": null}, "macro.dbt.default__build_snapshot_table": {"name": "default__build_snapshot_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.default__build_snapshot_table", "macro_sql": "{% macro default__build_snapshot_table(strategy, sql) %}\n {% set columns = config.get('snapshot_table_column_names') or get_snapshot_table_column_names() %}\n\n select *,\n {{ strategy.scd_id }} as {{ columns.dbt_scd_id }},\n {{ strategy.updated_at }} as {{ columns.dbt_updated_at }},\n {{ strategy.updated_at }} as {{ columns.dbt_valid_from }},\n {{ get_dbt_valid_to_current(strategy, columns) }}\n from (\n {{ sql }}\n ) sbq\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_snapshot_table_column_names", "macro.dbt.get_dbt_valid_to_current"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.38915, "supported_languages": null}, "macro.dbt.build_snapshot_staging_table": {"name": "build_snapshot_staging_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.build_snapshot_staging_table", "macro_sql": "{% macro build_snapshot_staging_table(strategy, sql, target_relation) %}\n {% set temp_relation = make_temp_relation(target_relation) %}\n\n {% set select = snapshot_staging_table(strategy, sql, target_relation) %}\n\n {% call statement('build_snapshot_staging_relation') %}\n {{ create_table_as(True, temp_relation, select) }}\n {% endcall %}\n\n {% do return(temp_relation) %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.make_temp_relation", "macro.dbt.snapshot_staging_table", "macro.dbt.statement", "macro.dbt.create_table_as"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389154, "supported_languages": null}, "macro.dbt.get_updated_at_column_data_type": {"name": "get_updated_at_column_data_type", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.get_updated_at_column_data_type", "macro_sql": "{% macro get_updated_at_column_data_type(snapshot_sql) %}\n {% set snapshot_sql_column_schema = get_column_schema_from_query(snapshot_sql) %}\n {% set dbt_updated_at_data_type = null %}\n {% set ns = namespace() -%} {#-- handle for-loop scoping with a namespace --#}\n {% set ns.dbt_updated_at_data_type = null -%}\n {% for column in snapshot_sql_column_schema %}\n {% if ((column.column == 'dbt_updated_at') or (column.column == 'DBT_UPDATED_AT')) %}\n {% set ns.dbt_updated_at_data_type = column.dtype %}\n {% endif %}\n {% endfor %}\n {{ return(ns.dbt_updated_at_data_type or none) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_column_schema_from_query"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3891578, "supported_languages": null}, "macro.dbt.check_time_data_types": {"name": "check_time_data_types", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.check_time_data_types", "macro_sql": "{% macro check_time_data_types(sql) %}\n {% set dbt_updated_at_data_type = get_updated_at_column_data_type(sql) %}\n {% set snapshot_get_time_data_type = get_snapshot_get_time_data_type() %}\n {% if snapshot_get_time_data_type is not none and dbt_updated_at_data_type is not none and snapshot_get_time_data_type != dbt_updated_at_data_type %}\n {% if exceptions.warn_snapshot_timestamp_data_types %}\n {{ exceptions.warn_snapshot_timestamp_data_types(snapshot_get_time_data_type, dbt_updated_at_data_type) }}\n {% endif %}\n {% endif %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_updated_at_column_data_type", "macro.dbt.get_snapshot_get_time_data_type"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389166, "supported_languages": null}, "macro.dbt.get_dbt_valid_to_current": {"name": "get_dbt_valid_to_current", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/helpers.sql", "original_file_path": "macros/materializations/snapshots/helpers.sql", "unique_id": "macro.dbt.get_dbt_valid_to_current", "macro_sql": "{% macro get_dbt_valid_to_current(strategy, columns) %}\n {% set dbt_valid_to_current = config.get('dbt_valid_to_current') or \"null\" %}\n coalesce(nullif({{ strategy.updated_at }}, {{ strategy.updated_at }}), {{dbt_valid_to_current}})\n as {{ columns.dbt_valid_to }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389173, "supported_languages": null}, "macro.dbt.materialization_snapshot_default": {"name": "materialization_snapshot_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/snapshots/snapshot.sql", "original_file_path": "macros/materializations/snapshots/snapshot.sql", "unique_id": "macro.dbt.materialization_snapshot_default", "macro_sql": "{% materialization snapshot, default %}\n\n {%- set target_table = model.get('alias', model.get('name')) -%}\n\n {%- set strategy_name = config.get('strategy') -%}\n {%- set unique_key = config.get('unique_key') %}\n -- grab current tables grants config for comparision later on\n {%- set grant_config = config.get('grants') -%}\n\n {% set target_relation_exists, target_relation = get_or_create_relation(\n database=model.database,\n schema=model.schema,\n identifier=target_table,\n type='table') -%}\n\n {%- if not target_relation.is_table -%}\n {% do exceptions.relation_wrong_type(target_relation, 'table') %}\n {%- endif -%}\n\n\n {{ run_hooks(pre_hooks, inside_transaction=False) }}\n\n {{ run_hooks(pre_hooks, inside_transaction=True) }}\n\n {% set strategy_macro = strategy_dispatch(strategy_name) %}\n {# The model['config'] parameter below is no longer used, but passing anyway for compatibility #}\n {# It was a dictionary of config, instead of the config object from the context #}\n {% set strategy = strategy_macro(model, \"snapshotted_data\", \"source_data\", model['config'], target_relation_exists) %}\n\n {% if not target_relation_exists %}\n\n {% set build_sql = build_snapshot_table(strategy, model['compiled_code']) %}\n {% set build_or_select_sql = build_sql %}\n {% set final_sql = create_table_as(False, target_relation, build_sql) %}\n\n {% else %}\n\n {% set columns = config.get(\"snapshot_table_column_names\") or get_snapshot_table_column_names() %}\n\n {{ adapter.valid_snapshot_target(target_relation, columns) }}\n\n {% set build_or_select_sql = snapshot_staging_table(strategy, sql, target_relation) %}\n {% set staging_table = build_snapshot_staging_table(strategy, sql, target_relation) %}\n\n -- this may no-op if the database does not require column expansion\n {% do adapter.expand_target_column_types(from_relation=staging_table,\n to_relation=target_relation) %}\n\n {% set missing_columns = adapter.get_missing_columns(staging_table, target_relation)\n | rejectattr('name', 'equalto', 'dbt_change_type')\n | rejectattr('name', 'equalto', 'DBT_CHANGE_TYPE')\n | rejectattr('name', 'equalto', 'dbt_unique_key')\n | rejectattr('name', 'equalto', 'DBT_UNIQUE_KEY')\n | list %}\n\n {% do create_columns(target_relation, missing_columns) %}\n\n {% set source_columns = adapter.get_columns_in_relation(staging_table)\n | rejectattr('name', 'equalto', 'dbt_change_type')\n | rejectattr('name', 'equalto', 'DBT_CHANGE_TYPE')\n | rejectattr('name', 'equalto', 'dbt_unique_key')\n | rejectattr('name', 'equalto', 'DBT_UNIQUE_KEY')\n | list %}\n\n {% set quoted_source_columns = [] %}\n {% for column in source_columns %}\n {% do quoted_source_columns.append(adapter.quote(column.name)) %}\n {% endfor %}\n\n {% set final_sql = snapshot_merge_sql(\n target = target_relation,\n source = staging_table,\n insert_cols = quoted_source_columns\n )\n %}\n\n {% endif %}\n\n\n {{ check_time_data_types(build_or_select_sql) }}\n\n {% call statement('main') %}\n {{ final_sql }}\n {% endcall %}\n\n {% set should_revoke = should_revoke(target_relation_exists, full_refresh_mode=False) %}\n {% do apply_grants(target_relation, grant_config, should_revoke=should_revoke) %}\n\n {% do persist_docs(target_relation, model) %}\n\n {% if not target_relation_exists %}\n {% do create_indexes(target_relation) %}\n {% endif %}\n\n {{ run_hooks(post_hooks, inside_transaction=True) }}\n\n {{ adapter.commit() }}\n\n {% if staging_table is defined %}\n {% do post_snapshot(staging_table) %}\n {% endif %}\n\n {{ run_hooks(post_hooks, inside_transaction=False) }}\n\n {{ return({'relations': [target_relation]}) }}\n\n{% endmaterialization %}", "depends_on": {"macros": ["macro.dbt.get_or_create_relation", "macro.dbt.run_hooks", "macro.dbt.strategy_dispatch", "macro.dbt.build_snapshot_table", "macro.dbt.create_table_as", "macro.dbt.get_snapshot_table_column_names", "macro.dbt.snapshot_staging_table", "macro.dbt.build_snapshot_staging_table", "macro.dbt.create_columns", "macro.dbt.snapshot_merge_sql", "macro.dbt.check_time_data_types", "macro.dbt.statement", "macro.dbt.should_revoke", "macro.dbt.apply_grants", "macro.dbt.persist_docs", "macro.dbt.create_indexes", "macro.dbt.post_snapshot"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3891861, "supported_languages": ["sql"]}, "macro.dbt.materialization_test_default": {"name": "materialization_test_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/tests/test.sql", "original_file_path": "macros/materializations/tests/test.sql", "unique_id": "macro.dbt.materialization_test_default", "macro_sql": "{%- materialization test, default -%}\n\n {% set relations = [] %}\n\n {% if should_store_failures() %}\n\n {% set identifier = model['alias'] %}\n {% set old_relation = adapter.get_relation(database=database, schema=schema, identifier=identifier) %}\n\n {% set store_failures_as = config.get('store_failures_as') %}\n -- if `--store-failures` is invoked via command line and `store_failures_as` is not set,\n -- config.get('store_failures_as', 'table') returns None, not 'table'\n {% if store_failures_as == none %}{% set store_failures_as = 'table' %}{% endif %}\n {% if store_failures_as not in ['table', 'view'] %}\n {{ exceptions.raise_compiler_error(\n \"'\" ~ store_failures_as ~ \"' is not a valid value for `store_failures_as`. \"\n \"Accepted values are: ['ephemeral', 'table', 'view']\"\n ) }}\n {% endif %}\n\n {% set target_relation = api.Relation.create(\n identifier=identifier, schema=schema, database=database, type=store_failures_as) -%} %}\n\n {% if old_relation %}\n {% do adapter.drop_relation(old_relation) %}\n {% endif %}\n\n {% call statement(auto_begin=True) %}\n {{ get_create_sql(target_relation, sql) }}\n {% endcall %}\n\n {% do relations.append(target_relation) %}\n\n {% set main_sql %}\n select *\n from {{ target_relation }}\n {% endset %}\n\n {{ adapter.commit() }}\n\n {% else %}\n\n {% set main_sql = sql %}\n\n {% endif %}\n\n {% set limit = config.get('limit') %}\n {% set fail_calc = config.get('fail_calc') %}\n {% set warn_if = config.get('warn_if') %}\n {% set error_if = config.get('error_if') %}\n\n {% call statement('main', fetch_result=True) -%}\n\n {{ get_test_sql(main_sql, fail_calc, warn_if, error_if, limit)}}\n\n {%- endcall %}\n\n {{ return({'relations': relations}) }}\n\n{%- endmaterialization -%}", "depends_on": {"macros": ["macro.dbt.should_store_failures", "macro.dbt.statement", "macro.dbt.get_create_sql", "macro.dbt.get_test_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389204, "supported_languages": ["sql"]}, "macro.dbt.get_test_sql": {"name": "get_test_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/tests/helpers.sql", "original_file_path": "macros/materializations/tests/helpers.sql", "unique_id": "macro.dbt.get_test_sql", "macro_sql": "{% macro get_test_sql(main_sql, fail_calc, warn_if, error_if, limit) -%}\n {{ adapter.dispatch('get_test_sql', 'dbt')(main_sql, fail_calc, warn_if, error_if, limit) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_test_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3892212, "supported_languages": null}, "macro.dbt.default__get_test_sql": {"name": "default__get_test_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/tests/helpers.sql", "original_file_path": "macros/materializations/tests/helpers.sql", "unique_id": "macro.dbt.default__get_test_sql", "macro_sql": "{% macro default__get_test_sql(main_sql, fail_calc, warn_if, error_if, limit) -%}\n select\n {{ fail_calc }} as failures,\n {{ fail_calc }} {{ warn_if }} as should_warn,\n {{ fail_calc }} {{ error_if }} as should_error\n from (\n {{ main_sql }}\n {{ \"limit \" ~ limit if limit != none }}\n ) dbt_internal_test\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389226, "supported_languages": null}, "macro.dbt.get_unit_test_sql": {"name": "get_unit_test_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/tests/helpers.sql", "original_file_path": "macros/materializations/tests/helpers.sql", "unique_id": "macro.dbt.get_unit_test_sql", "macro_sql": "{% macro get_unit_test_sql(main_sql, expected_fixture_sql, expected_column_names) -%}\n {{ adapter.dispatch('get_unit_test_sql', 'dbt')(main_sql, expected_fixture_sql, expected_column_names) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_unit_test_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.38923, "supported_languages": null}, "macro.dbt.default__get_unit_test_sql": {"name": "default__get_unit_test_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/tests/helpers.sql", "original_file_path": "macros/materializations/tests/helpers.sql", "unique_id": "macro.dbt.default__get_unit_test_sql", "macro_sql": "{% macro default__get_unit_test_sql(main_sql, expected_fixture_sql, expected_column_names) -%}\n-- Build actual result given inputs\nwith dbt_internal_unit_test_actual as (\n select\n {% for expected_column_name in expected_column_names %}{{expected_column_name}}{% if not loop.last -%},{% endif %}{%- endfor -%}, {{ dbt.string_literal(\"actual\") }} as {{ adapter.quote(\"actual_or_expected\") }}\n from (\n {{ main_sql }}\n ) _dbt_internal_unit_test_actual\n),\n-- Build expected result\ndbt_internal_unit_test_expected as (\n select\n {% for expected_column_name in expected_column_names %}{{expected_column_name}}{% if not loop.last -%}, {% endif %}{%- endfor -%}, {{ dbt.string_literal(\"expected\") }} as {{ adapter.quote(\"actual_or_expected\") }}\n from (\n {{ expected_fixture_sql }}\n ) _dbt_internal_unit_test_expected\n)\n-- Union actual and expected results\nselect * from dbt_internal_unit_test_actual\nunion all\nselect * from dbt_internal_unit_test_expected\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.string_literal"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389239, "supported_languages": null}, "macro.dbt.get_where_subquery": {"name": "get_where_subquery", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/tests/where_subquery.sql", "original_file_path": "macros/materializations/tests/where_subquery.sql", "unique_id": "macro.dbt.get_where_subquery", "macro_sql": "{% macro get_where_subquery(relation) -%}\n {% do return(adapter.dispatch('get_where_subquery', 'dbt')(relation)) %}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_where_subquery"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389247, "supported_languages": null}, "macro.dbt.default__get_where_subquery": {"name": "default__get_where_subquery", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/tests/where_subquery.sql", "original_file_path": "macros/materializations/tests/where_subquery.sql", "unique_id": "macro.dbt.default__get_where_subquery", "macro_sql": "{% macro default__get_where_subquery(relation) -%}\n {% set where = config.get('where', '') %}\n {% if where %}\n {%- set filtered -%}\n (select * from {{ relation }} where {{ where }}) dbt_subquery\n {%- endset -%}\n {% do return(filtered) %}\n {%- else -%}\n {% do return(relation) %}\n {%- endif -%}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389251, "supported_languages": null}, "macro.dbt.materialization_unit_default": {"name": "materialization_unit_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/tests/unit.sql", "original_file_path": "macros/materializations/tests/unit.sql", "unique_id": "macro.dbt.materialization_unit_default", "macro_sql": "{%- materialization unit, default -%}\n\n {% set relations = [] %}\n\n {% set expected_rows = config.get('expected_rows') %}\n {% set expected_sql = config.get('expected_sql') %}\n {% set tested_expected_column_names = expected_rows[0].keys() if (expected_rows | length ) > 0 else get_columns_in_query(sql) %} %}\n\n {%- set target_relation = this.incorporate(type='table') -%}\n {%- set temp_relation = make_temp_relation(target_relation)-%}\n {% do run_query(get_create_table_as_sql(True, temp_relation, get_empty_subquery_sql(sql))) %}\n {%- set columns_in_relation = adapter.get_columns_in_relation(temp_relation) -%}\n {%- set column_name_to_data_types = {} -%}\n {%- for column in columns_in_relation -%}\n {%- do column_name_to_data_types.update({column.name|lower: column.data_type}) -%}\n {%- endfor -%}\n\n {% if not expected_sql %}\n {% set expected_sql = get_expected_sql(expected_rows, column_name_to_data_types) %}\n {% endif %}\n {% set unit_test_sql = get_unit_test_sql(sql, expected_sql, tested_expected_column_names) %}\n\n {% call statement('main', fetch_result=True) -%}\n\n {{ unit_test_sql }}\n\n {%- endcall %}\n\n {% do adapter.drop_relation(temp_relation) %}\n\n {{ return({'relations': relations}) }}\n\n{%- endmaterialization -%}", "depends_on": {"macros": ["macro.dbt.get_columns_in_query", "macro.dbt.make_temp_relation", "macro.dbt.run_query", "macro.dbt.get_create_table_as_sql", "macro.dbt.get_empty_subquery_sql", "macro.dbt.get_expected_sql", "macro.dbt.get_unit_test_sql", "macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3892648, "supported_languages": ["sql"]}, "macro.dbt.materialization_materialized_view_default": {"name": "materialization_materialized_view_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/materialized_view.sql", "original_file_path": "macros/materializations/models/materialized_view.sql", "unique_id": "macro.dbt.materialization_materialized_view_default", "macro_sql": "{% materialization materialized_view, default %}\n {% set existing_relation = load_cached_relation(this) %}\n {% set target_relation = this.incorporate(type=this.MaterializedView) %}\n {% set intermediate_relation = make_intermediate_relation(target_relation) %}\n {% set backup_relation_type = target_relation.MaterializedView if existing_relation is none else existing_relation.type %}\n {% set backup_relation = make_backup_relation(target_relation, backup_relation_type) %}\n\n {{ materialized_view_setup(backup_relation, intermediate_relation, pre_hooks) }}\n\n {% set build_sql = materialized_view_get_build_sql(existing_relation, target_relation, backup_relation, intermediate_relation) %}\n\n {% if build_sql == '' %}\n {{ materialized_view_execute_no_op(target_relation) }}\n {% else %}\n {{ materialized_view_execute_build_sql(build_sql, existing_relation, target_relation, post_hooks) }}\n {% endif %}\n\n {{ materialized_view_teardown(backup_relation, intermediate_relation, post_hooks) }}\n\n {{ return({'relations': [target_relation]}) }}\n\n{% endmaterialization %}", "depends_on": {"macros": ["macro.dbt.load_cached_relation", "macro.dbt.make_intermediate_relation", "macro.dbt.make_backup_relation", "macro.dbt.materialized_view_setup", "macro.dbt.materialized_view_get_build_sql", "macro.dbt.materialized_view_execute_no_op", "macro.dbt.materialized_view_execute_build_sql", "macro.dbt.materialized_view_teardown"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3892832, "supported_languages": ["sql"]}, "macro.dbt.materialized_view_setup": {"name": "materialized_view_setup", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/materialized_view.sql", "original_file_path": "macros/materializations/models/materialized_view.sql", "unique_id": "macro.dbt.materialized_view_setup", "macro_sql": "{% macro materialized_view_setup(backup_relation, intermediate_relation, pre_hooks) %}\n\n -- backup_relation and intermediate_relation should not already exist in the database\n -- it's possible these exist because of a previous run that exited unexpectedly\n {% set preexisting_backup_relation = load_cached_relation(backup_relation) %}\n {% set preexisting_intermediate_relation = load_cached_relation(intermediate_relation) %}\n\n -- drop the temp relations if they exist already in the database\n {{ drop_relation_if_exists(preexisting_backup_relation) }}\n {{ drop_relation_if_exists(preexisting_intermediate_relation) }}\n\n {{ run_hooks(pre_hooks, inside_transaction=False) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.load_cached_relation", "macro.dbt.drop_relation_if_exists", "macro.dbt.run_hooks"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389296, "supported_languages": null}, "macro.dbt.materialized_view_teardown": {"name": "materialized_view_teardown", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/materialized_view.sql", "original_file_path": "macros/materializations/models/materialized_view.sql", "unique_id": "macro.dbt.materialized_view_teardown", "macro_sql": "{% macro materialized_view_teardown(backup_relation, intermediate_relation, post_hooks) %}\n\n -- drop the temp relations if they exist to leave the database clean for the next run\n {{ drop_relation_if_exists(backup_relation) }}\n {{ drop_relation_if_exists(intermediate_relation) }}\n\n {{ run_hooks(post_hooks, inside_transaction=False) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.drop_relation_if_exists", "macro.dbt.run_hooks"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389302, "supported_languages": null}, "macro.dbt.materialized_view_get_build_sql": {"name": "materialized_view_get_build_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/materialized_view.sql", "original_file_path": "macros/materializations/models/materialized_view.sql", "unique_id": "macro.dbt.materialized_view_get_build_sql", "macro_sql": "{% macro materialized_view_get_build_sql(existing_relation, target_relation, backup_relation, intermediate_relation) %}\n\n {% set full_refresh_mode = should_full_refresh() %}\n\n -- determine the scenario we're in: create, full_refresh, alter, refresh data\n {% if existing_relation is none %}\n {% set build_sql = get_create_materialized_view_as_sql(target_relation, sql) %}\n {% elif full_refresh_mode or not existing_relation.is_materialized_view %}\n {% set build_sql = get_replace_sql(existing_relation, target_relation, sql) %}\n {% else %}\n\n -- get config options\n {% set on_configuration_change = config.get('on_configuration_change') %}\n {% set configuration_changes = get_materialized_view_configuration_changes(existing_relation, config) %}\n\n {% if configuration_changes is none %}\n {% set build_sql = refresh_materialized_view(target_relation) %}\n\n {% elif on_configuration_change == 'apply' %}\n {% set build_sql = get_alter_materialized_view_as_sql(target_relation, configuration_changes, sql, existing_relation, backup_relation, intermediate_relation) %}\n {% elif on_configuration_change == 'continue' %}\n {% set build_sql = '' %}\n {{ exceptions.warn(\"Configuration changes were identified and `on_configuration_change` was set to `continue` for `\" ~ target_relation.render() ~ \"`\") }}\n {% elif on_configuration_change == 'fail' %}\n {{ exceptions.raise_fail_fast_error(\"Configuration changes were identified and `on_configuration_change` was set to `fail` for `\" ~ target_relation.render() ~ \"`\") }}\n\n {% else %}\n -- this only happens if the user provides a value other than `apply`, 'skip', 'fail'\n {{ exceptions.raise_compiler_error(\"Unexpected configuration scenario\") }}\n\n {% endif %}\n\n {% endif %}\n\n {% do return(build_sql) %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.should_full_refresh", "macro.dbt.get_create_materialized_view_as_sql", "macro.dbt.get_replace_sql", "macro.dbt.get_materialized_view_configuration_changes", "macro.dbt.refresh_materialized_view", "macro.dbt.get_alter_materialized_view_as_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.38931, "supported_languages": null}, "macro.dbt.materialized_view_execute_no_op": {"name": "materialized_view_execute_no_op", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/materialized_view.sql", "original_file_path": "macros/materializations/models/materialized_view.sql", "unique_id": "macro.dbt.materialized_view_execute_no_op", "macro_sql": "{% macro materialized_view_execute_no_op(target_relation) %}\n {% do store_raw_result(\n name=\"main\",\n message=\"skip \" ~ target_relation,\n code=\"skip\",\n rows_affected=\"-1\"\n ) %}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.38932, "supported_languages": null}, "macro.dbt.materialized_view_execute_build_sql": {"name": "materialized_view_execute_build_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/materialized_view.sql", "original_file_path": "macros/materializations/models/materialized_view.sql", "unique_id": "macro.dbt.materialized_view_execute_build_sql", "macro_sql": "{% macro materialized_view_execute_build_sql(build_sql, existing_relation, target_relation, post_hooks) %}\n\n -- `BEGIN` happens here:\n {{ run_hooks(pre_hooks, inside_transaction=True) }}\n\n {% set grant_config = config.get('grants') %}\n\n {% call statement(name=\"main\") %}\n {{ build_sql }}\n {% endcall %}\n\n {% set should_revoke = should_revoke(existing_relation, full_refresh_mode=True) %}\n {% do apply_grants(target_relation, grant_config, should_revoke=should_revoke) %}\n\n {% do persist_docs(target_relation, model) %}\n\n {{ run_hooks(post_hooks, inside_transaction=True) }}\n\n {{ adapter.commit() }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.run_hooks", "macro.dbt.statement", "macro.dbt.should_revoke", "macro.dbt.apply_grants", "macro.dbt.persist_docs"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3893309, "supported_languages": null}, "macro.dbt.materialization_view_default": {"name": "materialization_view_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/view.sql", "original_file_path": "macros/materializations/models/view.sql", "unique_id": "macro.dbt.materialization_view_default", "macro_sql": "{%- materialization view, default -%}\n\n {%- set existing_relation = load_cached_relation(this) -%}\n {%- set target_relation = this.incorporate(type='view') -%}\n {%- set intermediate_relation = make_intermediate_relation(target_relation) -%}\n\n -- the intermediate_relation should not already exist in the database; get_relation\n -- will return None in that case. Otherwise, we get a relation that we can drop\n -- later, before we try to use this name for the current operation\n {%- set preexisting_intermediate_relation = load_cached_relation(intermediate_relation) -%}\n /*\n This relation (probably) doesn't exist yet. If it does exist, it's a leftover from\n a previous run, and we're going to try to drop it immediately. At the end of this\n materialization, we're going to rename the \"existing_relation\" to this identifier,\n and then we're going to drop it. In order to make sure we run the correct one of:\n - drop view ...\n - drop table ...\n\n We need to set the type of this relation to be the type of the existing_relation, if it exists,\n or else \"view\" as a sane default if it does not. Note that if the existing_relation does not\n exist, then there is nothing to move out of the way and subsequentally drop. In that case,\n this relation will be effectively unused.\n */\n {%- set backup_relation_type = 'view' if existing_relation is none else existing_relation.type -%}\n {%- set backup_relation = make_backup_relation(target_relation, backup_relation_type) -%}\n -- as above, the backup_relation should not already exist\n {%- set preexisting_backup_relation = load_cached_relation(backup_relation) -%}\n -- grab current tables grants config for comparision later on\n {% set grant_config = config.get('grants') %}\n\n {{ run_hooks(pre_hooks, inside_transaction=False) }}\n\n -- drop the temp relations if they exist already in the database\n {{ drop_relation_if_exists(preexisting_intermediate_relation) }}\n {{ drop_relation_if_exists(preexisting_backup_relation) }}\n\n -- `BEGIN` happens here:\n {{ run_hooks(pre_hooks, inside_transaction=True) }}\n\n -- build model\n {% call statement('main') -%}\n {{ get_create_view_as_sql(intermediate_relation, sql) }}\n {%- endcall %}\n\n -- cleanup\n -- move the existing view out of the way\n {% if existing_relation is not none %}\n /* Do the equivalent of rename_if_exists. 'existing_relation' could have been dropped\n since the variable was first set. */\n {% set existing_relation = load_cached_relation(existing_relation) %}\n {% if existing_relation is not none %}\n {{ adapter.rename_relation(existing_relation, backup_relation) }}\n {% endif %}\n {% endif %}\n {{ adapter.rename_relation(intermediate_relation, target_relation) }}\n\n {% set should_revoke = should_revoke(existing_relation, full_refresh_mode=True) %}\n {% do apply_grants(target_relation, grant_config, should_revoke=should_revoke) %}\n\n {% do persist_docs(target_relation, model) %}\n\n {{ run_hooks(post_hooks, inside_transaction=True) }}\n\n {{ adapter.commit() }}\n\n {{ drop_relation_if_exists(backup_relation) }}\n\n {{ run_hooks(post_hooks, inside_transaction=False) }}\n\n {{ return({'relations': [target_relation]}) }}\n\n{%- endmaterialization -%}", "depends_on": {"macros": ["macro.dbt.load_cached_relation", "macro.dbt.make_intermediate_relation", "macro.dbt.make_backup_relation", "macro.dbt.run_hooks", "macro.dbt.drop_relation_if_exists", "macro.dbt.statement", "macro.dbt.get_create_view_as_sql", "macro.dbt.should_revoke", "macro.dbt.apply_grants", "macro.dbt.persist_docs"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389365, "supported_languages": ["sql"]}, "macro.dbt.materialization_table_default": {"name": "materialization_table_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/table.sql", "original_file_path": "macros/materializations/models/table.sql", "unique_id": "macro.dbt.materialization_table_default", "macro_sql": "{% materialization table, default %}\n\n {%- set existing_relation = load_cached_relation(this) -%}\n {%- set target_relation = this.incorporate(type='table') %}\n {%- set intermediate_relation = make_intermediate_relation(target_relation) -%}\n -- the intermediate_relation should not already exist in the database; get_relation\n -- will return None in that case. Otherwise, we get a relation that we can drop\n -- later, before we try to use this name for the current operation\n {%- set preexisting_intermediate_relation = load_cached_relation(intermediate_relation) -%}\n /*\n See ../view/view.sql for more information about this relation.\n */\n {%- set backup_relation_type = 'table' if existing_relation is none else existing_relation.type -%}\n {%- set backup_relation = make_backup_relation(target_relation, backup_relation_type) -%}\n -- as above, the backup_relation should not already exist\n {%- set preexisting_backup_relation = load_cached_relation(backup_relation) -%}\n -- grab current tables grants config for comparision later on\n {% set grant_config = config.get('grants') %}\n\n -- drop the temp relations if they exist already in the database\n {{ drop_relation_if_exists(preexisting_intermediate_relation) }}\n {{ drop_relation_if_exists(preexisting_backup_relation) }}\n\n {{ run_hooks(pre_hooks, inside_transaction=False) }}\n\n -- `BEGIN` happens here:\n {{ run_hooks(pre_hooks, inside_transaction=True) }}\n\n -- build model\n {% call statement('main') -%}\n {{ get_create_table_as_sql(False, intermediate_relation, sql) }}\n {%- endcall %}\n\n -- cleanup\n {% if existing_relation is not none %}\n /* Do the equivalent of rename_if_exists. 'existing_relation' could have been dropped\n since the variable was first set. */\n {% set existing_relation = load_cached_relation(existing_relation) %}\n {% if existing_relation is not none %}\n {{ adapter.rename_relation(existing_relation, backup_relation) }}\n {% endif %}\n {% endif %}\n\n {{ adapter.rename_relation(intermediate_relation, target_relation) }}\n\n {% do create_indexes(target_relation) %}\n\n {{ run_hooks(post_hooks, inside_transaction=True) }}\n\n {% set should_revoke = should_revoke(existing_relation, full_refresh_mode=True) %}\n {% do apply_grants(target_relation, grant_config, should_revoke=should_revoke) %}\n\n {% do persist_docs(target_relation, model) %}\n\n -- `COMMIT` happens here\n {{ adapter.commit() }}\n\n -- finally, drop the existing/backup relation after the commit\n {{ drop_relation_if_exists(backup_relation) }}\n\n {{ run_hooks(post_hooks, inside_transaction=False) }}\n\n {{ return({'relations': [target_relation]}) }}\n{% endmaterialization %}", "depends_on": {"macros": ["macro.dbt.load_cached_relation", "macro.dbt.make_intermediate_relation", "macro.dbt.make_backup_relation", "macro.dbt.drop_relation_if_exists", "macro.dbt.run_hooks", "macro.dbt.statement", "macro.dbt.get_create_table_as_sql", "macro.dbt.create_indexes", "macro.dbt.should_revoke", "macro.dbt.apply_grants", "macro.dbt.persist_docs"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3893812, "supported_languages": ["sql"]}, "macro.dbt.get_quoted_csv": {"name": "get_quoted_csv", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/column_helpers.sql", "original_file_path": "macros/materializations/models/incremental/column_helpers.sql", "unique_id": "macro.dbt.get_quoted_csv", "macro_sql": "{% macro get_quoted_csv(column_names) %}\n\n {% set quoted = [] %}\n {% for col in column_names -%}\n {%- do quoted.append(adapter.quote(col)) -%}\n {%- endfor %}\n\n {%- set dest_cols_csv = quoted | join(', ') -%}\n {{ return(dest_cols_csv) }}\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389407, "supported_languages": null}, "macro.dbt.diff_columns": {"name": "diff_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/column_helpers.sql", "original_file_path": "macros/materializations/models/incremental/column_helpers.sql", "unique_id": "macro.dbt.diff_columns", "macro_sql": "{% macro diff_columns(source_columns, target_columns) %}\n\n {% set result = [] %}\n {% set source_names = source_columns | map(attribute = 'column') | list %}\n {% set target_names = target_columns | map(attribute = 'column') | list %}\n\n {# --check whether the name attribute exists in the target - this does not perform a data type check #}\n {% for sc in source_columns %}\n {% if sc.name not in target_names %}\n {{ result.append(sc) }}\n {% endif %}\n {% endfor %}\n\n {{ return(result) }}\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389411, "supported_languages": null}, "macro.dbt.diff_column_data_types": {"name": "diff_column_data_types", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/column_helpers.sql", "original_file_path": "macros/materializations/models/incremental/column_helpers.sql", "unique_id": "macro.dbt.diff_column_data_types", "macro_sql": "{% macro diff_column_data_types(source_columns, target_columns) %}\n\n {% set result = [] %}\n {% for sc in source_columns %}\n {% set tc = target_columns | selectattr(\"name\", \"equalto\", sc.name) | list | first %}\n {% if tc %}\n {% if sc.data_type != tc.data_type and not sc.can_expand_to(other_column=tc) %}\n {{ result.append( { 'column_name': tc.name, 'new_type': sc.data_type } ) }}\n {% endif %}\n {% endif %}\n {% endfor %}\n\n {{ return(result) }}\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389417, "supported_languages": null}, "macro.dbt.get_merge_update_columns": {"name": "get_merge_update_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/column_helpers.sql", "original_file_path": "macros/materializations/models/incremental/column_helpers.sql", "unique_id": "macro.dbt.get_merge_update_columns", "macro_sql": "{% macro get_merge_update_columns(merge_update_columns, merge_exclude_columns, dest_columns) %}\n {{ return(adapter.dispatch('get_merge_update_columns', 'dbt')(merge_update_columns, merge_exclude_columns, dest_columns)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_merge_update_columns"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389421, "supported_languages": null}, "macro.dbt.default__get_merge_update_columns": {"name": "default__get_merge_update_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/column_helpers.sql", "original_file_path": "macros/materializations/models/incremental/column_helpers.sql", "unique_id": "macro.dbt.default__get_merge_update_columns", "macro_sql": "{% macro default__get_merge_update_columns(merge_update_columns, merge_exclude_columns, dest_columns) %}\n {%- set default_cols = dest_columns | map(attribute=\"quoted\") | list -%}\n\n {%- if merge_update_columns and merge_exclude_columns -%}\n {{ exceptions.raise_compiler_error(\n 'Model cannot specify merge_update_columns and merge_exclude_columns. Please update model to use only one config'\n )}}\n {%- elif merge_update_columns -%}\n {%- set update_columns = merge_update_columns -%}\n {%- elif merge_exclude_columns -%}\n {%- set update_columns = [] -%}\n {%- for column in dest_columns -%}\n {% if column.column | lower not in merge_exclude_columns | map(\"lower\") | list %}\n {%- do update_columns.append(column.quoted) -%}\n {% endif %}\n {%- endfor -%}\n {%- else -%}\n {%- set update_columns = default_cols -%}\n {%- endif -%}\n\n {{ return(update_columns) }}\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3894248, "supported_languages": null}, "macro.dbt.get_merge_sql": {"name": "get_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/merge.sql", "original_file_path": "macros/materializations/models/incremental/merge.sql", "unique_id": "macro.dbt.get_merge_sql", "macro_sql": "{% macro get_merge_sql(target, source, unique_key, dest_columns, incremental_predicates=none) -%}\n -- back compat for old kwarg name\n {% set incremental_predicates = kwargs.get('predicates', incremental_predicates) %}\n {{ adapter.dispatch('get_merge_sql', 'dbt')(target, source, unique_key, dest_columns, incremental_predicates) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_merge_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3894432, "supported_languages": null}, "macro.dbt.default__get_merge_sql": {"name": "default__get_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/merge.sql", "original_file_path": "macros/materializations/models/incremental/merge.sql", "unique_id": "macro.dbt.default__get_merge_sql", "macro_sql": "{% macro default__get_merge_sql(target, source, unique_key, dest_columns, incremental_predicates=none) -%}\n {%- set predicates = [] if incremental_predicates is none else [] + incremental_predicates -%}\n {%- set dest_cols_csv = get_quoted_csv(dest_columns | map(attribute=\"name\")) -%}\n {%- set merge_update_columns = config.get('merge_update_columns') -%}\n {%- set merge_exclude_columns = config.get('merge_exclude_columns') -%}\n {%- set update_columns = get_merge_update_columns(merge_update_columns, merge_exclude_columns, dest_columns) -%}\n {%- set sql_header = config.get('sql_header', none) -%}\n\n {% if unique_key %}\n {% if unique_key is sequence and unique_key is not mapping and unique_key is not string %}\n {% for key in unique_key %}\n {% set this_key_match %}\n DBT_INTERNAL_SOURCE.{{ key }} = DBT_INTERNAL_DEST.{{ key }}\n {% endset %}\n {% do predicates.append(this_key_match) %}\n {% endfor %}\n {% else %}\n {% set unique_key_match %}\n DBT_INTERNAL_SOURCE.{{ unique_key }} = DBT_INTERNAL_DEST.{{ unique_key }}\n {% endset %}\n {% do predicates.append(unique_key_match) %}\n {% endif %}\n {% else %}\n {% do predicates.append('FALSE') %}\n {% endif %}\n\n {{ sql_header if sql_header is not none }}\n\n merge into {{ target }} as DBT_INTERNAL_DEST\n using {{ source }} as DBT_INTERNAL_SOURCE\n on {{\"(\" ~ predicates | join(\") and (\") ~ \")\"}}\n\n {% if unique_key %}\n when matched then update set\n {% for column_name in update_columns -%}\n {{ column_name }} = DBT_INTERNAL_SOURCE.{{ column_name }}\n {%- if not loop.last %}, {%- endif %}\n {%- endfor %}\n {% endif %}\n\n when not matched then insert\n ({{ dest_cols_csv }})\n values\n ({{ dest_cols_csv }})\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_quoted_csv", "macro.dbt.get_merge_update_columns"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389449, "supported_languages": null}, "macro.dbt.get_delete_insert_merge_sql": {"name": "get_delete_insert_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/merge.sql", "original_file_path": "macros/materializations/models/incremental/merge.sql", "unique_id": "macro.dbt.get_delete_insert_merge_sql", "macro_sql": "{% macro get_delete_insert_merge_sql(target, source, unique_key, dest_columns, incremental_predicates) -%}\n {{ adapter.dispatch('get_delete_insert_merge_sql', 'dbt')(target, source, unique_key, dest_columns, incremental_predicates) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_delete_insert_merge_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.38946, "supported_languages": null}, "macro.dbt.default__get_delete_insert_merge_sql": {"name": "default__get_delete_insert_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/merge.sql", "original_file_path": "macros/materializations/models/incremental/merge.sql", "unique_id": "macro.dbt.default__get_delete_insert_merge_sql", "macro_sql": "{% macro default__get_delete_insert_merge_sql(target, source, unique_key, dest_columns, incremental_predicates) -%}\n\n {%- set dest_cols_csv = get_quoted_csv(dest_columns | map(attribute=\"name\")) -%}\n\n {% if unique_key %}\n {% if unique_key is sequence and unique_key is not string %}\n delete from {{target }}\n using {{ source }}\n where (\n {% for key in unique_key %}\n {{ source }}.{{ key }} = {{ target }}.{{ key }}\n {{ \"and \" if not loop.last}}\n {% endfor %}\n {% if incremental_predicates %}\n {% for predicate in incremental_predicates %}\n and {{ predicate }}\n {% endfor %}\n {% endif %}\n );\n {% else %}\n delete from {{ target }}\n where (\n {{ unique_key }}) in (\n select ({{ unique_key }})\n from {{ source }}\n )\n {%- if incremental_predicates %}\n {% for predicate in incremental_predicates %}\n and {{ predicate }}\n {% endfor %}\n {%- endif -%};\n\n {% endif %}\n {% endif %}\n\n insert into {{ target }} ({{ dest_cols_csv }})\n (\n select {{ dest_cols_csv }}\n from {{ source }}\n )\n\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.get_quoted_csv"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389465, "supported_languages": null}, "macro.dbt.get_insert_overwrite_merge_sql": {"name": "get_insert_overwrite_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/merge.sql", "original_file_path": "macros/materializations/models/incremental/merge.sql", "unique_id": "macro.dbt.get_insert_overwrite_merge_sql", "macro_sql": "{% macro get_insert_overwrite_merge_sql(target, source, dest_columns, predicates, include_sql_header=false) -%}\n {{ adapter.dispatch('get_insert_overwrite_merge_sql', 'dbt')(target, source, dest_columns, predicates, include_sql_header) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_insert_overwrite_merge_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3894699, "supported_languages": null}, "macro.dbt.default__get_insert_overwrite_merge_sql": {"name": "default__get_insert_overwrite_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/merge.sql", "original_file_path": "macros/materializations/models/incremental/merge.sql", "unique_id": "macro.dbt.default__get_insert_overwrite_merge_sql", "macro_sql": "{% macro default__get_insert_overwrite_merge_sql(target, source, dest_columns, predicates, include_sql_header) -%}\n {#-- The only time include_sql_header is True: --#}\n {#-- BigQuery + insert_overwrite strategy + \"static\" partitions config --#}\n {#-- We should consider including the sql header at the materialization level instead --#}\n\n {%- set predicates = [] if predicates is none else [] + predicates -%}\n {%- set dest_cols_csv = get_quoted_csv(dest_columns | map(attribute=\"name\")) -%}\n {%- set sql_header = config.get('sql_header', none) -%}\n\n {{ sql_header if sql_header is not none and include_sql_header }}\n\n merge into {{ target }} as DBT_INTERNAL_DEST\n using {{ source }} as DBT_INTERNAL_SOURCE\n on FALSE\n\n when not matched by source\n {% if predicates %} and {{ predicates | join(' and ') }} {% endif %}\n then delete\n\n when not matched then insert\n ({{ dest_cols_csv }})\n values\n ({{ dest_cols_csv }})\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_quoted_csv"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389478, "supported_languages": null}, "macro.dbt.is_incremental": {"name": "is_incremental", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/is_incremental.sql", "original_file_path": "macros/materializations/models/incremental/is_incremental.sql", "unique_id": "macro.dbt.is_incremental", "macro_sql": "{% macro is_incremental() %}\n {#-- do not run introspective queries in parsing #}\n {% if not execute %}\n {{ return(False) }}\n {% else %}\n {% set relation = adapter.get_relation(this.database, this.schema, this.table) %}\n {{ return(relation is not none\n and relation.type == 'table'\n and model.config.materialized == 'incremental'\n and not should_full_refresh()) }}\n {% endif %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.should_full_refresh"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389487, "supported_languages": null}, "macro.dbt.get_incremental_append_sql": {"name": "get_incremental_append_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.get_incremental_append_sql", "macro_sql": "{% macro get_incremental_append_sql(arg_dict) %}\n\n {{ return(adapter.dispatch('get_incremental_append_sql', 'dbt')(arg_dict)) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_incremental_append_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389496, "supported_languages": null}, "macro.dbt.default__get_incremental_append_sql": {"name": "default__get_incremental_append_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.default__get_incremental_append_sql", "macro_sql": "{% macro default__get_incremental_append_sql(arg_dict) %}\n\n {% do return(get_insert_into_sql(arg_dict[\"target_relation\"], arg_dict[\"temp_relation\"], arg_dict[\"dest_columns\"])) %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_insert_into_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389503, "supported_languages": null}, "macro.dbt.get_incremental_delete_insert_sql": {"name": "get_incremental_delete_insert_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.get_incremental_delete_insert_sql", "macro_sql": "{% macro get_incremental_delete_insert_sql(arg_dict) %}\n\n {{ return(adapter.dispatch('get_incremental_delete_insert_sql', 'dbt')(arg_dict)) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_incremental_delete_insert_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389507, "supported_languages": null}, "macro.dbt.default__get_incremental_delete_insert_sql": {"name": "default__get_incremental_delete_insert_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.default__get_incremental_delete_insert_sql", "macro_sql": "{% macro default__get_incremental_delete_insert_sql(arg_dict) %}\n\n {% do return(get_delete_insert_merge_sql(arg_dict[\"target_relation\"], arg_dict[\"temp_relation\"], arg_dict[\"unique_key\"], arg_dict[\"dest_columns\"], arg_dict[\"incremental_predicates\"])) %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_delete_insert_merge_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3895109, "supported_languages": null}, "macro.dbt.get_incremental_merge_sql": {"name": "get_incremental_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.get_incremental_merge_sql", "macro_sql": "{% macro get_incremental_merge_sql(arg_dict) %}\n\n {{ return(adapter.dispatch('get_incremental_merge_sql', 'dbt')(arg_dict)) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_incremental_merge_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389515, "supported_languages": null}, "macro.dbt.default__get_incremental_merge_sql": {"name": "default__get_incremental_merge_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.default__get_incremental_merge_sql", "macro_sql": "{% macro default__get_incremental_merge_sql(arg_dict) %}\n\n {% do return(get_merge_sql(arg_dict[\"target_relation\"], arg_dict[\"temp_relation\"], arg_dict[\"unique_key\"], arg_dict[\"dest_columns\"], arg_dict[\"incremental_predicates\"])) %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_merge_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389521, "supported_languages": null}, "macro.dbt.get_incremental_insert_overwrite_sql": {"name": "get_incremental_insert_overwrite_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.get_incremental_insert_overwrite_sql", "macro_sql": "{% macro get_incremental_insert_overwrite_sql(arg_dict) %}\n\n {{ return(adapter.dispatch('get_incremental_insert_overwrite_sql', 'dbt')(arg_dict)) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_incremental_insert_overwrite_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389525, "supported_languages": null}, "macro.dbt.default__get_incremental_insert_overwrite_sql": {"name": "default__get_incremental_insert_overwrite_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.default__get_incremental_insert_overwrite_sql", "macro_sql": "{% macro default__get_incremental_insert_overwrite_sql(arg_dict) %}\n\n {% do return(get_insert_overwrite_merge_sql(arg_dict[\"target_relation\"], arg_dict[\"temp_relation\"], arg_dict[\"dest_columns\"], arg_dict[\"incremental_predicates\"])) %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_insert_overwrite_merge_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389529, "supported_languages": null}, "macro.dbt.get_incremental_default_sql": {"name": "get_incremental_default_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.get_incremental_default_sql", "macro_sql": "{% macro get_incremental_default_sql(arg_dict) %}\n\n {{ return(adapter.dispatch('get_incremental_default_sql', 'dbt')(arg_dict)) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_incremental_default_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389534, "supported_languages": null}, "macro.dbt.default__get_incremental_default_sql": {"name": "default__get_incremental_default_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.default__get_incremental_default_sql", "macro_sql": "{% macro default__get_incremental_default_sql(arg_dict) %}\n\n {% do return(get_incremental_append_sql(arg_dict)) %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_incremental_append_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389538, "supported_languages": null}, "macro.dbt.get_incremental_microbatch_sql": {"name": "get_incremental_microbatch_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.get_incremental_microbatch_sql", "macro_sql": "{% macro get_incremental_microbatch_sql(arg_dict) %}\n\n {{ return(adapter.dispatch('get_incremental_microbatch_sql', 'dbt')(arg_dict)) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_incremental_microbatch_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3895411, "supported_languages": null}, "macro.dbt.default__get_incremental_microbatch_sql": {"name": "default__get_incremental_microbatch_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.default__get_incremental_microbatch_sql", "macro_sql": "{% macro default__get_incremental_microbatch_sql(arg_dict) %}\n\n {{ exceptions.raise_not_implemented('microbatch materialization strategy not implemented for adapter ' + adapter.type()) }}\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389545, "supported_languages": null}, "macro.dbt.get_insert_into_sql": {"name": "get_insert_into_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/strategies.sql", "original_file_path": "macros/materializations/models/incremental/strategies.sql", "unique_id": "macro.dbt.get_insert_into_sql", "macro_sql": "{% macro get_insert_into_sql(target_relation, temp_relation, dest_columns) %}\n\n {%- set dest_cols_csv = get_quoted_csv(dest_columns | map(attribute=\"name\")) -%}\n\n insert into {{ target_relation }} ({{ dest_cols_csv }})\n (\n select {{ dest_cols_csv }}\n from {{ temp_relation }}\n )\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_quoted_csv"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3895512, "supported_languages": null}, "macro.dbt.materialization_incremental_default": {"name": "materialization_incremental_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/incremental.sql", "original_file_path": "macros/materializations/models/incremental/incremental.sql", "unique_id": "macro.dbt.materialization_incremental_default", "macro_sql": "{% materialization incremental, default -%}\n\n -- relations\n {%- set existing_relation = load_cached_relation(this) -%}\n {%- set target_relation = this.incorporate(type='table') -%}\n {%- set temp_relation = make_temp_relation(target_relation)-%}\n {%- set intermediate_relation = make_intermediate_relation(target_relation)-%}\n {%- set backup_relation_type = 'table' if existing_relation is none else existing_relation.type -%}\n {%- set backup_relation = make_backup_relation(target_relation, backup_relation_type) -%}\n\n -- configs\n {%- set unique_key = config.get('unique_key') -%}\n {%- set full_refresh_mode = (should_full_refresh() or existing_relation.is_view) -%}\n {%- set on_schema_change = incremental_validate_on_schema_change(config.get('on_schema_change'), default='ignore') -%}\n\n -- the temp_ and backup_ relations should not already exist in the database; get_relation\n -- will return None in that case. Otherwise, we get a relation that we can drop\n -- later, before we try to use this name for the current operation. This has to happen before\n -- BEGIN, in a separate transaction\n {%- set preexisting_intermediate_relation = load_cached_relation(intermediate_relation)-%}\n {%- set preexisting_backup_relation = load_cached_relation(backup_relation) -%}\n -- grab current tables grants config for comparision later on\n {% set grant_config = config.get('grants') %}\n {{ drop_relation_if_exists(preexisting_intermediate_relation) }}\n {{ drop_relation_if_exists(preexisting_backup_relation) }}\n\n {{ run_hooks(pre_hooks, inside_transaction=False) }}\n\n -- `BEGIN` happens here:\n {{ run_hooks(pre_hooks, inside_transaction=True) }}\n\n {% set to_drop = [] %}\n\n {% if existing_relation is none %}\n {% set build_sql = get_create_table_as_sql(False, target_relation, sql) %}\n {% elif full_refresh_mode %}\n {% set build_sql = get_create_table_as_sql(False, intermediate_relation, sql) %}\n {% set need_swap = true %}\n {% else %}\n {% do run_query(get_create_table_as_sql(True, temp_relation, sql)) %}\n {% set contract_config = config.get('contract') %}\n {% if not contract_config or not contract_config.enforced %}\n {% do adapter.expand_target_column_types(\n from_relation=temp_relation,\n to_relation=target_relation) %}\n {% endif %}\n {#-- Process schema changes. Returns dict of changes if successful. Use source columns for upserting/merging --#}\n {% set dest_columns = process_schema_changes(on_schema_change, temp_relation, existing_relation) %}\n {% if not dest_columns %}\n {% set dest_columns = adapter.get_columns_in_relation(existing_relation) %}\n {% endif %}\n\n {#-- Get the incremental_strategy, the macro to use for the strategy, and build the sql --#}\n {% set incremental_strategy = config.get('incremental_strategy') or 'default' %}\n {% set incremental_predicates = config.get('predicates', none) or config.get('incremental_predicates', none) %}\n {% set strategy_sql_macro_func = adapter.get_incremental_strategy_macro(context, incremental_strategy) %}\n {% set strategy_arg_dict = ({'target_relation': target_relation, 'temp_relation': temp_relation, 'unique_key': unique_key, 'dest_columns': dest_columns, 'incremental_predicates': incremental_predicates }) %}\n {% set build_sql = strategy_sql_macro_func(strategy_arg_dict) %}\n\n {% endif %}\n\n {% call statement(\"main\") %}\n {{ build_sql }}\n {% endcall %}\n\n {% if need_swap %}\n {% do adapter.rename_relation(target_relation, backup_relation) %}\n {% do adapter.rename_relation(intermediate_relation, target_relation) %}\n {% do to_drop.append(backup_relation) %}\n {% endif %}\n\n {% set should_revoke = should_revoke(existing_relation, full_refresh_mode) %}\n {% do apply_grants(target_relation, grant_config, should_revoke=should_revoke) %}\n\n {% do persist_docs(target_relation, model) %}\n\n {% if existing_relation is none or existing_relation.is_view or should_full_refresh() %}\n {% do create_indexes(target_relation) %}\n {% endif %}\n\n {{ run_hooks(post_hooks, inside_transaction=True) }}\n\n -- `COMMIT` happens here\n {% do adapter.commit() %}\n\n {% for rel in to_drop %}\n {% do adapter.drop_relation(rel) %}\n {% endfor %}\n\n {{ run_hooks(post_hooks, inside_transaction=False) }}\n\n {{ return({'relations': [target_relation]}) }}\n\n{%- endmaterialization %}", "depends_on": {"macros": ["macro.dbt.load_cached_relation", "macro.dbt.make_temp_relation", "macro.dbt.make_intermediate_relation", "macro.dbt.make_backup_relation", "macro.dbt.should_full_refresh", "macro.dbt.incremental_validate_on_schema_change", "macro.dbt.drop_relation_if_exists", "macro.dbt.run_hooks", "macro.dbt.get_create_table_as_sql", "macro.dbt.run_query", "macro.dbt.process_schema_changes", "macro.dbt.statement", "macro.dbt.should_revoke", "macro.dbt.apply_grants", "macro.dbt.persist_docs", "macro.dbt.create_indexes"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389565, "supported_languages": ["sql"]}, "macro.dbt.incremental_validate_on_schema_change": {"name": "incremental_validate_on_schema_change", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/on_schema_change.sql", "original_file_path": "macros/materializations/models/incremental/on_schema_change.sql", "unique_id": "macro.dbt.incremental_validate_on_schema_change", "macro_sql": "{% macro incremental_validate_on_schema_change(on_schema_change, default='ignore') %}\n\n {% if on_schema_change not in ['sync_all_columns', 'append_new_columns', 'fail', 'ignore'] %}\n\n {% set log_message = 'Invalid value for on_schema_change (%s) specified. Setting default value of %s.' % (on_schema_change, default) %}\n {% do log(log_message) %}\n\n {{ return(default) }}\n\n {% else %}\n\n {{ return(on_schema_change) }}\n\n {% endif %}\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389589, "supported_languages": null}, "macro.dbt.check_for_schema_changes": {"name": "check_for_schema_changes", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/on_schema_change.sql", "original_file_path": "macros/materializations/models/incremental/on_schema_change.sql", "unique_id": "macro.dbt.check_for_schema_changes", "macro_sql": "{% macro check_for_schema_changes(source_relation, target_relation) %}\n\n {% set schema_changed = False %}\n\n {%- set source_columns = adapter.get_columns_in_relation(source_relation) -%}\n {%- set target_columns = adapter.get_columns_in_relation(target_relation) -%}\n {%- set source_not_in_target = diff_columns(source_columns, target_columns) -%}\n {%- set target_not_in_source = diff_columns(target_columns, source_columns) -%}\n\n {% set new_target_types = diff_column_data_types(source_columns, target_columns) %}\n\n {% if source_not_in_target != [] %}\n {% set schema_changed = True %}\n {% elif target_not_in_source != [] or new_target_types != [] %}\n {% set schema_changed = True %}\n {% elif new_target_types != [] %}\n {% set schema_changed = True %}\n {% endif %}\n\n {% set changes_dict = {\n 'schema_changed': schema_changed,\n 'source_not_in_target': source_not_in_target,\n 'target_not_in_source': target_not_in_source,\n 'source_columns': source_columns,\n 'target_columns': target_columns,\n 'new_target_types': new_target_types\n } %}\n\n {% set msg %}\n In {{ target_relation }}:\n Schema changed: {{ schema_changed }}\n Source columns not in target: {{ source_not_in_target }}\n Target columns not in source: {{ target_not_in_source }}\n New column types: {{ new_target_types }}\n {% endset %}\n\n {% do log(msg) %}\n\n {{ return(changes_dict) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.diff_columns", "macro.dbt.diff_column_data_types"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389594, "supported_languages": null}, "macro.dbt.sync_column_schemas": {"name": "sync_column_schemas", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/on_schema_change.sql", "original_file_path": "macros/materializations/models/incremental/on_schema_change.sql", "unique_id": "macro.dbt.sync_column_schemas", "macro_sql": "{% macro sync_column_schemas(on_schema_change, target_relation, schema_changes_dict) %}\n\n {%- set add_to_target_arr = schema_changes_dict['source_not_in_target'] -%}\n\n {%- if on_schema_change == 'append_new_columns'-%}\n {%- if add_to_target_arr | length > 0 -%}\n {%- do alter_relation_add_remove_columns(target_relation, add_to_target_arr, none) -%}\n {%- endif -%}\n\n {% elif on_schema_change == 'sync_all_columns' %}\n {%- set remove_from_target_arr = schema_changes_dict['target_not_in_source'] -%}\n {%- set new_target_types = schema_changes_dict['new_target_types'] -%}\n\n {% if add_to_target_arr | length > 0 or remove_from_target_arr | length > 0 %}\n {%- do alter_relation_add_remove_columns(target_relation, add_to_target_arr, remove_from_target_arr) -%}\n {% endif %}\n\n {% if new_target_types != [] %}\n {% for ntt in new_target_types %}\n {% set column_name = ntt['column_name'] %}\n {% set new_type = ntt['new_type'] %}\n {% do alter_column_type(target_relation, column_name, new_type) %}\n {% endfor %}\n {% endif %}\n\n {% endif %}\n\n {% set schema_change_message %}\n In {{ target_relation }}:\n Schema change approach: {{ on_schema_change }}\n Columns added: {{ add_to_target_arr }}\n Columns removed: {{ remove_from_target_arr }}\n Data types changed: {{ new_target_types }}\n {% endset %}\n\n {% do log(schema_change_message) %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.alter_relation_add_remove_columns", "macro.dbt.alter_column_type"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389602, "supported_languages": null}, "macro.dbt.process_schema_changes": {"name": "process_schema_changes", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/incremental/on_schema_change.sql", "original_file_path": "macros/materializations/models/incremental/on_schema_change.sql", "unique_id": "macro.dbt.process_schema_changes", "macro_sql": "{% macro process_schema_changes(on_schema_change, source_relation, target_relation) %}\n\n {% if on_schema_change == 'ignore' %}\n\n {{ return({}) }}\n\n {% else %}\n\n {% set schema_changes_dict = check_for_schema_changes(source_relation, target_relation) %}\n\n {% if schema_changes_dict['schema_changed'] %}\n\n {% if on_schema_change == 'fail' %}\n\n {% set fail_msg %}\n The source and target schemas on this incremental model are out of sync!\n They can be reconciled in several ways:\n - set the `on_schema_change` config to either append_new_columns or sync_all_columns, depending on your situation.\n - Re-run the incremental model with `full_refresh: True` to update the target schema.\n - update the schema manually and re-run the process.\n\n Additional troubleshooting context:\n Source columns not in target: {{ schema_changes_dict['source_not_in_target'] }}\n Target columns not in source: {{ schema_changes_dict['target_not_in_source'] }}\n New column types: {{ schema_changes_dict['new_target_types'] }}\n {% endset %}\n\n {% do exceptions.raise_compiler_error(fail_msg) %}\n\n {# -- unless we ignore, run the sync operation per the config #}\n {% else %}\n\n {% do sync_column_schemas(on_schema_change, target_relation, schema_changes_dict) %}\n\n {% endif %}\n\n {% endif %}\n\n {{ return(schema_changes_dict['source_columns']) }}\n\n {% endif %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.check_for_schema_changes", "macro.dbt.sync_column_schemas"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389606, "supported_languages": null}, "macro.dbt.can_clone_table": {"name": "can_clone_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/clone/can_clone_table.sql", "original_file_path": "macros/materializations/models/clone/can_clone_table.sql", "unique_id": "macro.dbt.can_clone_table", "macro_sql": "{% macro can_clone_table() %}\n {{ return(adapter.dispatch('can_clone_table', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__can_clone_table"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389616, "supported_languages": null}, "macro.dbt.default__can_clone_table": {"name": "default__can_clone_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/clone/can_clone_table.sql", "original_file_path": "macros/materializations/models/clone/can_clone_table.sql", "unique_id": "macro.dbt.default__can_clone_table", "macro_sql": "{% macro default__can_clone_table() %}\n {{ return(False) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.38962, "supported_languages": null}, "macro.dbt.create_or_replace_clone": {"name": "create_or_replace_clone", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/clone/create_or_replace_clone.sql", "original_file_path": "macros/materializations/models/clone/create_or_replace_clone.sql", "unique_id": "macro.dbt.create_or_replace_clone", "macro_sql": "{% macro create_or_replace_clone(this_relation, defer_relation) %}\n {{ return(adapter.dispatch('create_or_replace_clone', 'dbt')(this_relation, defer_relation)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__create_or_replace_clone"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389628, "supported_languages": null}, "macro.dbt.default__create_or_replace_clone": {"name": "default__create_or_replace_clone", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/clone/create_or_replace_clone.sql", "original_file_path": "macros/materializations/models/clone/create_or_replace_clone.sql", "unique_id": "macro.dbt.default__create_or_replace_clone", "macro_sql": "{% macro default__create_or_replace_clone(this_relation, defer_relation) %}\n create or replace table {{ this_relation.render() }} clone {{ defer_relation.render() }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389632, "supported_languages": null}, "macro.dbt.materialization_clone_default": {"name": "materialization_clone_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/models/clone/clone.sql", "original_file_path": "macros/materializations/models/clone/clone.sql", "unique_id": "macro.dbt.materialization_clone_default", "macro_sql": "{%- materialization clone, default -%}\n\n {%- set relations = {'relations': []} -%}\n\n {%- if not defer_relation -%}\n -- nothing to do\n {{ log(\"No relation found in state manifest for \" ~ model.unique_id, info=True) }}\n {{ return(relations) }}\n {%- endif -%}\n\n {%- set existing_relation = load_cached_relation(this) -%}\n\n {%- if existing_relation and not flags.FULL_REFRESH -%}\n -- noop!\n {{ log(\"Relation \" ~ existing_relation ~ \" already exists\", info=True) }}\n {{ return(relations) }}\n {%- endif -%}\n\n {%- set other_existing_relation = load_cached_relation(defer_relation) -%}\n\n -- If this is a database that can do zero-copy cloning of tables, and the other relation is a table, then this will be a table\n -- Otherwise, this will be a view\n\n {% set can_clone_table = can_clone_table() %}\n\n {%- if other_existing_relation and other_existing_relation.type == 'table' and can_clone_table -%}\n\n {%- set target_relation = this.incorporate(type='table') -%}\n {% if existing_relation is not none and not existing_relation.is_table %}\n {{ log(\"Dropping relation \" ~ existing_relation.render() ~ \" because it is of type \" ~ existing_relation.type) }}\n {{ drop_relation_if_exists(existing_relation) }}\n {% endif %}\n\n -- as a general rule, data platforms that can clone tables can also do atomic 'create or replace'\n {% call statement('main') %}\n {% if target_relation and defer_relation and target_relation == defer_relation %}\n {{ log(\"Target relation and defer relation are the same, skipping clone for relation: \" ~ target_relation.render()) }}\n {% else %}\n {{ create_or_replace_clone(target_relation, defer_relation) }}\n {% endif %}\n\n {% endcall %}\n\n {% set should_revoke = should_revoke(existing_relation, full_refresh_mode=True) %}\n {% do apply_grants(target_relation, grant_config, should_revoke=should_revoke) %}\n {% do persist_docs(target_relation, model) %}\n\n {{ return({'relations': [target_relation]}) }}\n\n {%- else -%}\n\n {%- set target_relation = this.incorporate(type='view') -%}\n\n -- reuse the view materialization\n -- TODO: support actual dispatch for materialization macros\n -- Tracking ticket: https://github.com/dbt-labs/dbt-core/issues/7799\n {% set search_name = \"materialization_view_\" ~ adapter.type() %}\n {% if not search_name in context %}\n {% set search_name = \"materialization_view_default\" %}\n {% endif %}\n {% set materialization_macro = context[search_name] %}\n {% set relations = materialization_macro() %}\n {{ return(relations) }}\n\n {%- endif -%}\n\n{%- endmaterialization -%}", "depends_on": {"macros": ["macro.dbt.load_cached_relation", "macro.dbt.can_clone_table", "macro.dbt.drop_relation_if_exists", "macro.dbt.statement", "macro.dbt.create_or_replace_clone", "macro.dbt.should_revoke", "macro.dbt.apply_grants", "macro.dbt.persist_docs"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389646, "supported_languages": ["sql"]}, "macro.dbt.materialization_seed_default": {"name": "materialization_seed_default", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/seed.sql", "original_file_path": "macros/materializations/seeds/seed.sql", "unique_id": "macro.dbt.materialization_seed_default", "macro_sql": "{% materialization seed, default %}\n\n {%- set identifier = model['alias'] -%}\n {%- set full_refresh_mode = (should_full_refresh()) -%}\n\n {%- set old_relation = adapter.get_relation(database=database, schema=schema, identifier=identifier) -%}\n\n {%- set exists_as_table = (old_relation is not none and old_relation.is_table) -%}\n {%- set exists_as_view = (old_relation is not none and old_relation.is_view) -%}\n\n {%- set grant_config = config.get('grants') -%}\n {%- set agate_table = load_agate_table() -%}\n -- grab current tables grants config for comparison later on\n\n {%- do store_result('agate_table', response='OK', agate_table=agate_table) -%}\n\n {{ run_hooks(pre_hooks, inside_transaction=False) }}\n\n -- `BEGIN` happens here:\n {{ run_hooks(pre_hooks, inside_transaction=True) }}\n\n -- build model\n {% set create_table_sql = \"\" %}\n {% if exists_as_view %}\n {{ exceptions.raise_compiler_error(\"Cannot seed to '{}', it is a view\".format(old_relation.render())) }}\n {% elif exists_as_table %}\n {% set create_table_sql = reset_csv_table(model, full_refresh_mode, old_relation, agate_table) %}\n {% else %}\n {% set create_table_sql = create_csv_table(model, agate_table) %}\n {% endif %}\n\n {% set code = 'CREATE' if full_refresh_mode else 'INSERT' %}\n {% set rows_affected = (agate_table.rows | length) %}\n {% set sql = load_csv_rows(model, agate_table) %}\n\n {% call noop_statement('main', code ~ ' ' ~ rows_affected, code, rows_affected) %}\n {{ get_csv_sql(create_table_sql, sql) }};\n {% endcall %}\n\n {% set target_relation = this.incorporate(type='table') %}\n\n {% set should_revoke = should_revoke(old_relation, full_refresh_mode) %}\n {% do apply_grants(target_relation, grant_config, should_revoke=should_revoke) %}\n\n {% do persist_docs(target_relation, model) %}\n\n {% if full_refresh_mode or not exists_as_table %}\n {% do create_indexes(target_relation) %}\n {% endif %}\n\n {{ run_hooks(post_hooks, inside_transaction=True) }}\n\n -- `COMMIT` happens here\n {{ adapter.commit() }}\n\n {{ run_hooks(post_hooks, inside_transaction=False) }}\n\n {{ return({'relations': [target_relation]}) }}\n\n{% endmaterialization %}", "depends_on": {"macros": ["macro.dbt.should_full_refresh", "macro.dbt.run_hooks", "macro.dbt.reset_csv_table", "macro.dbt.create_csv_table", "macro.dbt.load_csv_rows", "macro.dbt.noop_statement", "macro.dbt.get_csv_sql", "macro.dbt.should_revoke", "macro.dbt.apply_grants", "macro.dbt.persist_docs", "macro.dbt.create_indexes"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389663, "supported_languages": ["sql"]}, "macro.dbt.create_csv_table": {"name": "create_csv_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.create_csv_table", "macro_sql": "{% macro create_csv_table(model, agate_table) -%}\n {{ adapter.dispatch('create_csv_table', 'dbt')(model, agate_table) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__create_csv_table"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3896809, "supported_languages": null}, "macro.dbt.default__create_csv_table": {"name": "default__create_csv_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.default__create_csv_table", "macro_sql": "{% macro default__create_csv_table(model, agate_table) %}\n {%- set column_override = model['config'].get('column_types', {}) -%}\n {%- set quote_seed_column = model['config'].get('quote_columns', None) -%}\n\n {% set sql %}\n create table {{ this.render() }} (\n {%- for col_name in agate_table.column_names -%}\n {%- set inferred_type = adapter.convert_type(agate_table, loop.index0) -%}\n {%- set type = column_override.get(col_name, inferred_type) -%}\n {%- set column_name = (col_name | string) -%}\n {{ adapter.quote_seed_column(column_name, quote_seed_column) }} {{ type }} {%- if not loop.last -%}, {%- endif -%}\n {%- endfor -%}\n )\n {% endset %}\n\n {% call statement('_') -%}\n {{ sql }}\n {%- endcall %}\n\n {{ return(sql) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3896902, "supported_languages": null}, "macro.dbt.reset_csv_table": {"name": "reset_csv_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.reset_csv_table", "macro_sql": "{% macro reset_csv_table(model, full_refresh, old_relation, agate_table) -%}\n {{ adapter.dispatch('reset_csv_table', 'dbt')(model, full_refresh, old_relation, agate_table) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__reset_csv_table"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389696, "supported_languages": null}, "macro.dbt.default__reset_csv_table": {"name": "default__reset_csv_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.default__reset_csv_table", "macro_sql": "{% macro default__reset_csv_table(model, full_refresh, old_relation, agate_table) %}\n {% set sql = \"\" %}\n {% if full_refresh %}\n {{ adapter.drop_relation(old_relation) }}\n {% set sql = create_csv_table(model, agate_table) %}\n {% else %}\n {{ adapter.truncate_relation(old_relation) }}\n {% set sql = \"truncate table \" ~ old_relation.render() %}\n {% endif %}\n\n {{ return(sql) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.create_csv_table"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3897002, "supported_languages": null}, "macro.dbt.get_csv_sql": {"name": "get_csv_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.get_csv_sql", "macro_sql": "{% macro get_csv_sql(create_or_truncate_sql, insert_sql) %}\n {{ adapter.dispatch('get_csv_sql', 'dbt')(create_or_truncate_sql, insert_sql) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_csv_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389704, "supported_languages": null}, "macro.dbt.default__get_csv_sql": {"name": "default__get_csv_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.default__get_csv_sql", "macro_sql": "{% macro default__get_csv_sql(create_or_truncate_sql, insert_sql) %}\n {{ create_or_truncate_sql }};\n -- dbt seed --\n {{ insert_sql }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389708, "supported_languages": null}, "macro.dbt.get_binding_char": {"name": "get_binding_char", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.get_binding_char", "macro_sql": "{% macro get_binding_char() -%}\n {{ adapter.dispatch('get_binding_char', 'dbt')() }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_binding_char"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389726, "supported_languages": null}, "macro.dbt.default__get_binding_char": {"name": "default__get_binding_char", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.default__get_binding_char", "macro_sql": "{% macro default__get_binding_char() %}\n {{ return('%s') }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389736, "supported_languages": null}, "macro.dbt.get_batch_size": {"name": "get_batch_size", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.get_batch_size", "macro_sql": "{% macro get_batch_size() -%}\n {{ return(adapter.dispatch('get_batch_size', 'dbt')()) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_batch_size"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389743, "supported_languages": null}, "macro.dbt.default__get_batch_size": {"name": "default__get_batch_size", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.default__get_batch_size", "macro_sql": "{% macro default__get_batch_size() %}\n {{ return(10000) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389747, "supported_languages": null}, "macro.dbt.get_seed_column_quoted_csv": {"name": "get_seed_column_quoted_csv", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.get_seed_column_quoted_csv", "macro_sql": "{% macro get_seed_column_quoted_csv(model, column_names) %}\n {%- set quote_seed_column = model['config'].get('quote_columns', None) -%}\n {% set quoted = [] %}\n {% for col in column_names -%}\n {%- do quoted.append(adapter.quote_seed_column(col, quote_seed_column)) -%}\n {%- endfor %}\n\n {%- set dest_cols_csv = quoted | join(', ') -%}\n {{ return(dest_cols_csv) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.38975, "supported_languages": null}, "macro.dbt.load_csv_rows": {"name": "load_csv_rows", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.load_csv_rows", "macro_sql": "{% macro load_csv_rows(model, agate_table) -%}\n {{ adapter.dispatch('load_csv_rows', 'dbt')(model, agate_table) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__load_csv_rows"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389756, "supported_languages": null}, "macro.dbt.default__load_csv_rows": {"name": "default__load_csv_rows", "resource_type": "macro", "package_name": "dbt", "path": "macros/materializations/seeds/helpers.sql", "original_file_path": "macros/materializations/seeds/helpers.sql", "unique_id": "macro.dbt.default__load_csv_rows", "macro_sql": "{% macro default__load_csv_rows(model, agate_table) %}\n\n {% set batch_size = get_batch_size() %}\n\n {% set cols_sql = get_seed_column_quoted_csv(model, agate_table.column_names) %}\n {% set bindings = [] %}\n\n {% set statements = [] %}\n\n {% for chunk in agate_table.rows | batch(batch_size) %}\n {% set bindings = [] %}\n\n {% for row in chunk %}\n {% do bindings.extend(row) %}\n {% endfor %}\n\n {% set sql %}\n insert into {{ this.render() }} ({{ cols_sql }}) values\n {% for row in chunk -%}\n ({%- for column in agate_table.column_names -%}\n {{ get_binding_char() }}\n {%- if not loop.last%},{%- endif %}\n {%- endfor -%})\n {%- if not loop.last%},{%- endif %}\n {%- endfor %}\n {% endset %}\n\n {% do adapter.add_query(sql, bindings=bindings, abridge_sql_log=True) %}\n\n {% if loop.index0 == 0 %}\n {% do statements.append(sql) %}\n {% endif %}\n {% endfor %}\n\n {# Return SQL so we can render it out into the compiled files #}\n {{ return(statements[0]) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_batch_size", "macro.dbt.get_seed_column_quoted_csv", "macro.dbt.get_binding_char"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389761, "supported_languages": null}, "macro.dbt.generate_alias_name": {"name": "generate_alias_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/get_custom_name/get_custom_alias.sql", "original_file_path": "macros/get_custom_name/get_custom_alias.sql", "unique_id": "macro.dbt.generate_alias_name", "macro_sql": "{% macro generate_alias_name(custom_alias_name=none, node=none) -%}\n {% do return(adapter.dispatch('generate_alias_name', 'dbt')(custom_alias_name, node)) %}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__generate_alias_name"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389771, "supported_languages": null}, "macro.dbt.default__generate_alias_name": {"name": "default__generate_alias_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/get_custom_name/get_custom_alias.sql", "original_file_path": "macros/get_custom_name/get_custom_alias.sql", "unique_id": "macro.dbt.default__generate_alias_name", "macro_sql": "{% macro default__generate_alias_name(custom_alias_name=none, node=none) -%}\n\n {%- if custom_alias_name -%}\n\n {{ custom_alias_name | trim }}\n\n {%- elif node.version -%}\n\n {{ return(node.name ~ \"_v\" ~ (node.version | replace(\".\", \"_\"))) }}\n\n {%- else -%}\n\n {{ node.name }}\n\n {%- endif -%}\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389775, "supported_languages": null}, "macro.dbt.generate_schema_name": {"name": "generate_schema_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/get_custom_name/get_custom_schema.sql", "original_file_path": "macros/get_custom_name/get_custom_schema.sql", "unique_id": "macro.dbt.generate_schema_name", "macro_sql": "{% macro generate_schema_name(custom_schema_name=none, node=none) -%}\n {{ return(adapter.dispatch('generate_schema_name', 'dbt')(custom_schema_name, node)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__generate_schema_name"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389785, "supported_languages": null}, "macro.dbt.default__generate_schema_name": {"name": "default__generate_schema_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/get_custom_name/get_custom_schema.sql", "original_file_path": "macros/get_custom_name/get_custom_schema.sql", "unique_id": "macro.dbt.default__generate_schema_name", "macro_sql": "{% macro default__generate_schema_name(custom_schema_name, node) -%}\n\n {%- set default_schema = target.schema -%}\n {%- if custom_schema_name is none -%}\n\n {{ default_schema }}\n\n {%- else -%}\n\n {{ default_schema }}_{{ custom_schema_name | trim }}\n\n {%- endif -%}\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389791, "supported_languages": null}, "macro.dbt.generate_schema_name_for_env": {"name": "generate_schema_name_for_env", "resource_type": "macro", "package_name": "dbt", "path": "macros/get_custom_name/get_custom_schema.sql", "original_file_path": "macros/get_custom_name/get_custom_schema.sql", "unique_id": "macro.dbt.generate_schema_name_for_env", "macro_sql": "{% macro generate_schema_name_for_env(custom_schema_name, node) -%}\n\n {%- set default_schema = target.schema -%}\n {%- if target.name == 'prod' and custom_schema_name is not none -%}\n\n {{ custom_schema_name | trim }}\n\n {%- else -%}\n\n {{ default_schema }}\n\n {%- endif -%}\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389795, "supported_languages": null}, "macro.dbt.generate_database_name": {"name": "generate_database_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/get_custom_name/get_custom_database.sql", "original_file_path": "macros/get_custom_name/get_custom_database.sql", "unique_id": "macro.dbt.generate_database_name", "macro_sql": "{% macro generate_database_name(custom_database_name=none, node=none) -%}\n {% do return(adapter.dispatch('generate_database_name', 'dbt')(custom_database_name, node)) %}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__generate_database_name"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389807, "supported_languages": null}, "macro.dbt.default__generate_database_name": {"name": "default__generate_database_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/get_custom_name/get_custom_database.sql", "original_file_path": "macros/get_custom_name/get_custom_database.sql", "unique_id": "macro.dbt.default__generate_database_name", "macro_sql": "{% macro default__generate_database_name(custom_database_name=none, node=none) -%}\n {%- set default_database = target.database -%}\n {%- if custom_database_name is none -%}\n\n {{ default_database }}\n\n {%- else -%}\n\n {{ custom_database_name }}\n\n {%- endif -%}\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389811, "supported_languages": null}, "macro.dbt.get_drop_sql": {"name": "get_drop_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/drop.sql", "original_file_path": "macros/relations/drop.sql", "unique_id": "macro.dbt.get_drop_sql", "macro_sql": "{%- macro get_drop_sql(relation) -%}\n {{- log('Applying DROP to: ' ~ relation) -}}\n {{- adapter.dispatch('get_drop_sql', 'dbt')(relation) -}}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.default__get_drop_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3898199, "supported_languages": null}, "macro.dbt.default__get_drop_sql": {"name": "default__get_drop_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/drop.sql", "original_file_path": "macros/relations/drop.sql", "unique_id": "macro.dbt.default__get_drop_sql", "macro_sql": "{%- macro default__get_drop_sql(relation) -%}\n\n {%- if relation.is_view -%}\n {{ drop_view(relation) }}\n\n {%- elif relation.is_table -%}\n {{ drop_table(relation) }}\n\n {%- elif relation.is_materialized_view -%}\n {{ drop_materialized_view(relation) }}\n\n {%- else -%}\n drop {{ relation.type }} if exists {{ relation.render() }} cascade\n\n {%- endif -%}\n\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.drop_view", "macro.dbt.drop_table", "macro.dbt.drop_materialized_view"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389824, "supported_languages": null}, "macro.dbt.drop_relation": {"name": "drop_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/drop.sql", "original_file_path": "macros/relations/drop.sql", "unique_id": "macro.dbt.drop_relation", "macro_sql": "{% macro drop_relation(relation) -%}\n {{ return(adapter.dispatch('drop_relation', 'dbt')(relation)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__drop_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389832, "supported_languages": null}, "macro.dbt.default__drop_relation": {"name": "default__drop_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/drop.sql", "original_file_path": "macros/relations/drop.sql", "unique_id": "macro.dbt.default__drop_relation", "macro_sql": "{% macro default__drop_relation(relation) -%}\n {% call statement('drop_relation', auto_begin=False) -%}\n {{ get_drop_sql(relation) }}\n {%- endcall %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement", "macro.dbt.get_drop_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389836, "supported_languages": null}, "macro.dbt.drop_relation_if_exists": {"name": "drop_relation_if_exists", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/drop.sql", "original_file_path": "macros/relations/drop.sql", "unique_id": "macro.dbt.drop_relation_if_exists", "macro_sql": "{% macro drop_relation_if_exists(relation) %}\n {% if relation is not none %}\n {{ adapter.drop_relation(relation) }}\n {% endif %}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.38984, "supported_languages": null}, "macro.dbt.get_replace_sql": {"name": "get_replace_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/replace.sql", "original_file_path": "macros/relations/replace.sql", "unique_id": "macro.dbt.get_replace_sql", "macro_sql": "{% macro get_replace_sql(existing_relation, target_relation, sql) %}\n {{- log('Applying REPLACE to: ' ~ existing_relation) -}}\n {{- adapter.dispatch('get_replace_sql', 'dbt')(existing_relation, target_relation, sql) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_replace_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3898518, "supported_languages": null}, "macro.dbt.default__get_replace_sql": {"name": "default__get_replace_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/replace.sql", "original_file_path": "macros/relations/replace.sql", "unique_id": "macro.dbt.default__get_replace_sql", "macro_sql": "{% macro default__get_replace_sql(existing_relation, target_relation, sql) %}\n\n {# /* use a create or replace statement if possible */ #}\n\n {% set is_replaceable = existing_relation.type == target_relation_type and existing_relation.can_be_replaced %}\n\n {% if is_replaceable and existing_relation.is_view %}\n {{ get_replace_view_sql(target_relation, sql) }}\n\n {% elif is_replaceable and existing_relation.is_table %}\n {{ get_replace_table_sql(target_relation, sql) }}\n\n {% elif is_replaceable and existing_relation.is_materialized_view %}\n {{ get_replace_materialized_view_sql(target_relation, sql) }}\n\n {# /* a create or replace statement is not possible, so try to stage and/or backup to be safe */ #}\n\n {# /* create target_relation as an intermediate relation, then swap it out with the existing one using a backup */ #}\n {%- elif target_relation.can_be_renamed and existing_relation.can_be_renamed -%}\n {{ get_create_intermediate_sql(target_relation, sql) }};\n {{ get_create_backup_sql(existing_relation) }};\n {{ get_rename_intermediate_sql(target_relation) }};\n {{ get_drop_backup_sql(existing_relation) }}\n\n {# /* create target_relation as an intermediate relation, then swap it out with the existing one without using a backup */ #}\n {%- elif target_relation.can_be_renamed -%}\n {{ get_create_intermediate_sql(target_relation, sql) }};\n {{ get_drop_sql(existing_relation) }};\n {{ get_rename_intermediate_sql(target_relation) }}\n\n {# /* create target_relation in place by first backing up the existing relation */ #}\n {%- elif existing_relation.can_be_renamed -%}\n {{ get_create_backup_sql(existing_relation) }};\n {{ get_create_sql(target_relation, sql) }};\n {{ get_drop_backup_sql(existing_relation) }}\n\n {# /* no renaming is allowed, so just drop and create */ #}\n {%- else -%}\n {{ get_drop_sql(existing_relation) }};\n {{ get_create_sql(target_relation, sql) }}\n\n {%- endif -%}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_replace_view_sql", "macro.dbt.get_replace_table_sql", "macro.dbt.get_replace_materialized_view_sql", "macro.dbt.get_create_intermediate_sql", "macro.dbt.get_create_backup_sql", "macro.dbt.get_rename_intermediate_sql", "macro.dbt.get_drop_backup_sql", "macro.dbt.get_drop_sql", "macro.dbt.get_create_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389861, "supported_languages": null}, "macro.dbt.get_create_intermediate_sql": {"name": "get_create_intermediate_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/create_intermediate.sql", "original_file_path": "macros/relations/create_intermediate.sql", "unique_id": "macro.dbt.get_create_intermediate_sql", "macro_sql": "{%- macro get_create_intermediate_sql(relation, sql) -%}\n {{- log('Applying CREATE INTERMEDIATE to: ' ~ relation) -}}\n {{- adapter.dispatch('get_create_intermediate_sql', 'dbt')(relation, sql) -}}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.default__get_create_intermediate_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389872, "supported_languages": null}, "macro.dbt.default__get_create_intermediate_sql": {"name": "default__get_create_intermediate_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/create_intermediate.sql", "original_file_path": "macros/relations/create_intermediate.sql", "unique_id": "macro.dbt.default__get_create_intermediate_sql", "macro_sql": "{%- macro default__get_create_intermediate_sql(relation, sql) -%}\n\n -- get the standard intermediate name\n {% set intermediate_relation = make_intermediate_relation(relation) %}\n\n -- drop any pre-existing intermediate\n {{ get_drop_sql(intermediate_relation) }};\n\n {{ get_create_sql(intermediate_relation, sql) }}\n\n{%- endmacro -%}", "depends_on": {"macros": ["macro.dbt.make_intermediate_relation", "macro.dbt.get_drop_sql", "macro.dbt.get_create_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389876, "supported_languages": null}, "macro.dbt.drop_schema_named": {"name": "drop_schema_named", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/schema.sql", "original_file_path": "macros/relations/schema.sql", "unique_id": "macro.dbt.drop_schema_named", "macro_sql": "{% macro drop_schema_named(schema_name) %}\n {{ return(adapter.dispatch('drop_schema_named', 'dbt') (schema_name)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__drop_schema_named"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3898828, "supported_languages": null}, "macro.dbt.default__drop_schema_named": {"name": "default__drop_schema_named", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/schema.sql", "original_file_path": "macros/relations/schema.sql", "unique_id": "macro.dbt.default__drop_schema_named", "macro_sql": "{% macro default__drop_schema_named(schema_name) %}\n {% set schema_relation = api.Relation.create(schema=schema_name) %}\n {{ adapter.drop_schema(schema_relation) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389895, "supported_languages": null}, "macro.dbt.get_drop_backup_sql": {"name": "get_drop_backup_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/drop_backup.sql", "original_file_path": "macros/relations/drop_backup.sql", "unique_id": "macro.dbt.get_drop_backup_sql", "macro_sql": "{%- macro get_drop_backup_sql(relation) -%}\n {{- log('Applying DROP BACKUP to: ' ~ relation) -}}\n {{- adapter.dispatch('get_drop_backup_sql', 'dbt')(relation) -}}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.default__get_drop_backup_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3899038, "supported_languages": null}, "macro.dbt.default__get_drop_backup_sql": {"name": "default__get_drop_backup_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/drop_backup.sql", "original_file_path": "macros/relations/drop_backup.sql", "unique_id": "macro.dbt.default__get_drop_backup_sql", "macro_sql": "{%- macro default__get_drop_backup_sql(relation) -%}\n\n -- get the standard backup name\n {% set backup_relation = make_backup_relation(relation, relation.type) %}\n\n {{ get_drop_sql(backup_relation) }}\n\n{%- endmacro -%}", "depends_on": {"macros": ["macro.dbt.make_backup_relation", "macro.dbt.get_drop_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389911, "supported_languages": null}, "macro.dbt.get_rename_sql": {"name": "get_rename_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/rename.sql", "original_file_path": "macros/relations/rename.sql", "unique_id": "macro.dbt.get_rename_sql", "macro_sql": "{%- macro get_rename_sql(relation, new_name) -%}\n {{- log('Applying RENAME to: ' ~ relation) -}}\n {{- adapter.dispatch('get_rename_sql', 'dbt')(relation, new_name) -}}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.default__get_rename_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389919, "supported_languages": null}, "macro.dbt.default__get_rename_sql": {"name": "default__get_rename_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/rename.sql", "original_file_path": "macros/relations/rename.sql", "unique_id": "macro.dbt.default__get_rename_sql", "macro_sql": "{%- macro default__get_rename_sql(relation, new_name) -%}\n\n {%- if relation.is_view -%}\n {{ get_rename_view_sql(relation, new_name) }}\n\n {%- elif relation.is_table -%}\n {{ get_rename_table_sql(relation, new_name) }}\n\n {%- elif relation.is_materialized_view -%}\n {{ get_rename_materialized_view_sql(relation, new_name) }}\n\n {%- else -%}\n {{- exceptions.raise_compiler_error(\"`get_rename_sql` has not been implemented for: \" ~ relation.type ) -}}\n\n {%- endif -%}\n\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.get_rename_view_sql", "macro.dbt.get_rename_table_sql", "macro.dbt.get_rename_materialized_view_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389926, "supported_languages": null}, "macro.dbt.rename_relation": {"name": "rename_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/rename.sql", "original_file_path": "macros/relations/rename.sql", "unique_id": "macro.dbt.rename_relation", "macro_sql": "{% macro rename_relation(from_relation, to_relation) -%}\n {{ return(adapter.dispatch('rename_relation', 'dbt')(from_relation, to_relation)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__rename_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.38993, "supported_languages": null}, "macro.dbt.default__rename_relation": {"name": "default__rename_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/rename.sql", "original_file_path": "macros/relations/rename.sql", "unique_id": "macro.dbt.default__rename_relation", "macro_sql": "{% macro default__rename_relation(from_relation, to_relation) -%}\n {% set target_name = adapter.quote_as_configured(to_relation.identifier, 'identifier') %}\n {% call statement('rename_relation') -%}\n alter table {{ from_relation.render() }} rename to {{ target_name }}\n {%- endcall %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3899362, "supported_languages": null}, "macro.dbt.get_create_backup_sql": {"name": "get_create_backup_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/create_backup.sql", "original_file_path": "macros/relations/create_backup.sql", "unique_id": "macro.dbt.get_create_backup_sql", "macro_sql": "{%- macro get_create_backup_sql(relation) -%}\n {{- log('Applying CREATE BACKUP to: ' ~ relation) -}}\n {{- adapter.dispatch('get_create_backup_sql', 'dbt')(relation) -}}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.default__get_create_backup_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389947, "supported_languages": null}, "macro.dbt.default__get_create_backup_sql": {"name": "default__get_create_backup_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/create_backup.sql", "original_file_path": "macros/relations/create_backup.sql", "unique_id": "macro.dbt.default__get_create_backup_sql", "macro_sql": "{%- macro default__get_create_backup_sql(relation) -%}\n\n -- get the standard backup name\n {% set backup_relation = make_backup_relation(relation, relation.type) %}\n\n -- drop any pre-existing backup\n {{ get_drop_sql(backup_relation) }};\n\n {{ get_rename_sql(relation, backup_relation.identifier) }}\n\n{%- endmacro -%}", "depends_on": {"macros": ["macro.dbt.make_backup_relation", "macro.dbt.get_drop_sql", "macro.dbt.get_rename_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389951, "supported_languages": null}, "macro.dbt.get_create_sql": {"name": "get_create_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/create.sql", "original_file_path": "macros/relations/create.sql", "unique_id": "macro.dbt.get_create_sql", "macro_sql": "{%- macro get_create_sql(relation, sql) -%}\n {{- log('Applying CREATE to: ' ~ relation) -}}\n {{- adapter.dispatch('get_create_sql', 'dbt')(relation, sql) -}}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.default__get_create_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389961, "supported_languages": null}, "macro.dbt.default__get_create_sql": {"name": "default__get_create_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/create.sql", "original_file_path": "macros/relations/create.sql", "unique_id": "macro.dbt.default__get_create_sql", "macro_sql": "{%- macro default__get_create_sql(relation, sql) -%}\n\n {%- if relation.is_view -%}\n {{ get_create_view_as_sql(relation, sql) }}\n\n {%- elif relation.is_table -%}\n {{ get_create_table_as_sql(False, relation, sql) }}\n\n {%- elif relation.is_materialized_view -%}\n {{ get_create_materialized_view_as_sql(relation, sql) }}\n\n {%- else -%}\n {{- exceptions.raise_compiler_error(\"`get_create_sql` has not been implemented for: \" ~ relation.type ) -}}\n\n {%- endif -%}\n\n{%- endmacro -%}", "depends_on": {"macros": ["macro.dbt.get_create_view_as_sql", "macro.dbt.get_create_table_as_sql", "macro.dbt.get_create_materialized_view_as_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389965, "supported_languages": null}, "macro.dbt.get_rename_intermediate_sql": {"name": "get_rename_intermediate_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/rename_intermediate.sql", "original_file_path": "macros/relations/rename_intermediate.sql", "unique_id": "macro.dbt.get_rename_intermediate_sql", "macro_sql": "{%- macro get_rename_intermediate_sql(relation) -%}\n {{- log('Applying RENAME INTERMEDIATE to: ' ~ relation) -}}\n {{- adapter.dispatch('get_rename_intermediate_sql', 'dbt')(relation) -}}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": ["macro.dbt.default__get_rename_intermediate_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389973, "supported_languages": null}, "macro.dbt.default__get_rename_intermediate_sql": {"name": "default__get_rename_intermediate_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/rename_intermediate.sql", "original_file_path": "macros/relations/rename_intermediate.sql", "unique_id": "macro.dbt.default__get_rename_intermediate_sql", "macro_sql": "{%- macro default__get_rename_intermediate_sql(relation) -%}\n\n -- get the standard intermediate name\n {% set intermediate_relation = make_intermediate_relation(relation) %}\n\n {{ get_rename_sql(intermediate_relation, relation.identifier) }}\n\n{%- endmacro -%}", "depends_on": {"macros": ["macro.dbt.make_intermediate_relation", "macro.dbt.get_rename_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389977, "supported_languages": null}, "macro.dbt.drop_materialized_view": {"name": "drop_materialized_view", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/drop.sql", "original_file_path": "macros/relations/materialized_view/drop.sql", "unique_id": "macro.dbt.drop_materialized_view", "macro_sql": "{% macro drop_materialized_view(relation) -%}\n {{- adapter.dispatch('drop_materialized_view', 'dbt')(relation) -}}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__drop_materialized_view"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.389986, "supported_languages": null}, "macro.dbt.default__drop_materialized_view": {"name": "default__drop_materialized_view", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/drop.sql", "original_file_path": "macros/relations/materialized_view/drop.sql", "unique_id": "macro.dbt.default__drop_materialized_view", "macro_sql": "{% macro default__drop_materialized_view(relation) -%}\n drop materialized view if exists {{ relation.render() }} cascade\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3899899, "supported_languages": null}, "macro.dbt.get_replace_materialized_view_sql": {"name": "get_replace_materialized_view_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/replace.sql", "original_file_path": "macros/relations/materialized_view/replace.sql", "unique_id": "macro.dbt.get_replace_materialized_view_sql", "macro_sql": "{% macro get_replace_materialized_view_sql(relation, sql) %}\n {{- adapter.dispatch('get_replace_materialized_view_sql', 'dbt')(relation, sql) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_replace_materialized_view_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3899982, "supported_languages": null}, "macro.dbt.default__get_replace_materialized_view_sql": {"name": "default__get_replace_materialized_view_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/replace.sql", "original_file_path": "macros/relations/materialized_view/replace.sql", "unique_id": "macro.dbt.default__get_replace_materialized_view_sql", "macro_sql": "{% macro default__get_replace_materialized_view_sql(relation, sql) %}\n {{ exceptions.raise_compiler_error(\n \"`get_replace_materialized_view_sql` has not been implemented for this adapter.\"\n ) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390002, "supported_languages": null}, "macro.dbt.refresh_materialized_view": {"name": "refresh_materialized_view", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/refresh.sql", "original_file_path": "macros/relations/materialized_view/refresh.sql", "unique_id": "macro.dbt.refresh_materialized_view", "macro_sql": "{% macro refresh_materialized_view(relation) %}\n {{- log('Applying REFRESH to: ' ~ relation) -}}\n {{- adapter.dispatch('refresh_materialized_view', 'dbt')(relation) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__refresh_materialized_view"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3900118, "supported_languages": null}, "macro.dbt.default__refresh_materialized_view": {"name": "default__refresh_materialized_view", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/refresh.sql", "original_file_path": "macros/relations/materialized_view/refresh.sql", "unique_id": "macro.dbt.default__refresh_materialized_view", "macro_sql": "{% macro default__refresh_materialized_view(relation) %}\n {{ exceptions.raise_compiler_error(\"`refresh_materialized_view` has not been implemented for this adapter.\") }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390016, "supported_languages": null}, "macro.dbt.get_rename_materialized_view_sql": {"name": "get_rename_materialized_view_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/rename.sql", "original_file_path": "macros/relations/materialized_view/rename.sql", "unique_id": "macro.dbt.get_rename_materialized_view_sql", "macro_sql": "{% macro get_rename_materialized_view_sql(relation, new_name) %}\n {{- adapter.dispatch('get_rename_materialized_view_sql', 'dbt')(relation, new_name) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_rename_materialized_view_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390023, "supported_languages": null}, "macro.dbt.default__get_rename_materialized_view_sql": {"name": "default__get_rename_materialized_view_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/rename.sql", "original_file_path": "macros/relations/materialized_view/rename.sql", "unique_id": "macro.dbt.default__get_rename_materialized_view_sql", "macro_sql": "{% macro default__get_rename_materialized_view_sql(relation, new_name) %}\n {{ exceptions.raise_compiler_error(\n \"`get_rename_materialized_view_sql` has not been implemented for this adapter.\"\n ) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390029, "supported_languages": null}, "macro.dbt.get_alter_materialized_view_as_sql": {"name": "get_alter_materialized_view_as_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/alter.sql", "original_file_path": "macros/relations/materialized_view/alter.sql", "unique_id": "macro.dbt.get_alter_materialized_view_as_sql", "macro_sql": "{% macro get_alter_materialized_view_as_sql(\n relation,\n configuration_changes,\n sql,\n existing_relation,\n backup_relation,\n intermediate_relation\n) %}\n {{- log('Applying ALTER to: ' ~ relation) -}}\n {{- adapter.dispatch('get_alter_materialized_view_as_sql', 'dbt')(\n relation,\n configuration_changes,\n sql,\n existing_relation,\n backup_relation,\n intermediate_relation\n ) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_alter_materialized_view_as_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390039, "supported_languages": null}, "macro.dbt.default__get_alter_materialized_view_as_sql": {"name": "default__get_alter_materialized_view_as_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/alter.sql", "original_file_path": "macros/relations/materialized_view/alter.sql", "unique_id": "macro.dbt.default__get_alter_materialized_view_as_sql", "macro_sql": "{% macro default__get_alter_materialized_view_as_sql(\n relation,\n configuration_changes,\n sql,\n existing_relation,\n backup_relation,\n intermediate_relation\n) %}\n {{ exceptions.raise_compiler_error(\"Materialized views have not been implemented for this adapter.\") }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390044, "supported_languages": null}, "macro.dbt.get_materialized_view_configuration_changes": {"name": "get_materialized_view_configuration_changes", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/alter.sql", "original_file_path": "macros/relations/materialized_view/alter.sql", "unique_id": "macro.dbt.get_materialized_view_configuration_changes", "macro_sql": "{% macro get_materialized_view_configuration_changes(existing_relation, new_config) %}\n /* {#\n It's recommended that configuration changes be formatted as follows:\n {\"\": [{\"action\": \"\", \"context\": ...}]}\n\n For example:\n {\n \"indexes\": [\n {\"action\": \"drop\", \"context\": \"index_abc\"},\n {\"action\": \"create\", \"context\": {\"columns\": [\"column_1\", \"column_2\"], \"type\": \"hash\", \"unique\": True}},\n ],\n }\n\n Either way, `get_materialized_view_configuration_changes` needs to align with `get_alter_materialized_view_as_sql`.\n #} */\n {{- log('Determining configuration changes on: ' ~ existing_relation) -}}\n {%- do return(adapter.dispatch('get_materialized_view_configuration_changes', 'dbt')(existing_relation, new_config)) -%}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_materialized_view_configuration_changes"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390048, "supported_languages": null}, "macro.dbt.default__get_materialized_view_configuration_changes": {"name": "default__get_materialized_view_configuration_changes", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/alter.sql", "original_file_path": "macros/relations/materialized_view/alter.sql", "unique_id": "macro.dbt.default__get_materialized_view_configuration_changes", "macro_sql": "{% macro default__get_materialized_view_configuration_changes(existing_relation, new_config) %}\n {{ exceptions.raise_compiler_error(\"Materialized views have not been implemented for this adapter.\") }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390052, "supported_languages": null}, "macro.dbt.get_create_materialized_view_as_sql": {"name": "get_create_materialized_view_as_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/create.sql", "original_file_path": "macros/relations/materialized_view/create.sql", "unique_id": "macro.dbt.get_create_materialized_view_as_sql", "macro_sql": "{% macro get_create_materialized_view_as_sql(relation, sql) -%}\n {{- adapter.dispatch('get_create_materialized_view_as_sql', 'dbt')(relation, sql) -}}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_create_materialized_view_as_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.39006, "supported_languages": null}, "macro.dbt.default__get_create_materialized_view_as_sql": {"name": "default__get_create_materialized_view_as_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/materialized_view/create.sql", "original_file_path": "macros/relations/materialized_view/create.sql", "unique_id": "macro.dbt.default__get_create_materialized_view_as_sql", "macro_sql": "{% macro default__get_create_materialized_view_as_sql(relation, sql) -%}\n {{ exceptions.raise_compiler_error(\n \"`get_create_materialized_view_as_sql` has not been implemented for this adapter.\"\n ) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3900642, "supported_languages": null}, "macro.dbt.get_table_columns_and_constraints": {"name": "get_table_columns_and_constraints", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/column/columns_spec_ddl.sql", "original_file_path": "macros/relations/column/columns_spec_ddl.sql", "unique_id": "macro.dbt.get_table_columns_and_constraints", "macro_sql": "{%- macro get_table_columns_and_constraints() -%}\n {{ adapter.dispatch('get_table_columns_and_constraints', 'dbt')() }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__get_table_columns_and_constraints"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390074, "supported_languages": null}, "macro.dbt.default__get_table_columns_and_constraints": {"name": "default__get_table_columns_and_constraints", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/column/columns_spec_ddl.sql", "original_file_path": "macros/relations/column/columns_spec_ddl.sql", "unique_id": "macro.dbt.default__get_table_columns_and_constraints", "macro_sql": "{% macro default__get_table_columns_and_constraints() -%}\n {{ return(table_columns_and_constraints()) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.table_columns_and_constraints"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.39008, "supported_languages": null}, "macro.dbt.table_columns_and_constraints": {"name": "table_columns_and_constraints", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/column/columns_spec_ddl.sql", "original_file_path": "macros/relations/column/columns_spec_ddl.sql", "unique_id": "macro.dbt.table_columns_and_constraints", "macro_sql": "{% macro table_columns_and_constraints() %}\n {# loop through user_provided_columns to create DDL with data types and constraints #}\n {%- set raw_column_constraints = adapter.render_raw_columns_constraints(raw_columns=model['columns']) -%}\n {%- set raw_model_constraints = adapter.render_raw_model_constraints(raw_constraints=model['constraints']) -%}\n (\n {% for c in raw_column_constraints -%}\n {{ c }}{{ \",\" if not loop.last or raw_model_constraints }}\n {% endfor %}\n {% for c in raw_model_constraints -%}\n {{ c }}{{ \",\" if not loop.last }}\n {% endfor -%}\n )\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390086, "supported_languages": null}, "macro.dbt.get_assert_columns_equivalent": {"name": "get_assert_columns_equivalent", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/column/columns_spec_ddl.sql", "original_file_path": "macros/relations/column/columns_spec_ddl.sql", "unique_id": "macro.dbt.get_assert_columns_equivalent", "macro_sql": "\n\n{%- macro get_assert_columns_equivalent(sql) -%}\n {{ adapter.dispatch('get_assert_columns_equivalent', 'dbt')(sql) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__get_assert_columns_equivalent"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.39009, "supported_languages": null}, "macro.dbt.default__get_assert_columns_equivalent": {"name": "default__get_assert_columns_equivalent", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/column/columns_spec_ddl.sql", "original_file_path": "macros/relations/column/columns_spec_ddl.sql", "unique_id": "macro.dbt.default__get_assert_columns_equivalent", "macro_sql": "{% macro default__get_assert_columns_equivalent(sql) -%}\n {{ return(assert_columns_equivalent(sql)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.assert_columns_equivalent"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390094, "supported_languages": null}, "macro.dbt.assert_columns_equivalent": {"name": "assert_columns_equivalent", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/column/columns_spec_ddl.sql", "original_file_path": "macros/relations/column/columns_spec_ddl.sql", "unique_id": "macro.dbt.assert_columns_equivalent", "macro_sql": "{% macro assert_columns_equivalent(sql) %}\n\n {#-- First ensure the user has defined 'columns' in yaml specification --#}\n {%- set user_defined_columns = model['columns'] -%}\n {%- if not user_defined_columns -%}\n {{ exceptions.raise_contract_error([], []) }}\n {%- endif -%}\n\n {#-- Obtain the column schema provided by sql file. #}\n {%- set sql_file_provided_columns = get_column_schema_from_query(sql, config.get('sql_header', none)) -%}\n {#--Obtain the column schema provided by the schema file by generating an 'empty schema' query from the model's columns. #}\n {%- set schema_file_provided_columns = get_column_schema_from_query(get_empty_schema_sql(user_defined_columns)) -%}\n\n {#-- create dictionaries with name and formatted data type and strings for exception #}\n {%- set sql_columns = format_columns(sql_file_provided_columns) -%}\n {%- set yaml_columns = format_columns(schema_file_provided_columns) -%}\n\n {%- if sql_columns|length != yaml_columns|length -%}\n {%- do exceptions.raise_contract_error(yaml_columns, sql_columns) -%}\n {%- endif -%}\n\n {%- for sql_col in sql_columns -%}\n {%- set yaml_col = [] -%}\n {%- for this_col in yaml_columns -%}\n {%- if this_col['name'] == sql_col['name'] -%}\n {%- do yaml_col.append(this_col) -%}\n {%- break -%}\n {%- endif -%}\n {%- endfor -%}\n {%- if not yaml_col -%}\n {#-- Column with name not found in yaml #}\n {%- do exceptions.raise_contract_error(yaml_columns, sql_columns) -%}\n {%- endif -%}\n {%- if sql_col['formatted'] != yaml_col[0]['formatted'] -%}\n {#-- Column data types don't match #}\n {%- do exceptions.raise_contract_error(yaml_columns, sql_columns) -%}\n {%- endif -%}\n {%- endfor -%}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_column_schema_from_query", "macro.dbt.get_empty_schema_sql", "macro.dbt.format_columns"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390101, "supported_languages": null}, "macro.dbt.format_columns": {"name": "format_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/column/columns_spec_ddl.sql", "original_file_path": "macros/relations/column/columns_spec_ddl.sql", "unique_id": "macro.dbt.format_columns", "macro_sql": "{% macro format_columns(columns) %}\n {% set formatted_columns = [] %}\n {% for column in columns %}\n {%- set formatted_column = adapter.dispatch('format_column', 'dbt')(column) -%}\n {%- do formatted_columns.append(formatted_column) -%}\n {% endfor %}\n {{ return(formatted_columns) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__format_column"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3901079, "supported_languages": null}, "macro.dbt.default__format_column": {"name": "default__format_column", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/column/columns_spec_ddl.sql", "original_file_path": "macros/relations/column/columns_spec_ddl.sql", "unique_id": "macro.dbt.default__format_column", "macro_sql": "{% macro default__format_column(column) -%}\n {% set data_type = column.dtype %}\n {% set formatted = column.column.lower() ~ \" \" ~ data_type %}\n {{ return({'name': column.name, 'data_type': data_type, 'formatted': formatted}) }}\n{%- endmacro -%}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390113, "supported_languages": null}, "macro.dbt.drop_table": {"name": "drop_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/drop.sql", "original_file_path": "macros/relations/table/drop.sql", "unique_id": "macro.dbt.drop_table", "macro_sql": "{% macro drop_table(relation) -%}\n {{- adapter.dispatch('drop_table', 'dbt')(relation) -}}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__drop_table"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390121, "supported_languages": null}, "macro.dbt.default__drop_table": {"name": "default__drop_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/drop.sql", "original_file_path": "macros/relations/table/drop.sql", "unique_id": "macro.dbt.default__drop_table", "macro_sql": "{% macro default__drop_table(relation) -%}\n drop table if exists {{ relation.render() }} cascade\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390125, "supported_languages": null}, "macro.dbt.get_replace_table_sql": {"name": "get_replace_table_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/replace.sql", "original_file_path": "macros/relations/table/replace.sql", "unique_id": "macro.dbt.get_replace_table_sql", "macro_sql": "{% macro get_replace_table_sql(relation, sql) %}\n {{- adapter.dispatch('get_replace_table_sql', 'dbt')(relation, sql) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_replace_table_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390135, "supported_languages": null}, "macro.dbt.default__get_replace_table_sql": {"name": "default__get_replace_table_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/replace.sql", "original_file_path": "macros/relations/table/replace.sql", "unique_id": "macro.dbt.default__get_replace_table_sql", "macro_sql": "{% macro default__get_replace_table_sql(relation, sql) %}\n {{ exceptions.raise_compiler_error(\n \"`get_replace_table_sql` has not been implemented for this adapter.\"\n ) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390141, "supported_languages": null}, "macro.dbt.get_rename_table_sql": {"name": "get_rename_table_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/rename.sql", "original_file_path": "macros/relations/table/rename.sql", "unique_id": "macro.dbt.get_rename_table_sql", "macro_sql": "{% macro get_rename_table_sql(relation, new_name) %}\n {{- adapter.dispatch('get_rename_table_sql', 'dbt')(relation, new_name) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_rename_table_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390149, "supported_languages": null}, "macro.dbt.default__get_rename_table_sql": {"name": "default__get_rename_table_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/rename.sql", "original_file_path": "macros/relations/table/rename.sql", "unique_id": "macro.dbt.default__get_rename_table_sql", "macro_sql": "{% macro default__get_rename_table_sql(relation, new_name) %}\n {{ exceptions.raise_compiler_error(\n \"`get_rename_table_sql` has not been implemented for this adapter.\"\n ) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390153, "supported_languages": null}, "macro.dbt.get_create_table_as_sql": {"name": "get_create_table_as_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/create.sql", "original_file_path": "macros/relations/table/create.sql", "unique_id": "macro.dbt.get_create_table_as_sql", "macro_sql": "{% macro get_create_table_as_sql(temporary, relation, sql) -%}\n {{ adapter.dispatch('get_create_table_as_sql', 'dbt')(temporary, relation, sql) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_create_table_as_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3901608, "supported_languages": null}, "macro.dbt.default__get_create_table_as_sql": {"name": "default__get_create_table_as_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/create.sql", "original_file_path": "macros/relations/table/create.sql", "unique_id": "macro.dbt.default__get_create_table_as_sql", "macro_sql": "{% macro default__get_create_table_as_sql(temporary, relation, sql) -%}\n {{ return(create_table_as(temporary, relation, sql)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.create_table_as"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390165, "supported_languages": null}, "macro.dbt.create_table_as": {"name": "create_table_as", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/create.sql", "original_file_path": "macros/relations/table/create.sql", "unique_id": "macro.dbt.create_table_as", "macro_sql": "{% macro create_table_as(temporary, relation, compiled_code, language='sql') -%}\n {# backward compatibility for create_table_as that does not support language #}\n {% if language == \"sql\" %}\n {{ adapter.dispatch('create_table_as', 'dbt')(temporary, relation, compiled_code)}}\n {% else %}\n {{ adapter.dispatch('create_table_as', 'dbt')(temporary, relation, compiled_code, language) }}\n {% endif %}\n\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__create_table_as"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390173, "supported_languages": null}, "macro.dbt.default__create_table_as": {"name": "default__create_table_as", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/create.sql", "original_file_path": "macros/relations/table/create.sql", "unique_id": "macro.dbt.default__create_table_as", "macro_sql": "{% macro default__create_table_as(temporary, relation, sql) -%}\n {%- set sql_header = config.get('sql_header', none) -%}\n\n {{ sql_header if sql_header is not none }}\n\n create {% if temporary: -%}temporary{%- endif %} table\n {{ relation.include(database=(not temporary), schema=(not temporary)) }}\n {% set contract_config = config.get('contract') %}\n {% if contract_config.enforced and (not temporary) %}\n {{ get_assert_columns_equivalent(sql) }}\n {{ get_table_columns_and_constraints() }}\n {%- set sql = get_select_subquery(sql) %}\n {% endif %}\n as (\n {{ sql }}\n );\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.get_assert_columns_equivalent", "macro.dbt.get_table_columns_and_constraints", "macro.dbt.get_select_subquery"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390177, "supported_languages": null}, "macro.dbt.default__get_column_names": {"name": "default__get_column_names", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/create.sql", "original_file_path": "macros/relations/table/create.sql", "unique_id": "macro.dbt.default__get_column_names", "macro_sql": "{% macro default__get_column_names() %}\n {#- loop through user_provided_columns to get column names -#}\n {%- set user_provided_columns = model['columns'] -%}\n {%- for i in user_provided_columns %}\n {%- set col = user_provided_columns[i] -%}\n {%- set col_name = adapter.quote(col['name']) if col.get('quote') else col['name'] -%}\n {{ col_name }}{{ \", \" if not loop.last }}\n {%- endfor -%}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3901849, "supported_languages": null}, "macro.dbt.get_select_subquery": {"name": "get_select_subquery", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/create.sql", "original_file_path": "macros/relations/table/create.sql", "unique_id": "macro.dbt.get_select_subquery", "macro_sql": "{% macro get_select_subquery(sql) %}\n {{ return(adapter.dispatch('get_select_subquery', 'dbt')(sql)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_select_subquery"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390202, "supported_languages": null}, "macro.dbt.default__get_select_subquery": {"name": "default__get_select_subquery", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/table/create.sql", "original_file_path": "macros/relations/table/create.sql", "unique_id": "macro.dbt.default__get_select_subquery", "macro_sql": "{% macro default__get_select_subquery(sql) %}\n select {{ adapter.dispatch('get_column_names', 'dbt')() }}\n from (\n {{ sql }}\n ) as model_subq\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_column_names"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3902059, "supported_languages": null}, "macro.dbt.drop_view": {"name": "drop_view", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/drop.sql", "original_file_path": "macros/relations/view/drop.sql", "unique_id": "macro.dbt.drop_view", "macro_sql": "{% macro drop_view(relation) -%}\n {{- adapter.dispatch('drop_view', 'dbt')(relation) -}}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__drop_view"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390214, "supported_languages": null}, "macro.dbt.default__drop_view": {"name": "default__drop_view", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/drop.sql", "original_file_path": "macros/relations/view/drop.sql", "unique_id": "macro.dbt.default__drop_view", "macro_sql": "{% macro default__drop_view(relation) -%}\n drop view if exists {{ relation.render() }} cascade\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.39022, "supported_languages": null}, "macro.dbt.get_replace_view_sql": {"name": "get_replace_view_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/replace.sql", "original_file_path": "macros/relations/view/replace.sql", "unique_id": "macro.dbt.get_replace_view_sql", "macro_sql": "{% macro get_replace_view_sql(relation, sql) %}\n {{- adapter.dispatch('get_replace_view_sql', 'dbt')(relation, sql) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_replace_view_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3902311, "supported_languages": null}, "macro.dbt.default__get_replace_view_sql": {"name": "default__get_replace_view_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/replace.sql", "original_file_path": "macros/relations/view/replace.sql", "unique_id": "macro.dbt.default__get_replace_view_sql", "macro_sql": "{% macro default__get_replace_view_sql(relation, sql) %}\n {{ exceptions.raise_compiler_error(\n \"`get_replace_view_sql` has not been implemented for this adapter.\"\n ) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390235, "supported_languages": null}, "macro.dbt.create_or_replace_view": {"name": "create_or_replace_view", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/replace.sql", "original_file_path": "macros/relations/view/replace.sql", "unique_id": "macro.dbt.create_or_replace_view", "macro_sql": "{% macro create_or_replace_view() %}\n {%- set identifier = model['alias'] -%}\n\n {%- set old_relation = adapter.get_relation(database=database, schema=schema, identifier=identifier) -%}\n {%- set exists_as_view = (old_relation is not none and old_relation.is_view) -%}\n\n {%- set target_relation = api.Relation.create(\n identifier=identifier, schema=schema, database=database,\n type='view') -%}\n {% set grant_config = config.get('grants') %}\n\n {{ run_hooks(pre_hooks) }}\n\n -- If there's a table with the same name and we weren't told to full refresh,\n -- that's an error. If we were told to full refresh, drop it. This behavior differs\n -- for Snowflake and BigQuery, so multiple dispatch is used.\n {%- if old_relation is not none and old_relation.is_table -%}\n {{ handle_existing_table(should_full_refresh(), old_relation) }}\n {%- endif -%}\n\n -- build model\n {% call statement('main') -%}\n {{ get_create_view_as_sql(target_relation, sql) }}\n {%- endcall %}\n\n {% set should_revoke = should_revoke(exists_as_view, full_refresh_mode=True) %}\n {% do apply_grants(target_relation, grant_config, should_revoke=should_revoke) %}\n\n {{ run_hooks(post_hooks) }}\n\n {{ return({'relations': [target_relation]}) }}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.run_hooks", "macro.dbt.handle_existing_table", "macro.dbt.should_full_refresh", "macro.dbt.statement", "macro.dbt.get_create_view_as_sql", "macro.dbt.should_revoke", "macro.dbt.apply_grants"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390243, "supported_languages": null}, "macro.dbt.handle_existing_table": {"name": "handle_existing_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/replace.sql", "original_file_path": "macros/relations/view/replace.sql", "unique_id": "macro.dbt.handle_existing_table", "macro_sql": "{% macro handle_existing_table(full_refresh, old_relation) %}\n {{ adapter.dispatch('handle_existing_table', 'dbt')(full_refresh, old_relation) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__handle_existing_table"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390246, "supported_languages": null}, "macro.dbt.default__handle_existing_table": {"name": "default__handle_existing_table", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/replace.sql", "original_file_path": "macros/relations/view/replace.sql", "unique_id": "macro.dbt.default__handle_existing_table", "macro_sql": "{% macro default__handle_existing_table(full_refresh, old_relation) %}\n {{ log(\"Dropping relation \" ~ old_relation.render() ~ \" because it is of type \" ~ old_relation.type) }}\n {{ adapter.drop_relation(old_relation) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390253, "supported_languages": null}, "macro.dbt.get_rename_view_sql": {"name": "get_rename_view_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/rename.sql", "original_file_path": "macros/relations/view/rename.sql", "unique_id": "macro.dbt.get_rename_view_sql", "macro_sql": "{% macro get_rename_view_sql(relation, new_name) %}\n {{- adapter.dispatch('get_rename_view_sql', 'dbt')(relation, new_name) -}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_rename_view_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390263, "supported_languages": null}, "macro.dbt.default__get_rename_view_sql": {"name": "default__get_rename_view_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/rename.sql", "original_file_path": "macros/relations/view/rename.sql", "unique_id": "macro.dbt.default__get_rename_view_sql", "macro_sql": "{% macro default__get_rename_view_sql(relation, new_name) %}\n {{ exceptions.raise_compiler_error(\n \"`get_rename_view_sql` has not been implemented for this adapter.\"\n ) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390267, "supported_languages": null}, "macro.dbt.get_create_view_as_sql": {"name": "get_create_view_as_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/create.sql", "original_file_path": "macros/relations/view/create.sql", "unique_id": "macro.dbt.get_create_view_as_sql", "macro_sql": "{% macro get_create_view_as_sql(relation, sql) -%}\n {{ adapter.dispatch('get_create_view_as_sql', 'dbt')(relation, sql) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_create_view_as_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3902762, "supported_languages": null}, "macro.dbt.default__get_create_view_as_sql": {"name": "default__get_create_view_as_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/create.sql", "original_file_path": "macros/relations/view/create.sql", "unique_id": "macro.dbt.default__get_create_view_as_sql", "macro_sql": "{% macro default__get_create_view_as_sql(relation, sql) -%}\n {{ return(create_view_as(relation, sql)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.create_view_as"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3902788, "supported_languages": null}, "macro.dbt.create_view_as": {"name": "create_view_as", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/create.sql", "original_file_path": "macros/relations/view/create.sql", "unique_id": "macro.dbt.create_view_as", "macro_sql": "{% macro create_view_as(relation, sql) -%}\n {{ adapter.dispatch('create_view_as', 'dbt')(relation, sql) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__create_view_as"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390285, "supported_languages": null}, "macro.dbt.default__create_view_as": {"name": "default__create_view_as", "resource_type": "macro", "package_name": "dbt", "path": "macros/relations/view/create.sql", "original_file_path": "macros/relations/view/create.sql", "unique_id": "macro.dbt.default__create_view_as", "macro_sql": "{% macro default__create_view_as(relation, sql) -%}\n {%- set sql_header = config.get('sql_header', none) -%}\n\n {{ sql_header if sql_header is not none }}\n create view {{ relation.render() }}\n {% set contract_config = config.get('contract') %}\n {% if contract_config.enforced %}\n {{ get_assert_columns_equivalent(sql) }}\n {%- endif %}\n as (\n {{ sql }}\n );\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.get_assert_columns_equivalent"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3902888, "supported_languages": null}, "macro.dbt.default__test_relationships": {"name": "default__test_relationships", "resource_type": "macro", "package_name": "dbt", "path": "macros/generic_test_sql/relationships.sql", "original_file_path": "macros/generic_test_sql/relationships.sql", "unique_id": "macro.dbt.default__test_relationships", "macro_sql": "{% macro default__test_relationships(model, column_name, to, field) %}\n\nwith child as (\n select {{ column_name }} as from_field\n from {{ model }}\n where {{ column_name }} is not null\n),\n\nparent as (\n select {{ field }} as to_field\n from {{ to }}\n)\n\nselect\n from_field\n\nfrom child\nleft join parent\n on child.from_field = parent.to_field\n\nwhere parent.to_field is null\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390302, "supported_languages": null}, "macro.dbt.default__test_not_null": {"name": "default__test_not_null", "resource_type": "macro", "package_name": "dbt", "path": "macros/generic_test_sql/not_null.sql", "original_file_path": "macros/generic_test_sql/not_null.sql", "unique_id": "macro.dbt.default__test_not_null", "macro_sql": "{% macro default__test_not_null(model, column_name) %}\n\n{% set column_list = '*' if should_store_failures() else column_name %}\n\nselect {{ column_list }}\nfrom {{ model }}\nwhere {{ column_name }} is null\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.should_store_failures"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390309, "supported_languages": null}, "macro.dbt.default__test_unique": {"name": "default__test_unique", "resource_type": "macro", "package_name": "dbt", "path": "macros/generic_test_sql/unique.sql", "original_file_path": "macros/generic_test_sql/unique.sql", "unique_id": "macro.dbt.default__test_unique", "macro_sql": "{% macro default__test_unique(model, column_name) %}\n\nselect\n {{ column_name }} as unique_field,\n count(*) as n_records\n\nfrom {{ model }}\nwhere {{ column_name }} is not null\ngroup by {{ column_name }}\nhaving count(*) > 1\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3903182, "supported_languages": null}, "macro.dbt.default__test_accepted_values": {"name": "default__test_accepted_values", "resource_type": "macro", "package_name": "dbt", "path": "macros/generic_test_sql/accepted_values.sql", "original_file_path": "macros/generic_test_sql/accepted_values.sql", "unique_id": "macro.dbt.default__test_accepted_values", "macro_sql": "{% macro default__test_accepted_values(model, column_name, values, quote=True) %}\n\nwith all_values as (\n\n select\n {{ column_name }} as value_field,\n count(*) as n_records\n\n from {{ model }}\n group by {{ column_name }}\n\n)\n\nselect *\nfrom all_values\nwhere value_field not in (\n {% for value in values -%}\n {% if quote -%}\n '{{ value }}'\n {%- else -%}\n {{ value }}\n {%- endif -%}\n {%- if not loop.last -%},{%- endif %}\n {%- endfor %}\n)\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390325, "supported_languages": null}, "macro.dbt.statement": {"name": "statement", "resource_type": "macro", "package_name": "dbt", "path": "macros/etc/statement.sql", "original_file_path": "macros/etc/statement.sql", "unique_id": "macro.dbt.statement", "macro_sql": "\n{%- macro statement(name=None, fetch_result=False, auto_begin=True, language='sql') -%}\n {%- if execute: -%}\n {%- set compiled_code = caller() -%}\n\n {%- if name == 'main' -%}\n {{ log('Writing runtime {} for node \"{}\"'.format(language, model['unique_id'])) }}\n {{ write(compiled_code) }}\n {%- endif -%}\n {%- if language == 'sql'-%}\n {%- set res, table = adapter.execute(compiled_code, auto_begin=auto_begin, fetch=fetch_result) -%}\n {%- elif language == 'python' -%}\n {%- set res = submit_python_job(model, compiled_code) -%}\n {#-- TODO: What should table be for python models? --#}\n {%- set table = None -%}\n {%- else -%}\n {% do exceptions.raise_compiler_error(\"statement macro didn't get supported language\") %}\n {%- endif -%}\n\n {%- if name is not none -%}\n {{ store_result(name, response=res, agate_table=table) }}\n {%- endif -%}\n\n {%- endif -%}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390334, "supported_languages": null}, "macro.dbt.noop_statement": {"name": "noop_statement", "resource_type": "macro", "package_name": "dbt", "path": "macros/etc/statement.sql", "original_file_path": "macros/etc/statement.sql", "unique_id": "macro.dbt.noop_statement", "macro_sql": "{% macro noop_statement(name=None, message=None, code=None, rows_affected=None, res=None) -%}\n {%- set sql = caller() -%}\n\n {%- if name == 'main' -%}\n {{ log('Writing runtime SQL for node \"{}\"'.format(model['unique_id'])) }}\n {{ write(sql) }}\n {%- endif -%}\n\n {%- if name is not none -%}\n {{ store_raw_result(name, message=message, code=code, rows_affected=rows_affected, agate_table=res) }}\n {%- endif -%}\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3903391, "supported_languages": null}, "macro.dbt.run_query": {"name": "run_query", "resource_type": "macro", "package_name": "dbt", "path": "macros/etc/statement.sql", "original_file_path": "macros/etc/statement.sql", "unique_id": "macro.dbt.run_query", "macro_sql": "{% macro run_query(sql) %}\n {% call statement(\"run_query_statement\", fetch_result=true, auto_begin=false) %}\n {{ sql }}\n {% endcall %}\n\n {% do return(load_result(\"run_query_statement\").table) %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3903449, "supported_languages": null}, "macro.dbt.convert_datetime": {"name": "convert_datetime", "resource_type": "macro", "package_name": "dbt", "path": "macros/etc/datetime.sql", "original_file_path": "macros/etc/datetime.sql", "unique_id": "macro.dbt.convert_datetime", "macro_sql": "{% macro convert_datetime(date_str, date_fmt) %}\n\n {% set error_msg -%}\n The provided partition date '{{ date_str }}' does not match the expected format '{{ date_fmt }}'\n {%- endset %}\n\n {% set res = try_or_compiler_error(error_msg, modules.datetime.datetime.strptime, date_str.strip(), date_fmt) %}\n {{ return(res) }}\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390354, "supported_languages": null}, "macro.dbt.dates_in_range": {"name": "dates_in_range", "resource_type": "macro", "package_name": "dbt", "path": "macros/etc/datetime.sql", "original_file_path": "macros/etc/datetime.sql", "unique_id": "macro.dbt.dates_in_range", "macro_sql": "{% macro dates_in_range(start_date_str, end_date_str=none, in_fmt=\"%Y%m%d\", out_fmt=\"%Y%m%d\") %}\n {% set end_date_str = start_date_str if end_date_str is none else end_date_str %}\n\n {% set start_date = convert_datetime(start_date_str, in_fmt) %}\n {% set end_date = convert_datetime(end_date_str, in_fmt) %}\n\n {% set day_count = (end_date - start_date).days %}\n {% if day_count < 0 %}\n {% set msg -%}\n Partition start date is after the end date ({{ start_date }}, {{ end_date }})\n {%- endset %}\n\n {{ exceptions.raise_compiler_error(msg, model) }}\n {% endif %}\n\n {% set date_list = [] %}\n {% for i in range(0, day_count + 1) %}\n {% set the_date = (modules.datetime.timedelta(days=i) + start_date) %}\n {% if not out_fmt %}\n {% set _ = date_list.append(the_date) %}\n {% else %}\n {% set _ = date_list.append(the_date.strftime(out_fmt)) %}\n {% endif %}\n {% endfor %}\n\n {{ return(date_list) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.convert_datetime"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390359, "supported_languages": null}, "macro.dbt.partition_range": {"name": "partition_range", "resource_type": "macro", "package_name": "dbt", "path": "macros/etc/datetime.sql", "original_file_path": "macros/etc/datetime.sql", "unique_id": "macro.dbt.partition_range", "macro_sql": "{% macro partition_range(raw_partition_date, date_fmt='%Y%m%d') %}\n {% set partition_range = (raw_partition_date | string).split(\",\") %}\n\n {% if (partition_range | length) == 1 %}\n {% set start_date = partition_range[0] %}\n {% set end_date = none %}\n {% elif (partition_range | length) == 2 %}\n {% set start_date = partition_range[0] %}\n {% set end_date = partition_range[1] %}\n {% else %}\n {{ exceptions.raise_compiler_error(\"Invalid partition time. Expected format: {Start Date}[,{End Date}]. Got: \" ~ raw_partition_date) }}\n {% endif %}\n\n {{ return(dates_in_range(start_date, end_date, in_fmt=date_fmt)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.dates_in_range"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390364, "supported_languages": null}, "macro.dbt.py_current_timestring": {"name": "py_current_timestring", "resource_type": "macro", "package_name": "dbt", "path": "macros/etc/datetime.sql", "original_file_path": "macros/etc/datetime.sql", "unique_id": "macro.dbt.py_current_timestring", "macro_sql": "{% macro py_current_timestring() %}\n {% set dt = modules.datetime.datetime.now() %}\n {% do return(dt.strftime(\"%Y%m%d%H%M%S%f\")) %}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390369, "supported_languages": null}, "macro.dbt.except": {"name": "except", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/except.sql", "original_file_path": "macros/utils/except.sql", "unique_id": "macro.dbt.except", "macro_sql": "{% macro except() %}\n {{ return(adapter.dispatch('except', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__except"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3903759, "supported_languages": null}, "macro.dbt.default__except": {"name": "default__except", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/except.sql", "original_file_path": "macros/utils/except.sql", "unique_id": "macro.dbt.default__except", "macro_sql": "{% macro default__except() %}\n\n except\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3903801, "supported_languages": null}, "macro.dbt.get_intervals_between": {"name": "get_intervals_between", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/date_spine.sql", "original_file_path": "macros/utils/date_spine.sql", "unique_id": "macro.dbt.get_intervals_between", "macro_sql": "{% macro get_intervals_between(start_date, end_date, datepart) -%}\n {{ return(adapter.dispatch('get_intervals_between', 'dbt')(start_date, end_date, datepart)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_intervals_between"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.39039, "supported_languages": null}, "macro.dbt.default__get_intervals_between": {"name": "default__get_intervals_between", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/date_spine.sql", "original_file_path": "macros/utils/date_spine.sql", "unique_id": "macro.dbt.default__get_intervals_between", "macro_sql": "{% macro default__get_intervals_between(start_date, end_date, datepart) -%}\n {%- call statement('get_intervals_between', fetch_result=True) %}\n\n select {{ dbt.datediff(start_date, end_date, datepart) }}\n\n {%- endcall -%}\n\n {%- set value_list = load_result('get_intervals_between') -%}\n\n {%- if value_list and value_list['data'] -%}\n {%- set values = value_list['data'] | map(attribute=0) | list %}\n {{ return(values[0]) }}\n {%- else -%}\n {{ return(1) }}\n {%- endif -%}\n\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.statement", "macro.dbt.datediff"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390395, "supported_languages": null}, "macro.dbt.date_spine": {"name": "date_spine", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/date_spine.sql", "original_file_path": "macros/utils/date_spine.sql", "unique_id": "macro.dbt.date_spine", "macro_sql": "{% macro date_spine(datepart, start_date, end_date) %}\n {{ return(adapter.dispatch('date_spine', 'dbt')(datepart, start_date, end_date)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__date_spine"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390404, "supported_languages": null}, "macro.dbt.default__date_spine": {"name": "default__date_spine", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/date_spine.sql", "original_file_path": "macros/utils/date_spine.sql", "unique_id": "macro.dbt.default__date_spine", "macro_sql": "{% macro default__date_spine(datepart, start_date, end_date) %}\n\n\n {# call as follows:\n\n date_spine(\n \"day\",\n \"to_date('01/01/2016', 'mm/dd/yyyy')\",\n \"dbt.dateadd(week, 1, current_date)\"\n ) #}\n\n\n with rawdata as (\n\n {{dbt.generate_series(\n dbt.get_intervals_between(start_date, end_date, datepart)\n )}}\n\n ),\n\n all_periods as (\n\n select (\n {{\n dbt.dateadd(\n datepart,\n \"row_number() over (order by 1) - 1\",\n start_date\n )\n }}\n ) as date_{{datepart}}\n from rawdata\n\n ),\n\n filtered as (\n\n select *\n from all_periods\n where date_{{datepart}} <= {{ end_date }}\n\n )\n\n select * from filtered\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.generate_series", "macro.dbt.get_intervals_between", "macro.dbt.dateadd"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390408, "supported_languages": null}, "macro.dbt.date": {"name": "date", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/date.sql", "original_file_path": "macros/utils/date.sql", "unique_id": "macro.dbt.date", "macro_sql": "{% macro date(year, month, day) %}\n {{ return(adapter.dispatch('date', 'dbt') (year, month, day)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__date"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3904178, "supported_languages": null}, "macro.dbt.default__date": {"name": "default__date", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/date.sql", "original_file_path": "macros/utils/date.sql", "unique_id": "macro.dbt.default__date", "macro_sql": "{% macro default__date(year, month, day) -%}\n {%- set dt = modules.datetime.date(year, month, day) -%}\n {%- set iso_8601_formatted_date = dt.strftime('%Y-%m-%d') -%}\n to_date('{{ iso_8601_formatted_date }}', 'YYYY-MM-DD')\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390423, "supported_languages": null}, "macro.dbt.replace": {"name": "replace", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/replace.sql", "original_file_path": "macros/utils/replace.sql", "unique_id": "macro.dbt.replace", "macro_sql": "{% macro replace(field, old_chars, new_chars) -%}\n {{ return(adapter.dispatch('replace', 'dbt') (field, old_chars, new_chars)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__replace"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390435, "supported_languages": null}, "macro.dbt.default__replace": {"name": "default__replace", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/replace.sql", "original_file_path": "macros/utils/replace.sql", "unique_id": "macro.dbt.default__replace", "macro_sql": "{% macro default__replace(field, old_chars, new_chars) %}\n\n replace(\n {{ field }},\n {{ old_chars }},\n {{ new_chars }}\n )\n\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3904388, "supported_languages": null}, "macro.dbt.concat": {"name": "concat", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/concat.sql", "original_file_path": "macros/utils/concat.sql", "unique_id": "macro.dbt.concat", "macro_sql": "{% macro concat(fields) -%}\n {{ return(adapter.dispatch('concat', 'dbt')(fields)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__concat"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3904462, "supported_languages": null}, "macro.dbt.default__concat": {"name": "default__concat", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/concat.sql", "original_file_path": "macros/utils/concat.sql", "unique_id": "macro.dbt.default__concat", "macro_sql": "{% macro default__concat(fields) -%}\n {{ fields|join(' || ') }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.39045, "supported_languages": null}, "macro.dbt.get_powers_of_two": {"name": "get_powers_of_two", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/generate_series.sql", "original_file_path": "macros/utils/generate_series.sql", "unique_id": "macro.dbt.get_powers_of_two", "macro_sql": "{% macro get_powers_of_two(upper_bound) %}\n {{ return(adapter.dispatch('get_powers_of_two', 'dbt')(upper_bound)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_powers_of_two"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390458, "supported_languages": null}, "macro.dbt.default__get_powers_of_two": {"name": "default__get_powers_of_two", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/generate_series.sql", "original_file_path": "macros/utils/generate_series.sql", "unique_id": "macro.dbt.default__get_powers_of_two", "macro_sql": "{% macro default__get_powers_of_two(upper_bound) %}\n\n {% if upper_bound <= 0 %}\n {{ exceptions.raise_compiler_error(\"upper bound must be positive\") }}\n {% endif %}\n\n {% for _ in range(1, 100) %}\n {% if upper_bound <= 2 ** loop.index %}{{ return(loop.index) }}{% endif %}\n {% endfor %}\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390462, "supported_languages": null}, "macro.dbt.generate_series": {"name": "generate_series", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/generate_series.sql", "original_file_path": "macros/utils/generate_series.sql", "unique_id": "macro.dbt.generate_series", "macro_sql": "{% macro generate_series(upper_bound) %}\n {{ return(adapter.dispatch('generate_series', 'dbt')(upper_bound)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__generate_series"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3904672, "supported_languages": null}, "macro.dbt.default__generate_series": {"name": "default__generate_series", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/generate_series.sql", "original_file_path": "macros/utils/generate_series.sql", "unique_id": "macro.dbt.default__generate_series", "macro_sql": "{% macro default__generate_series(upper_bound) %}\n\n {% set n = dbt.get_powers_of_two(upper_bound) %}\n\n with p as (\n select 0 as generated_number union all select 1\n ), unioned as (\n\n select\n\n {% for i in range(n) %}\n p{{i}}.generated_number * power(2, {{i}})\n {% if not loop.last %} + {% endif %}\n {% endfor %}\n + 1\n as generated_number\n\n from\n\n {% for i in range(n) %}\n p as p{{i}}\n {% if not loop.last %} cross join {% endif %}\n {% endfor %}\n\n )\n\n select *\n from unioned\n where generated_number <= {{upper_bound}}\n order by generated_number\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_powers_of_two"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390473, "supported_languages": null}, "macro.dbt.length": {"name": "length", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/length.sql", "original_file_path": "macros/utils/length.sql", "unique_id": "macro.dbt.length", "macro_sql": "{% macro length(expression) -%}\n {{ return(adapter.dispatch('length', 'dbt') (expression)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__length"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3904798, "supported_languages": null}, "macro.dbt.default__length": {"name": "default__length", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/length.sql", "original_file_path": "macros/utils/length.sql", "unique_id": "macro.dbt.default__length", "macro_sql": "{% macro default__length(expression) %}\n\n length(\n {{ expression }}\n )\n\n{%- endmacro -%}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3904839, "supported_languages": null}, "macro.dbt.dateadd": {"name": "dateadd", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/dateadd.sql", "original_file_path": "macros/utils/dateadd.sql", "unique_id": "macro.dbt.dateadd", "macro_sql": "{% macro dateadd(datepart, interval, from_date_or_timestamp) %}\n {{ return(adapter.dispatch('dateadd', 'dbt')(datepart, interval, from_date_or_timestamp)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__dateadd"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3904908, "supported_languages": null}, "macro.dbt.default__dateadd": {"name": "default__dateadd", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/dateadd.sql", "original_file_path": "macros/utils/dateadd.sql", "unique_id": "macro.dbt.default__dateadd", "macro_sql": "{% macro default__dateadd(datepart, interval, from_date_or_timestamp) %}\n\n dateadd(\n {{ datepart }},\n {{ interval }},\n {{ from_date_or_timestamp }}\n )\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390499, "supported_languages": null}, "macro.dbt.intersect": {"name": "intersect", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/intersect.sql", "original_file_path": "macros/utils/intersect.sql", "unique_id": "macro.dbt.intersect", "macro_sql": "{% macro intersect() %}\n {{ return(adapter.dispatch('intersect', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__intersect"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390506, "supported_languages": null}, "macro.dbt.default__intersect": {"name": "default__intersect", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/intersect.sql", "original_file_path": "macros/utils/intersect.sql", "unique_id": "macro.dbt.default__intersect", "macro_sql": "{% macro default__intersect() %}\n\n intersect\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3905122, "supported_languages": null}, "macro.dbt.escape_single_quotes": {"name": "escape_single_quotes", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/escape_single_quotes.sql", "original_file_path": "macros/utils/escape_single_quotes.sql", "unique_id": "macro.dbt.escape_single_quotes", "macro_sql": "{% macro escape_single_quotes(expression) %}\n {{ return(adapter.dispatch('escape_single_quotes', 'dbt') (expression)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__escape_single_quotes"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3905191, "supported_languages": null}, "macro.dbt.default__escape_single_quotes": {"name": "default__escape_single_quotes", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/escape_single_quotes.sql", "original_file_path": "macros/utils/escape_single_quotes.sql", "unique_id": "macro.dbt.default__escape_single_quotes", "macro_sql": "{% macro default__escape_single_quotes(expression) -%}\n{{ expression | replace(\"'\",\"''\") }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390523, "supported_languages": null}, "macro.dbt.right": {"name": "right", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/right.sql", "original_file_path": "macros/utils/right.sql", "unique_id": "macro.dbt.right", "macro_sql": "{% macro right(string_text, length_expression) -%}\n {{ return(adapter.dispatch('right', 'dbt') (string_text, length_expression)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__right"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390531, "supported_languages": null}, "macro.dbt.default__right": {"name": "default__right", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/right.sql", "original_file_path": "macros/utils/right.sql", "unique_id": "macro.dbt.default__right", "macro_sql": "{% macro default__right(string_text, length_expression) %}\n\n right(\n {{ string_text }},\n {{ length_expression }}\n )\n\n{%- endmacro -%}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3905349, "supported_languages": null}, "macro.dbt.listagg": {"name": "listagg", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/listagg.sql", "original_file_path": "macros/utils/listagg.sql", "unique_id": "macro.dbt.listagg", "macro_sql": "{% macro listagg(measure, delimiter_text=\"','\", order_by_clause=none, limit_num=none) -%}\n {{ return(adapter.dispatch('listagg', 'dbt') (measure, delimiter_text, order_by_clause, limit_num)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__listagg"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390544, "supported_languages": null}, "macro.dbt.default__listagg": {"name": "default__listagg", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/listagg.sql", "original_file_path": "macros/utils/listagg.sql", "unique_id": "macro.dbt.default__listagg", "macro_sql": "{% macro default__listagg(measure, delimiter_text, order_by_clause, limit_num) -%}\n\n {% if limit_num -%}\n array_to_string(\n array_slice(\n array_agg(\n {{ measure }}\n ){% if order_by_clause -%}\n within group ({{ order_by_clause }})\n {%- endif %}\n ,0\n ,{{ limit_num }}\n ),\n {{ delimiter_text }}\n )\n {%- else %}\n listagg(\n {{ measure }},\n {{ delimiter_text }}\n )\n {% if order_by_clause -%}\n within group ({{ order_by_clause }})\n {%- endif %}\n {%- endif %}\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390551, "supported_languages": null}, "macro.dbt.datediff": {"name": "datediff", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/datediff.sql", "original_file_path": "macros/utils/datediff.sql", "unique_id": "macro.dbt.datediff", "macro_sql": "{% macro datediff(first_date, second_date, datepart) %}\n {{ return(adapter.dispatch('datediff', 'dbt')(first_date, second_date, datepart)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__datediff"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.39056, "supported_languages": null}, "macro.dbt.default__datediff": {"name": "default__datediff", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/datediff.sql", "original_file_path": "macros/utils/datediff.sql", "unique_id": "macro.dbt.default__datediff", "macro_sql": "{% macro default__datediff(first_date, second_date, datepart) -%}\n\n datediff(\n {{ datepart }},\n {{ first_date }},\n {{ second_date }}\n )\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3905652, "supported_languages": null}, "macro.dbt.safe_cast": {"name": "safe_cast", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/safe_cast.sql", "original_file_path": "macros/utils/safe_cast.sql", "unique_id": "macro.dbt.safe_cast", "macro_sql": "{% macro safe_cast(field, type) %}\n {{ return(adapter.dispatch('safe_cast', 'dbt') (field, type)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__safe_cast"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390574, "supported_languages": null}, "macro.dbt.default__safe_cast": {"name": "default__safe_cast", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/safe_cast.sql", "original_file_path": "macros/utils/safe_cast.sql", "unique_id": "macro.dbt.default__safe_cast", "macro_sql": "{% macro default__safe_cast(field, type) %}\n {# most databases don't support this function yet\n so we just need to use cast #}\n cast({{field}} as {{type}})\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390578, "supported_languages": null}, "macro.dbt.hash": {"name": "hash", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/hash.sql", "original_file_path": "macros/utils/hash.sql", "unique_id": "macro.dbt.hash", "macro_sql": "{% macro hash(field) -%}\n {{ return(adapter.dispatch('hash', 'dbt') (field)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__hash"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3905861, "supported_languages": null}, "macro.dbt.default__hash": {"name": "default__hash", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/hash.sql", "original_file_path": "macros/utils/hash.sql", "unique_id": "macro.dbt.default__hash", "macro_sql": "{% macro default__hash(field) -%}\n md5(cast({{ field }} as {{ api.Column.translate_type('string') }}))\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390593, "supported_languages": null}, "macro.dbt.cast_bool_to_text": {"name": "cast_bool_to_text", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/cast_bool_to_text.sql", "original_file_path": "macros/utils/cast_bool_to_text.sql", "unique_id": "macro.dbt.cast_bool_to_text", "macro_sql": "{% macro cast_bool_to_text(field) %}\n {{ adapter.dispatch('cast_bool_to_text', 'dbt') (field) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__cast_bool_to_text"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390601, "supported_languages": null}, "macro.dbt.default__cast_bool_to_text": {"name": "default__cast_bool_to_text", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/cast_bool_to_text.sql", "original_file_path": "macros/utils/cast_bool_to_text.sql", "unique_id": "macro.dbt.default__cast_bool_to_text", "macro_sql": "{% macro default__cast_bool_to_text(field) %}\n cast({{ field }} as {{ api.Column.translate_type('string') }})\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3906078, "supported_languages": null}, "macro.dbt.cast": {"name": "cast", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/cast.sql", "original_file_path": "macros/utils/cast.sql", "unique_id": "macro.dbt.cast", "macro_sql": "{% macro cast(field, type) %}\n {{ return(adapter.dispatch('cast', 'dbt') (field, type)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__cast"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3906171, "supported_languages": null}, "macro.dbt.default__cast": {"name": "default__cast", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/cast.sql", "original_file_path": "macros/utils/cast.sql", "unique_id": "macro.dbt.default__cast", "macro_sql": "{% macro default__cast(field, type) %}\n cast({{field}} as {{type}})\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390623, "supported_languages": null}, "macro.dbt.any_value": {"name": "any_value", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/any_value.sql", "original_file_path": "macros/utils/any_value.sql", "unique_id": "macro.dbt.any_value", "macro_sql": "{% macro any_value(expression) -%}\n {{ return(adapter.dispatch('any_value', 'dbt') (expression)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__any_value"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.39063, "supported_languages": null}, "macro.dbt.default__any_value": {"name": "default__any_value", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/any_value.sql", "original_file_path": "macros/utils/any_value.sql", "unique_id": "macro.dbt.default__any_value", "macro_sql": "{% macro default__any_value(expression) -%}\n\n any_value({{ expression }})\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390634, "supported_languages": null}, "macro.dbt.position": {"name": "position", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/position.sql", "original_file_path": "macros/utils/position.sql", "unique_id": "macro.dbt.position", "macro_sql": "{% macro position(substring_text, string_text) -%}\n {{ return(adapter.dispatch('position', 'dbt') (substring_text, string_text)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__position"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390645, "supported_languages": null}, "macro.dbt.default__position": {"name": "default__position", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/position.sql", "original_file_path": "macros/utils/position.sql", "unique_id": "macro.dbt.default__position", "macro_sql": "{% macro default__position(substring_text, string_text) %}\n\n position(\n {{ substring_text }} in {{ string_text }}\n )\n\n{%- endmacro -%}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390649, "supported_languages": null}, "macro.dbt.string_literal": {"name": "string_literal", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/literal.sql", "original_file_path": "macros/utils/literal.sql", "unique_id": "macro.dbt.string_literal", "macro_sql": "{%- macro string_literal(value) -%}\n {{ return(adapter.dispatch('string_literal', 'dbt') (value)) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__string_literal"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390656, "supported_languages": null}, "macro.dbt.default__string_literal": {"name": "default__string_literal", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/literal.sql", "original_file_path": "macros/utils/literal.sql", "unique_id": "macro.dbt.default__string_literal", "macro_sql": "{% macro default__string_literal(value) -%}\n '{{ value }}'\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.390661, "supported_languages": null}, "macro.dbt.type_string": {"name": "type_string", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.type_string", "macro_sql": "\n\n{%- macro type_string() -%}\n {{ return(adapter.dispatch('type_string', 'dbt')()) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__type_string"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391039, "supported_languages": null}, "macro.dbt.default__type_string": {"name": "default__type_string", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.default__type_string", "macro_sql": "{% macro default__type_string() %}\n {{ return(api.Column.translate_type(\"string\")) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3910441, "supported_languages": null}, "macro.dbt.type_timestamp": {"name": "type_timestamp", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.type_timestamp", "macro_sql": "\n\n{%- macro type_timestamp() -%}\n {{ return(adapter.dispatch('type_timestamp', 'dbt')()) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__type_timestamp"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391048, "supported_languages": null}, "macro.dbt.default__type_timestamp": {"name": "default__type_timestamp", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.default__type_timestamp", "macro_sql": "{% macro default__type_timestamp() %}\n {{ return(api.Column.translate_type(\"timestamp\")) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391052, "supported_languages": null}, "macro.dbt.type_float": {"name": "type_float", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.type_float", "macro_sql": "\n\n{%- macro type_float() -%}\n {{ return(adapter.dispatch('type_float', 'dbt')()) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__type_float"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391059, "supported_languages": null}, "macro.dbt.default__type_float": {"name": "default__type_float", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.default__type_float", "macro_sql": "{% macro default__type_float() %}\n {{ return(api.Column.translate_type(\"float\")) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391063, "supported_languages": null}, "macro.dbt.type_numeric": {"name": "type_numeric", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.type_numeric", "macro_sql": "\n\n{%- macro type_numeric() -%}\n {{ return(adapter.dispatch('type_numeric', 'dbt')()) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__type_numeric"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3910668, "supported_languages": null}, "macro.dbt.default__type_numeric": {"name": "default__type_numeric", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.default__type_numeric", "macro_sql": "{% macro default__type_numeric() %}\n {{ return(api.Column.numeric_type(\"numeric\", 28, 6)) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391072, "supported_languages": null}, "macro.dbt.type_bigint": {"name": "type_bigint", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.type_bigint", "macro_sql": "\n\n{%- macro type_bigint() -%}\n {{ return(adapter.dispatch('type_bigint', 'dbt')()) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__type_bigint"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391078, "supported_languages": null}, "macro.dbt.default__type_bigint": {"name": "default__type_bigint", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.default__type_bigint", "macro_sql": "{% macro default__type_bigint() %}\n {{ return(api.Column.translate_type(\"bigint\")) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391082, "supported_languages": null}, "macro.dbt.type_int": {"name": "type_int", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.type_int", "macro_sql": "\n\n{%- macro type_int() -%}\n {{ return(adapter.dispatch('type_int', 'dbt')()) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__type_int"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391087, "supported_languages": null}, "macro.dbt.default__type_int": {"name": "default__type_int", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.default__type_int", "macro_sql": "{%- macro default__type_int() -%}\n {{ return(api.Column.translate_type(\"integer\")) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391095, "supported_languages": null}, "macro.dbt.type_boolean": {"name": "type_boolean", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.type_boolean", "macro_sql": "\n\n{%- macro type_boolean() -%}\n {{ return(adapter.dispatch('type_boolean', 'dbt')()) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__type_boolean"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391099, "supported_languages": null}, "macro.dbt.default__type_boolean": {"name": "default__type_boolean", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/data_types.sql", "original_file_path": "macros/utils/data_types.sql", "unique_id": "macro.dbt.default__type_boolean", "macro_sql": "{%- macro default__type_boolean() -%}\n {{ return(api.Column.translate_type(\"boolean\")) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391104, "supported_languages": null}, "macro.dbt.array_concat": {"name": "array_concat", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/array_concat.sql", "original_file_path": "macros/utils/array_concat.sql", "unique_id": "macro.dbt.array_concat", "macro_sql": "{% macro array_concat(array_1, array_2) -%}\n {{ return(adapter.dispatch('array_concat', 'dbt')(array_1, array_2)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__array_concat"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391114, "supported_languages": null}, "macro.dbt.default__array_concat": {"name": "default__array_concat", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/array_concat.sql", "original_file_path": "macros/utils/array_concat.sql", "unique_id": "macro.dbt.default__array_concat", "macro_sql": "{% macro default__array_concat(array_1, array_2) -%}\n array_cat({{ array_1 }}, {{ array_2 }})\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391118, "supported_languages": null}, "macro.dbt.bool_or": {"name": "bool_or", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/bool_or.sql", "original_file_path": "macros/utils/bool_or.sql", "unique_id": "macro.dbt.bool_or", "macro_sql": "{% macro bool_or(expression) -%}\n {{ return(adapter.dispatch('bool_or', 'dbt') (expression)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__bool_or"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391126, "supported_languages": null}, "macro.dbt.default__bool_or": {"name": "default__bool_or", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/bool_or.sql", "original_file_path": "macros/utils/bool_or.sql", "unique_id": "macro.dbt.default__bool_or", "macro_sql": "{% macro default__bool_or(expression) -%}\n\n bool_or({{ expression }})\n\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391129, "supported_languages": null}, "macro.dbt.last_day": {"name": "last_day", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/last_day.sql", "original_file_path": "macros/utils/last_day.sql", "unique_id": "macro.dbt.last_day", "macro_sql": "{% macro last_day(date, datepart) %}\n {{ return(adapter.dispatch('last_day', 'dbt') (date, datepart)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__last_day"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391142, "supported_languages": null}, "macro.dbt.default_last_day": {"name": "default_last_day", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/last_day.sql", "original_file_path": "macros/utils/last_day.sql", "unique_id": "macro.dbt.default_last_day", "macro_sql": "\n\n{%- macro default_last_day(date, datepart) -%}\n cast(\n {{dbt.dateadd('day', '-1',\n dbt.dateadd(datepart, '1', dbt.date_trunc(datepart, date))\n )}}\n as date)\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.dateadd", "macro.dbt.date_trunc"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391145, "supported_languages": null}, "macro.dbt.default__last_day": {"name": "default__last_day", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/last_day.sql", "original_file_path": "macros/utils/last_day.sql", "unique_id": "macro.dbt.default__last_day", "macro_sql": "{% macro default__last_day(date, datepart) -%}\n {{dbt.default_last_day(date, datepart)}}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default_last_day"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391149, "supported_languages": null}, "macro.dbt.split_part": {"name": "split_part", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/split_part.sql", "original_file_path": "macros/utils/split_part.sql", "unique_id": "macro.dbt.split_part", "macro_sql": "{% macro split_part(string_text, delimiter_text, part_number) %}\n {{ return(adapter.dispatch('split_part', 'dbt') (string_text, delimiter_text, part_number)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__split_part"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391157, "supported_languages": null}, "macro.dbt.default__split_part": {"name": "default__split_part", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/split_part.sql", "original_file_path": "macros/utils/split_part.sql", "unique_id": "macro.dbt.default__split_part", "macro_sql": "{% macro default__split_part(string_text, delimiter_text, part_number) %}\n\n split_part(\n {{ string_text }},\n {{ delimiter_text }},\n {{ part_number }}\n )\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3911612, "supported_languages": null}, "macro.dbt._split_part_negative": {"name": "_split_part_negative", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/split_part.sql", "original_file_path": "macros/utils/split_part.sql", "unique_id": "macro.dbt._split_part_negative", "macro_sql": "{% macro _split_part_negative(string_text, delimiter_text, part_number) %}\n\n split_part(\n {{ string_text }},\n {{ delimiter_text }},\n length({{ string_text }})\n - length(\n replace({{ string_text }}, {{ delimiter_text }}, '')\n ) + 2 + {{ part_number }}\n )\n\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391165, "supported_languages": null}, "macro.dbt.date_trunc": {"name": "date_trunc", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/date_trunc.sql", "original_file_path": "macros/utils/date_trunc.sql", "unique_id": "macro.dbt.date_trunc", "macro_sql": "{% macro date_trunc(datepart, date) -%}\n {{ return(adapter.dispatch('date_trunc', 'dbt') (datepart, date)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__date_trunc"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3911731, "supported_languages": null}, "macro.dbt.default__date_trunc": {"name": "default__date_trunc", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/date_trunc.sql", "original_file_path": "macros/utils/date_trunc.sql", "unique_id": "macro.dbt.default__date_trunc", "macro_sql": "{% macro default__date_trunc(datepart, date) -%}\n date_trunc('{{datepart}}', {{date}})\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391178, "supported_languages": null}, "macro.dbt.array_construct": {"name": "array_construct", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/array_construct.sql", "original_file_path": "macros/utils/array_construct.sql", "unique_id": "macro.dbt.array_construct", "macro_sql": "{% macro array_construct(inputs=[], data_type=api.Column.translate_type('integer')) -%}\n {{ return(adapter.dispatch('array_construct', 'dbt')(inputs, data_type)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__array_construct"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391187, "supported_languages": null}, "macro.dbt.default__array_construct": {"name": "default__array_construct", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/array_construct.sql", "original_file_path": "macros/utils/array_construct.sql", "unique_id": "macro.dbt.default__array_construct", "macro_sql": "{% macro default__array_construct(inputs, data_type) -%}\n {% if inputs|length > 0 %}\n array[ {{ inputs|join(' , ') }} ]\n {% else %}\n array[]::{{data_type}}[]\n {% endif %}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391191, "supported_languages": null}, "macro.dbt.array_append": {"name": "array_append", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/array_append.sql", "original_file_path": "macros/utils/array_append.sql", "unique_id": "macro.dbt.array_append", "macro_sql": "{% macro array_append(array, new_element) -%}\n {{ return(adapter.dispatch('array_append', 'dbt')(array, new_element)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__array_append"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3911989, "supported_languages": null}, "macro.dbt.default__array_append": {"name": "default__array_append", "resource_type": "macro", "package_name": "dbt", "path": "macros/utils/array_append.sql", "original_file_path": "macros/utils/array_append.sql", "unique_id": "macro.dbt.default__array_append", "macro_sql": "{% macro default__array_append(array, new_element) -%}\n array_append({{ array }}, {{ new_element }})\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391204, "supported_languages": null}, "macro.dbt.create_schema": {"name": "create_schema", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/schema.sql", "original_file_path": "macros/adapters/schema.sql", "unique_id": "macro.dbt.create_schema", "macro_sql": "{% macro create_schema(relation) -%}\n {{ adapter.dispatch('create_schema', 'dbt')(relation) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__create_schema"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391213, "supported_languages": null}, "macro.dbt.default__create_schema": {"name": "default__create_schema", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/schema.sql", "original_file_path": "macros/adapters/schema.sql", "unique_id": "macro.dbt.default__create_schema", "macro_sql": "{% macro default__create_schema(relation) -%}\n {%- call statement('create_schema') -%}\n create schema if not exists {{ relation.without_identifier() }}\n {% endcall %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391217, "supported_languages": null}, "macro.dbt.drop_schema": {"name": "drop_schema", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/schema.sql", "original_file_path": "macros/adapters/schema.sql", "unique_id": "macro.dbt.drop_schema", "macro_sql": "{% macro drop_schema(relation) -%}\n {{ adapter.dispatch('drop_schema', 'dbt')(relation) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__drop_schema"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391221, "supported_languages": null}, "macro.dbt.default__drop_schema": {"name": "default__drop_schema", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/schema.sql", "original_file_path": "macros/adapters/schema.sql", "unique_id": "macro.dbt.default__drop_schema", "macro_sql": "{% macro default__drop_schema(relation) -%}\n {%- call statement('drop_schema') -%}\n drop schema if exists {{ relation.without_identifier() }} cascade\n {% endcall %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391226, "supported_languages": null}, "macro.dbt.current_timestamp": {"name": "current_timestamp", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/timestamps.sql", "original_file_path": "macros/adapters/timestamps.sql", "unique_id": "macro.dbt.current_timestamp", "macro_sql": "{%- macro current_timestamp() -%}\n {{ adapter.dispatch('current_timestamp', 'dbt')() }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt_postgres.postgres__current_timestamp"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391235, "supported_languages": null}, "macro.dbt.default__current_timestamp": {"name": "default__current_timestamp", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/timestamps.sql", "original_file_path": "macros/adapters/timestamps.sql", "unique_id": "macro.dbt.default__current_timestamp", "macro_sql": "{% macro default__current_timestamp() -%}\n {{ exceptions.raise_not_implemented(\n 'current_timestamp macro not implemented for adapter ' + adapter.type()) }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.39124, "supported_languages": null}, "macro.dbt.snapshot_get_time": {"name": "snapshot_get_time", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/timestamps.sql", "original_file_path": "macros/adapters/timestamps.sql", "unique_id": "macro.dbt.snapshot_get_time", "macro_sql": "\n\n{%- macro snapshot_get_time() -%}\n {{ adapter.dispatch('snapshot_get_time', 'dbt')() }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt_postgres.postgres__snapshot_get_time"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391243, "supported_languages": null}, "macro.dbt.default__snapshot_get_time": {"name": "default__snapshot_get_time", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/timestamps.sql", "original_file_path": "macros/adapters/timestamps.sql", "unique_id": "macro.dbt.default__snapshot_get_time", "macro_sql": "{% macro default__snapshot_get_time() %}\n {{ current_timestamp() }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.current_timestamp"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391247, "supported_languages": null}, "macro.dbt.get_snapshot_get_time_data_type": {"name": "get_snapshot_get_time_data_type", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/timestamps.sql", "original_file_path": "macros/adapters/timestamps.sql", "unique_id": "macro.dbt.get_snapshot_get_time_data_type", "macro_sql": "{% macro get_snapshot_get_time_data_type() %}\n {% set snapshot_time = adapter.dispatch('snapshot_get_time', 'dbt')() %}\n {% set time_data_type_sql = 'select ' ~ snapshot_time ~ ' as dbt_snapshot_time' %}\n {% set snapshot_time_column_schema = get_column_schema_from_query(time_data_type_sql) %}\n {% set time_data_type = snapshot_time_column_schema[0].dtype %}\n {{ return(time_data_type or none) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.snapshot_get_time", "macro.dbt_postgres.postgres__snapshot_get_time", "macro.dbt.get_column_schema_from_query"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3912508, "supported_languages": null}, "macro.dbt.current_timestamp_backcompat": {"name": "current_timestamp_backcompat", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/timestamps.sql", "original_file_path": "macros/adapters/timestamps.sql", "unique_id": "macro.dbt.current_timestamp_backcompat", "macro_sql": "{% macro current_timestamp_backcompat() %}\n {{ return(adapter.dispatch('current_timestamp_backcompat', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__current_timestamp_backcompat"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3912551, "supported_languages": null}, "macro.dbt.default__current_timestamp_backcompat": {"name": "default__current_timestamp_backcompat", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/timestamps.sql", "original_file_path": "macros/adapters/timestamps.sql", "unique_id": "macro.dbt.default__current_timestamp_backcompat", "macro_sql": "{% macro default__current_timestamp_backcompat() %}\n current_timestamp::timestamp\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.39126, "supported_languages": null}, "macro.dbt.current_timestamp_in_utc_backcompat": {"name": "current_timestamp_in_utc_backcompat", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/timestamps.sql", "original_file_path": "macros/adapters/timestamps.sql", "unique_id": "macro.dbt.current_timestamp_in_utc_backcompat", "macro_sql": "{% macro current_timestamp_in_utc_backcompat() %}\n {{ return(adapter.dispatch('current_timestamp_in_utc_backcompat', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__current_timestamp_in_utc_backcompat"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391264, "supported_languages": null}, "macro.dbt.default__current_timestamp_in_utc_backcompat": {"name": "default__current_timestamp_in_utc_backcompat", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/timestamps.sql", "original_file_path": "macros/adapters/timestamps.sql", "unique_id": "macro.dbt.default__current_timestamp_in_utc_backcompat", "macro_sql": "{% macro default__current_timestamp_in_utc_backcompat() %}\n {{ return(adapter.dispatch('current_timestamp_backcompat', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.current_timestamp_backcompat", "macro.dbt_postgres.postgres__current_timestamp_backcompat"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391268, "supported_languages": null}, "macro.dbt.get_create_index_sql": {"name": "get_create_index_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/indexes.sql", "original_file_path": "macros/adapters/indexes.sql", "unique_id": "macro.dbt.get_create_index_sql", "macro_sql": "{% macro get_create_index_sql(relation, index_dict) -%}\n {{ return(adapter.dispatch('get_create_index_sql', 'dbt')(relation, index_dict)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_create_index_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3912761, "supported_languages": null}, "macro.dbt.default__get_create_index_sql": {"name": "default__get_create_index_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/indexes.sql", "original_file_path": "macros/adapters/indexes.sql", "unique_id": "macro.dbt.default__get_create_index_sql", "macro_sql": "{% macro default__get_create_index_sql(relation, index_dict) -%}\n {% do return(None) %}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3912802, "supported_languages": null}, "macro.dbt.create_indexes": {"name": "create_indexes", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/indexes.sql", "original_file_path": "macros/adapters/indexes.sql", "unique_id": "macro.dbt.create_indexes", "macro_sql": "{% macro create_indexes(relation) -%}\n {{ adapter.dispatch('create_indexes', 'dbt')(relation) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.default__create_indexes"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391285, "supported_languages": null}, "macro.dbt.default__create_indexes": {"name": "default__create_indexes", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/indexes.sql", "original_file_path": "macros/adapters/indexes.sql", "unique_id": "macro.dbt.default__create_indexes", "macro_sql": "{% macro default__create_indexes(relation) -%}\n {%- set _indexes = config.get('indexes', default=[]) -%}\n\n {% for _index_dict in _indexes %}\n {% set create_index_sql = get_create_index_sql(relation, _index_dict) %}\n {% if create_index_sql %}\n {% do run_query(create_index_sql) %}\n {% endif %}\n {% endfor %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_create_index_sql", "macro.dbt.run_query"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391289, "supported_languages": null}, "macro.dbt.get_drop_index_sql": {"name": "get_drop_index_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/indexes.sql", "original_file_path": "macros/adapters/indexes.sql", "unique_id": "macro.dbt.get_drop_index_sql", "macro_sql": "{% macro get_drop_index_sql(relation, index_name) -%}\n {{ adapter.dispatch('get_drop_index_sql', 'dbt')(relation, index_name) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_drop_index_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3912928, "supported_languages": null}, "macro.dbt.default__get_drop_index_sql": {"name": "default__get_drop_index_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/indexes.sql", "original_file_path": "macros/adapters/indexes.sql", "unique_id": "macro.dbt.default__get_drop_index_sql", "macro_sql": "{% macro default__get_drop_index_sql(relation, index_name) -%}\n {{ exceptions.raise_compiler_error(\"`get_drop_index_sql has not been implemented for this adapter.\") }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391298, "supported_languages": null}, "macro.dbt.get_show_indexes_sql": {"name": "get_show_indexes_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/indexes.sql", "original_file_path": "macros/adapters/indexes.sql", "unique_id": "macro.dbt.get_show_indexes_sql", "macro_sql": "{% macro get_show_indexes_sql(relation) -%}\n {{ adapter.dispatch('get_show_indexes_sql', 'dbt')(relation) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_show_indexes_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391304, "supported_languages": null}, "macro.dbt.default__get_show_indexes_sql": {"name": "default__get_show_indexes_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/indexes.sql", "original_file_path": "macros/adapters/indexes.sql", "unique_id": "macro.dbt.default__get_show_indexes_sql", "macro_sql": "{% macro default__get_show_indexes_sql(relation) -%}\n {{ exceptions.raise_compiler_error(\"`get_show_indexes_sql has not been implemented for this adapter.\") }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391308, "supported_languages": null}, "macro.dbt.make_intermediate_relation": {"name": "make_intermediate_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.make_intermediate_relation", "macro_sql": "{% macro make_intermediate_relation(base_relation, suffix='__dbt_tmp') %}\n {{ return(adapter.dispatch('make_intermediate_relation', 'dbt')(base_relation, suffix)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__make_intermediate_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391318, "supported_languages": null}, "macro.dbt.default__make_intermediate_relation": {"name": "default__make_intermediate_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.default__make_intermediate_relation", "macro_sql": "{% macro default__make_intermediate_relation(base_relation, suffix) %}\n {{ return(default__make_temp_relation(base_relation, suffix)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__make_temp_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3913221, "supported_languages": null}, "macro.dbt.make_temp_relation": {"name": "make_temp_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.make_temp_relation", "macro_sql": "{% macro make_temp_relation(base_relation, suffix='__dbt_tmp') %}\n {{ return(adapter.dispatch('make_temp_relation', 'dbt')(base_relation, suffix)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__make_temp_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391327, "supported_languages": null}, "macro.dbt.default__make_temp_relation": {"name": "default__make_temp_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.default__make_temp_relation", "macro_sql": "{% macro default__make_temp_relation(base_relation, suffix) %}\n {%- set temp_identifier = base_relation.identifier ~ suffix -%}\n {%- set temp_relation = base_relation.incorporate(\n path={\"identifier\": temp_identifier}) -%}\n\n {{ return(temp_relation) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391331, "supported_languages": null}, "macro.dbt.make_backup_relation": {"name": "make_backup_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.make_backup_relation", "macro_sql": "{% macro make_backup_relation(base_relation, backup_relation_type, suffix='__dbt_backup') %}\n {{ return(adapter.dispatch('make_backup_relation', 'dbt')(base_relation, backup_relation_type, suffix)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__make_backup_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391344, "supported_languages": null}, "macro.dbt.default__make_backup_relation": {"name": "default__make_backup_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.default__make_backup_relation", "macro_sql": "{% macro default__make_backup_relation(base_relation, backup_relation_type, suffix) %}\n {%- set backup_identifier = base_relation.identifier ~ suffix -%}\n {%- set backup_relation = base_relation.incorporate(\n path={\"identifier\": backup_identifier},\n type=backup_relation_type\n ) -%}\n {{ return(backup_relation) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391349, "supported_languages": null}, "macro.dbt.truncate_relation": {"name": "truncate_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.truncate_relation", "macro_sql": "{% macro truncate_relation(relation) -%}\n {{ return(adapter.dispatch('truncate_relation', 'dbt')(relation)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__truncate_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3913522, "supported_languages": null}, "macro.dbt.default__truncate_relation": {"name": "default__truncate_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.default__truncate_relation", "macro_sql": "{% macro default__truncate_relation(relation) -%}\n {% call statement('truncate_relation') -%}\n truncate table {{ relation.render() }}\n {%- endcall %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391356, "supported_languages": null}, "macro.dbt.get_or_create_relation": {"name": "get_or_create_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.get_or_create_relation", "macro_sql": "{% macro get_or_create_relation(database, schema, identifier, type) -%}\n {{ return(adapter.dispatch('get_or_create_relation', 'dbt')(database, schema, identifier, type)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_or_create_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.39136, "supported_languages": null}, "macro.dbt.default__get_or_create_relation": {"name": "default__get_or_create_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.default__get_or_create_relation", "macro_sql": "{% macro default__get_or_create_relation(database, schema, identifier, type) %}\n {%- set target_relation = adapter.get_relation(database=database, schema=schema, identifier=identifier) %}\n\n {% if target_relation %}\n {% do return([true, target_relation]) %}\n {% endif %}\n\n {%- set new_relation = api.Relation.create(\n database=database,\n schema=schema,\n identifier=identifier,\n type=type\n ) -%}\n {% do return([false, new_relation]) %}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391368, "supported_languages": null}, "macro.dbt.load_cached_relation": {"name": "load_cached_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.load_cached_relation", "macro_sql": "{% macro load_cached_relation(relation) %}\n {% do return(adapter.get_relation(\n database=relation.database,\n schema=relation.schema,\n identifier=relation.identifier\n )) -%}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391374, "supported_languages": null}, "macro.dbt.load_relation": {"name": "load_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/relation.sql", "original_file_path": "macros/adapters/relation.sql", "unique_id": "macro.dbt.load_relation", "macro_sql": "{% macro load_relation(relation) %}\n {{ return(load_cached_relation(relation)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.load_cached_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391377, "supported_languages": null}, "macro.dbt.collect_freshness": {"name": "collect_freshness", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/freshness.sql", "original_file_path": "macros/adapters/freshness.sql", "unique_id": "macro.dbt.collect_freshness", "macro_sql": "{% macro collect_freshness(source, loaded_at_field, filter) %}\n {{ return(adapter.dispatch('collect_freshness', 'dbt')(source, loaded_at_field, filter))}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__collect_freshness"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391385, "supported_languages": null}, "macro.dbt.default__collect_freshness": {"name": "default__collect_freshness", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/freshness.sql", "original_file_path": "macros/adapters/freshness.sql", "unique_id": "macro.dbt.default__collect_freshness", "macro_sql": "{% macro default__collect_freshness(source, loaded_at_field, filter) %}\n {% call statement('collect_freshness', fetch_result=True, auto_begin=False) -%}\n select\n max({{ loaded_at_field }}) as max_loaded_at,\n {{ current_timestamp() }} as snapshotted_at\n from {{ source }}\n {% if filter %}\n where {{ filter }}\n {% endif %}\n {% endcall %}\n {{ return(load_result('collect_freshness')) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement", "macro.dbt.current_timestamp"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3913898, "supported_languages": null}, "macro.dbt.validate_sql": {"name": "validate_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/validate_sql.sql", "original_file_path": "macros/adapters/validate_sql.sql", "unique_id": "macro.dbt.validate_sql", "macro_sql": "{% macro validate_sql(sql) -%}\n {{ return(adapter.dispatch('validate_sql', 'dbt')(sql)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__validate_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391399, "supported_languages": null}, "macro.dbt.default__validate_sql": {"name": "default__validate_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/validate_sql.sql", "original_file_path": "macros/adapters/validate_sql.sql", "unique_id": "macro.dbt.default__validate_sql", "macro_sql": "{% macro default__validate_sql(sql) -%}\n {% call statement('validate_sql') -%}\n explain {{ sql }}\n {% endcall %}\n {{ return(load_result('validate_sql')) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391404, "supported_languages": null}, "macro.dbt.copy_grants": {"name": "copy_grants", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.copy_grants", "macro_sql": "{% macro copy_grants() %}\n {{ return(adapter.dispatch('copy_grants', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__copy_grants"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3914182, "supported_languages": null}, "macro.dbt.default__copy_grants": {"name": "default__copy_grants", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.default__copy_grants", "macro_sql": "{% macro default__copy_grants() %}\n {{ return(True) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3914251, "supported_languages": null}, "macro.dbt.support_multiple_grantees_per_dcl_statement": {"name": "support_multiple_grantees_per_dcl_statement", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.support_multiple_grantees_per_dcl_statement", "macro_sql": "{% macro support_multiple_grantees_per_dcl_statement() %}\n {{ return(adapter.dispatch('support_multiple_grantees_per_dcl_statement', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__support_multiple_grantees_per_dcl_statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.39143, "supported_languages": null}, "macro.dbt.default__support_multiple_grantees_per_dcl_statement": {"name": "default__support_multiple_grantees_per_dcl_statement", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.default__support_multiple_grantees_per_dcl_statement", "macro_sql": "\n\n{%- macro default__support_multiple_grantees_per_dcl_statement() -%}\n {{ return(True) }}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391449, "supported_languages": null}, "macro.dbt.should_revoke": {"name": "should_revoke", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.should_revoke", "macro_sql": "{% macro should_revoke(existing_relation, full_refresh_mode=True) %}\n\n {% if not existing_relation %}\n {#-- The table doesn't already exist, so no grants to copy over --#}\n {{ return(False) }}\n {% elif full_refresh_mode %}\n {#-- The object is being REPLACED -- whether grants are copied over depends on the value of user config --#}\n {{ return(copy_grants()) }}\n {% else %}\n {#-- The table is being merged/upserted/inserted -- grants will be carried over --#}\n {{ return(True) }}\n {% endif %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.copy_grants"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391453, "supported_languages": null}, "macro.dbt.get_show_grant_sql": {"name": "get_show_grant_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.get_show_grant_sql", "macro_sql": "{% macro get_show_grant_sql(relation) %}\n {{ return(adapter.dispatch(\"get_show_grant_sql\", \"dbt\")(relation)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_show_grant_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391458, "supported_languages": null}, "macro.dbt.default__get_show_grant_sql": {"name": "default__get_show_grant_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.default__get_show_grant_sql", "macro_sql": "{% macro default__get_show_grant_sql(relation) %}\n show grants on {{ relation.render() }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3914628, "supported_languages": null}, "macro.dbt.get_grant_sql": {"name": "get_grant_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.get_grant_sql", "macro_sql": "{% macro get_grant_sql(relation, privilege, grantees) %}\n {{ return(adapter.dispatch('get_grant_sql', 'dbt')(relation, privilege, grantees)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_grant_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391469, "supported_languages": null}, "macro.dbt.default__get_grant_sql": {"name": "default__get_grant_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.default__get_grant_sql", "macro_sql": "\n\n{%- macro default__get_grant_sql(relation, privilege, grantees) -%}\n grant {{ privilege }} on {{ relation.render() }} to {{ grantees | join(', ') }}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3914728, "supported_languages": null}, "macro.dbt.get_revoke_sql": {"name": "get_revoke_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.get_revoke_sql", "macro_sql": "{% macro get_revoke_sql(relation, privilege, grantees) %}\n {{ return(adapter.dispatch('get_revoke_sql', 'dbt')(relation, privilege, grantees)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_revoke_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391477, "supported_languages": null}, "macro.dbt.default__get_revoke_sql": {"name": "default__get_revoke_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.default__get_revoke_sql", "macro_sql": "\n\n{%- macro default__get_revoke_sql(relation, privilege, grantees) -%}\n revoke {{ privilege }} on {{ relation.render() }} from {{ grantees | join(', ') }}\n{%- endmacro -%}\n\n\n", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.39148, "supported_languages": null}, "macro.dbt.get_dcl_statement_list": {"name": "get_dcl_statement_list", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.get_dcl_statement_list", "macro_sql": "{% macro get_dcl_statement_list(relation, grant_config, get_dcl_macro) %}\n {{ return(adapter.dispatch('get_dcl_statement_list', 'dbt')(relation, grant_config, get_dcl_macro)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_dcl_statement_list"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391484, "supported_languages": null}, "macro.dbt.default__get_dcl_statement_list": {"name": "default__get_dcl_statement_list", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.default__get_dcl_statement_list", "macro_sql": "\n\n{%- macro default__get_dcl_statement_list(relation, grant_config, get_dcl_macro) -%}\n {#\n -- Unpack grant_config into specific privileges and the set of users who need them granted/revoked.\n -- Depending on whether this database supports multiple grantees per statement, pass in the list of\n -- all grantees per privilege, or (if not) template one statement per privilege-grantee pair.\n -- `get_dcl_macro` will be either `get_grant_sql` or `get_revoke_sql`\n #}\n {%- set dcl_statements = [] -%}\n {%- for privilege, grantees in grant_config.items() %}\n {%- if support_multiple_grantees_per_dcl_statement() and grantees -%}\n {%- set dcl = get_dcl_macro(relation, privilege, grantees) -%}\n {%- do dcl_statements.append(dcl) -%}\n {%- else -%}\n {%- for grantee in grantees -%}\n {% set dcl = get_dcl_macro(relation, privilege, [grantee]) %}\n {%- do dcl_statements.append(dcl) -%}\n {% endfor -%}\n {%- endif -%}\n {%- endfor -%}\n {{ return(dcl_statements) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt.support_multiple_grantees_per_dcl_statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391493, "supported_languages": null}, "macro.dbt.call_dcl_statements": {"name": "call_dcl_statements", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.call_dcl_statements", "macro_sql": "{% macro call_dcl_statements(dcl_statement_list) %}\n {{ return(adapter.dispatch(\"call_dcl_statements\", \"dbt\")(dcl_statement_list)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__call_dcl_statements"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391499, "supported_languages": null}, "macro.dbt.default__call_dcl_statements": {"name": "default__call_dcl_statements", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.default__call_dcl_statements", "macro_sql": "{% macro default__call_dcl_statements(dcl_statement_list) %}\n {#\n -- By default, supply all grant + revoke statements in a single semicolon-separated block,\n -- so that they're all processed together.\n\n -- Some databases do not support this. Those adapters will need to override this macro\n -- to run each statement individually.\n #}\n {% call statement('grants') %}\n {% for dcl_statement in dcl_statement_list %}\n {{ dcl_statement }};\n {% endfor %}\n {% endcall %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391503, "supported_languages": null}, "macro.dbt.apply_grants": {"name": "apply_grants", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.apply_grants", "macro_sql": "{% macro apply_grants(relation, grant_config, should_revoke) %}\n {{ return(adapter.dispatch(\"apply_grants\", \"dbt\")(relation, grant_config, should_revoke)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__apply_grants"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391506, "supported_languages": null}, "macro.dbt.default__apply_grants": {"name": "default__apply_grants", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/apply_grants.sql", "original_file_path": "macros/adapters/apply_grants.sql", "unique_id": "macro.dbt.default__apply_grants", "macro_sql": "{% macro default__apply_grants(relation, grant_config, should_revoke=True) %}\n {#-- If grant_config is {} or None, this is a no-op --#}\n {% if grant_config %}\n {% if should_revoke %}\n {#-- We think previous grants may have carried over --#}\n {#-- Show current grants and calculate diffs --#}\n {% set current_grants_table = run_query(get_show_grant_sql(relation)) %}\n {% set current_grants_dict = adapter.standardize_grants_dict(current_grants_table) %}\n {% set needs_granting = diff_of_two_dicts(grant_config, current_grants_dict) %}\n {% set needs_revoking = diff_of_two_dicts(current_grants_dict, grant_config) %}\n {% if not (needs_granting or needs_revoking) %}\n {{ log('On ' ~ relation.render() ~': All grants are in place, no revocation or granting needed.')}}\n {% endif %}\n {% else %}\n {#-- We don't think there's any chance of previous grants having carried over. --#}\n {#-- Jump straight to granting what the user has configured. --#}\n {% set needs_revoking = {} %}\n {% set needs_granting = grant_config %}\n {% endif %}\n {% if needs_granting or needs_revoking %}\n {% set revoke_statement_list = get_dcl_statement_list(relation, needs_revoking, get_revoke_sql) %}\n {% set grant_statement_list = get_dcl_statement_list(relation, needs_granting, get_grant_sql) %}\n {% set dcl_statement_list = revoke_statement_list + grant_statement_list %}\n {% if dcl_statement_list %}\n {{ call_dcl_statements(dcl_statement_list) }}\n {% endif %}\n {% endif %}\n {% endif %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.run_query", "macro.dbt.get_show_grant_sql", "macro.dbt.get_dcl_statement_list", "macro.dbt.call_dcl_statements"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.39151, "supported_languages": null}, "macro.dbt.get_show_sql": {"name": "get_show_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/show.sql", "original_file_path": "macros/adapters/show.sql", "unique_id": "macro.dbt.get_show_sql", "macro_sql": "{% macro get_show_sql(compiled_code, sql_header, limit) -%}\n {%- if sql_header is not none -%}\n {{ sql_header }}\n {%- endif %}\n {{ get_limit_subquery_sql(compiled_code, limit) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_limit_subquery_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391521, "supported_languages": null}, "macro.dbt.get_limit_subquery_sql": {"name": "get_limit_subquery_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/show.sql", "original_file_path": "macros/adapters/show.sql", "unique_id": "macro.dbt.get_limit_subquery_sql", "macro_sql": "\n{%- macro get_limit_subquery_sql(sql, limit) -%}\n {{ adapter.dispatch('get_limit_sql', 'dbt')(sql, limit) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__get_limit_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391525, "supported_languages": null}, "macro.dbt.default__get_limit_sql": {"name": "default__get_limit_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/show.sql", "original_file_path": "macros/adapters/show.sql", "unique_id": "macro.dbt.default__get_limit_sql", "macro_sql": "{% macro default__get_limit_sql(sql, limit) %}\n {{ compiled_code }}\n {% if limit is not none %}\n limit {{ limit }}\n {%- endif -%}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3915331, "supported_languages": null}, "macro.dbt.alter_column_comment": {"name": "alter_column_comment", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/persist_docs.sql", "original_file_path": "macros/adapters/persist_docs.sql", "unique_id": "macro.dbt.alter_column_comment", "macro_sql": "{% macro alter_column_comment(relation, column_dict) -%}\n {{ return(adapter.dispatch('alter_column_comment', 'dbt')(relation, column_dict)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__alter_column_comment"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391541, "supported_languages": null}, "macro.dbt.default__alter_column_comment": {"name": "default__alter_column_comment", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/persist_docs.sql", "original_file_path": "macros/adapters/persist_docs.sql", "unique_id": "macro.dbt.default__alter_column_comment", "macro_sql": "{% macro default__alter_column_comment(relation, column_dict) -%}\n {{ exceptions.raise_not_implemented(\n 'alter_column_comment macro not implemented for adapter '+adapter.type()) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391545, "supported_languages": null}, "macro.dbt.alter_relation_comment": {"name": "alter_relation_comment", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/persist_docs.sql", "original_file_path": "macros/adapters/persist_docs.sql", "unique_id": "macro.dbt.alter_relation_comment", "macro_sql": "{% macro alter_relation_comment(relation, relation_comment) -%}\n {{ return(adapter.dispatch('alter_relation_comment', 'dbt')(relation, relation_comment)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__alter_relation_comment"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391548, "supported_languages": null}, "macro.dbt.default__alter_relation_comment": {"name": "default__alter_relation_comment", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/persist_docs.sql", "original_file_path": "macros/adapters/persist_docs.sql", "unique_id": "macro.dbt.default__alter_relation_comment", "macro_sql": "{% macro default__alter_relation_comment(relation, relation_comment) -%}\n {{ exceptions.raise_not_implemented(\n 'alter_relation_comment macro not implemented for adapter '+adapter.type()) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391554, "supported_languages": null}, "macro.dbt.persist_docs": {"name": "persist_docs", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/persist_docs.sql", "original_file_path": "macros/adapters/persist_docs.sql", "unique_id": "macro.dbt.persist_docs", "macro_sql": "{% macro persist_docs(relation, model, for_relation=true, for_columns=true) -%}\n {{ return(adapter.dispatch('persist_docs', 'dbt')(relation, model, for_relation, for_columns)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__persist_docs"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391558, "supported_languages": null}, "macro.dbt.default__persist_docs": {"name": "default__persist_docs", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/persist_docs.sql", "original_file_path": "macros/adapters/persist_docs.sql", "unique_id": "macro.dbt.default__persist_docs", "macro_sql": "{% macro default__persist_docs(relation, model, for_relation, for_columns) -%}\n {% if for_relation and config.persist_relation_docs() and model.description %}\n {% do run_query(alter_relation_comment(relation, model.description)) %}\n {% endif %}\n\n {% if for_columns and config.persist_column_docs() and model.columns %}\n {% do run_query(alter_column_comment(relation, model.columns)) %}\n {% endif %}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.run_query", "macro.dbt.alter_relation_comment", "macro.dbt.alter_column_comment"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391562, "supported_languages": null}, "macro.dbt.get_catalog_relations": {"name": "get_catalog_relations", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.get_catalog_relations", "macro_sql": "{% macro get_catalog_relations(information_schema, relations) -%}\n {{ return(adapter.dispatch('get_catalog_relations', 'dbt')(information_schema, relations)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_catalog_relations"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391574, "supported_languages": null}, "macro.dbt.default__get_catalog_relations": {"name": "default__get_catalog_relations", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.default__get_catalog_relations", "macro_sql": "{% macro default__get_catalog_relations(information_schema, relations) -%}\n {% set typename = adapter.type() %}\n {% set msg -%}\n get_catalog_relations not implemented for {{ typename }}\n {%- endset %}\n\n {{ exceptions.raise_compiler_error(msg) }}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3915782, "supported_languages": null}, "macro.dbt.get_catalog": {"name": "get_catalog", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.get_catalog", "macro_sql": "{% macro get_catalog(information_schema, schemas) -%}\n {{ return(adapter.dispatch('get_catalog', 'dbt')(information_schema, schemas)) }}\n{%- endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_catalog"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391583, "supported_languages": null}, "macro.dbt.default__get_catalog": {"name": "default__get_catalog", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.default__get_catalog", "macro_sql": "{% macro default__get_catalog(information_schema, schemas) -%}\n\n {% set typename = adapter.type() %}\n {% set msg -%}\n get_catalog not implemented for {{ typename }}\n {%- endset %}\n\n {{ exceptions.raise_compiler_error(msg) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3915882, "supported_languages": null}, "macro.dbt.information_schema_name": {"name": "information_schema_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.information_schema_name", "macro_sql": "{% macro information_schema_name(database) %}\n {{ return(adapter.dispatch('information_schema_name', 'dbt')(database)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__information_schema_name"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391592, "supported_languages": null}, "macro.dbt.default__information_schema_name": {"name": "default__information_schema_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.default__information_schema_name", "macro_sql": "{% macro default__information_schema_name(database) -%}\n {%- if database -%}\n {{ database }}.INFORMATION_SCHEMA\n {%- else -%}\n INFORMATION_SCHEMA\n {%- endif -%}\n{%- endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391596, "supported_languages": null}, "macro.dbt.list_schemas": {"name": "list_schemas", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.list_schemas", "macro_sql": "{% macro list_schemas(database) -%}\n {{ return(adapter.dispatch('list_schemas', 'dbt')(database)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__list_schemas"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391604, "supported_languages": null}, "macro.dbt.default__list_schemas": {"name": "default__list_schemas", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.default__list_schemas", "macro_sql": "{% macro default__list_schemas(database) -%}\n {% set sql %}\n select distinct schema_name\n from {{ information_schema_name(database) }}.SCHEMATA\n where catalog_name ilike '{{ database }}'\n {% endset %}\n {{ return(run_query(sql)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.information_schema_name", "macro.dbt.run_query"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391608, "supported_languages": null}, "macro.dbt.check_schema_exists": {"name": "check_schema_exists", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.check_schema_exists", "macro_sql": "{% macro check_schema_exists(information_schema, schema) -%}\n {{ return(adapter.dispatch('check_schema_exists', 'dbt')(information_schema, schema)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__check_schema_exists"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3916109, "supported_languages": null}, "macro.dbt.default__check_schema_exists": {"name": "default__check_schema_exists", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.default__check_schema_exists", "macro_sql": "{% macro default__check_schema_exists(information_schema, schema) -%}\n {% set sql -%}\n select count(*)\n from {{ information_schema.replace(information_schema_view='SCHEMATA') }}\n where catalog_name='{{ information_schema.database }}'\n and schema_name='{{ schema }}'\n {%- endset %}\n {{ return(run_query(sql)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.replace", "macro.dbt.run_query"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391617, "supported_languages": null}, "macro.dbt.list_relations_without_caching": {"name": "list_relations_without_caching", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.list_relations_without_caching", "macro_sql": "{% macro list_relations_without_caching(schema_relation) %}\n {{ return(adapter.dispatch('list_relations_without_caching', 'dbt')(schema_relation)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__list_relations_without_caching"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391621, "supported_languages": null}, "macro.dbt.default__list_relations_without_caching": {"name": "default__list_relations_without_caching", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.default__list_relations_without_caching", "macro_sql": "{% macro default__list_relations_without_caching(schema_relation) %}\n {{ exceptions.raise_not_implemented(\n 'list_relations_without_caching macro not implemented for adapter '+adapter.type()) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391625, "supported_languages": null}, "macro.dbt.get_catalog_for_single_relation": {"name": "get_catalog_for_single_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.get_catalog_for_single_relation", "macro_sql": "{% macro get_catalog_for_single_relation(relation) %}\n {{ return(adapter.dispatch('get_catalog_for_single_relation', 'dbt')(relation)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_catalog_for_single_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391628, "supported_languages": null}, "macro.dbt.default__get_catalog_for_single_relation": {"name": "default__get_catalog_for_single_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.default__get_catalog_for_single_relation", "macro_sql": "{% macro default__get_catalog_for_single_relation(relation) %}\n {{ exceptions.raise_not_implemented(\n 'get_catalog_for_single_relation macro not implemented for adapter '+adapter.type()) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3916318, "supported_languages": null}, "macro.dbt.get_relations": {"name": "get_relations", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.get_relations", "macro_sql": "{% macro get_relations() %}\n {{ return(adapter.dispatch('get_relations', 'dbt')()) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_relations"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391636, "supported_languages": null}, "macro.dbt.default__get_relations": {"name": "default__get_relations", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.default__get_relations", "macro_sql": "{% macro default__get_relations() %}\n {{ exceptions.raise_not_implemented(\n 'get_relations macro not implemented for adapter '+adapter.type()) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3916411, "supported_languages": null}, "macro.dbt.get_relation_last_modified": {"name": "get_relation_last_modified", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.get_relation_last_modified", "macro_sql": "{% macro get_relation_last_modified(information_schema, relations) %}\n {{ return(adapter.dispatch('get_relation_last_modified', 'dbt')(information_schema, relations)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_relation_last_modified"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391648, "supported_languages": null}, "macro.dbt.default__get_relation_last_modified": {"name": "default__get_relation_last_modified", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/metadata.sql", "original_file_path": "macros/adapters/metadata.sql", "unique_id": "macro.dbt.default__get_relation_last_modified", "macro_sql": "{% macro default__get_relation_last_modified(information_schema, relations) %}\n {{ exceptions.raise_not_implemented(\n 'get_relation_last_modified macro not implemented for adapter ' + adapter.type()) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391656, "supported_languages": null}, "macro.dbt.get_columns_in_relation": {"name": "get_columns_in_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.get_columns_in_relation", "macro_sql": "{% macro get_columns_in_relation(relation) -%}\n {{ return(adapter.dispatch('get_columns_in_relation', 'dbt')(relation)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt_postgres.postgres__get_columns_in_relation"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391667, "supported_languages": null}, "macro.dbt.default__get_columns_in_relation": {"name": "default__get_columns_in_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.default__get_columns_in_relation", "macro_sql": "{% macro default__get_columns_in_relation(relation) -%}\n {{ exceptions.raise_not_implemented(\n 'get_columns_in_relation macro not implemented for adapter '+adapter.type()) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3916721, "supported_languages": null}, "macro.dbt.sql_convert_columns_in_relation": {"name": "sql_convert_columns_in_relation", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.sql_convert_columns_in_relation", "macro_sql": "{% macro sql_convert_columns_in_relation(table) -%}\n {% set columns = [] %}\n {% for row in table %}\n {% do columns.append(api.Column(*row)) %}\n {% endfor %}\n {{ return(columns) }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391675, "supported_languages": null}, "macro.dbt.get_empty_subquery_sql": {"name": "get_empty_subquery_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.get_empty_subquery_sql", "macro_sql": "{% macro get_empty_subquery_sql(select_sql, select_sql_header=none) -%}\n {{ return(adapter.dispatch('get_empty_subquery_sql', 'dbt')(select_sql, select_sql_header)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_empty_subquery_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391679, "supported_languages": null}, "macro.dbt.default__get_empty_subquery_sql": {"name": "default__get_empty_subquery_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.default__get_empty_subquery_sql", "macro_sql": "{% macro default__get_empty_subquery_sql(select_sql, select_sql_header=none) %}\n {%- if select_sql_header is not none -%}\n {{ select_sql_header }}\n {%- endif -%}\n select * from (\n {{ select_sql }}\n ) as __dbt_sbq\n where false\n limit 0\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3916879, "supported_languages": null}, "macro.dbt.get_empty_schema_sql": {"name": "get_empty_schema_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.get_empty_schema_sql", "macro_sql": "{% macro get_empty_schema_sql(columns) -%}\n {{ return(adapter.dispatch('get_empty_schema_sql', 'dbt')(columns)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_empty_schema_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391695, "supported_languages": null}, "macro.dbt.default__get_empty_schema_sql": {"name": "default__get_empty_schema_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.default__get_empty_schema_sql", "macro_sql": "{% macro default__get_empty_schema_sql(columns) %}\n {%- set col_err = [] -%}\n {%- set col_naked_numeric = [] -%}\n select\n {% for i in columns %}\n {%- set col = columns[i] -%}\n {%- if col['data_type'] is not defined -%}\n {%- do col_err.append(col['name']) -%}\n {#-- If this column's type is just 'numeric' then it is missing precision/scale, raise a warning --#}\n {%- elif col['data_type'].strip().lower() in ('numeric', 'decimal', 'number') -%}\n {%- do col_naked_numeric.append(col['name']) -%}\n {%- endif -%}\n {% set col_name = adapter.quote(col['name']) if col.get('quote') else col['name'] %}\n {{ cast('null', col['data_type']) }} as {{ col_name }}{{ \", \" if not loop.last }}\n {%- endfor -%}\n {%- if (col_err | length) > 0 -%}\n {{ exceptions.column_type_missing(column_names=col_err) }}\n {%- elif (col_naked_numeric | length) > 0 -%}\n {{ exceptions.warn(\"Detected columns with numeric type and unspecified precision/scale, this can lead to unintended rounding: \" ~ col_naked_numeric ~ \"`\") }}\n {%- endif -%}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.cast"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391704, "supported_languages": null}, "macro.dbt.get_column_schema_from_query": {"name": "get_column_schema_from_query", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.get_column_schema_from_query", "macro_sql": "{% macro get_column_schema_from_query(select_sql, select_sql_header=none) -%}\n {% set columns = [] %}\n {# -- Using an 'empty subquery' here to get the same schema as the given select_sql statement, without necessitating a data scan.#}\n {% set sql = get_empty_subquery_sql(select_sql, select_sql_header) %}\n {% set column_schema = adapter.get_column_schema_from_query(sql) %}\n {{ return(column_schema) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.get_empty_subquery_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391708, "supported_languages": null}, "macro.dbt.get_columns_in_query": {"name": "get_columns_in_query", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.get_columns_in_query", "macro_sql": "{% macro get_columns_in_query(select_sql) -%}\n {{ return(adapter.dispatch('get_columns_in_query', 'dbt')(select_sql)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__get_columns_in_query"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391714, "supported_languages": null}, "macro.dbt.default__get_columns_in_query": {"name": "default__get_columns_in_query", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.default__get_columns_in_query", "macro_sql": "{% macro default__get_columns_in_query(select_sql) %}\n {% call statement('get_columns_in_query', fetch_result=True, auto_begin=False) -%}\n {{ get_empty_subquery_sql(select_sql) }}\n {% endcall %}\n {{ return(load_result('get_columns_in_query').table.columns | map(attribute='name') | list) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement", "macro.dbt.get_empty_subquery_sql"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3917172, "supported_languages": null}, "macro.dbt.alter_column_type": {"name": "alter_column_type", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.alter_column_type", "macro_sql": "{% macro alter_column_type(relation, column_name, new_column_type) -%}\n {{ return(adapter.dispatch('alter_column_type', 'dbt')(relation, column_name, new_column_type)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__alter_column_type"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391723, "supported_languages": null}, "macro.dbt.default__alter_column_type": {"name": "default__alter_column_type", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.default__alter_column_type", "macro_sql": "{% macro default__alter_column_type(relation, column_name, new_column_type) -%}\n {#\n 1. Create a new column (w/ temp name and correct type)\n 2. Copy data over to it\n 3. Drop the existing column (cascade!)\n 4. Rename the new column to existing column\n #}\n {%- set tmp_column = column_name + \"__dbt_alter\" -%}\n\n {% call statement('alter_column_type') %}\n alter table {{ relation.render() }} add column {{ adapter.quote(tmp_column) }} {{ new_column_type }};\n update {{ relation.render() }} set {{ adapter.quote(tmp_column) }} = {{ adapter.quote(column_name) }};\n alter table {{ relation.render() }} drop column {{ adapter.quote(column_name) }} cascade;\n alter table {{ relation.render() }} rename column {{ adapter.quote(tmp_column) }} to {{ adapter.quote(column_name) }}\n {% endcall %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.statement"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391726, "supported_languages": null}, "macro.dbt.alter_relation_add_remove_columns": {"name": "alter_relation_add_remove_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.alter_relation_add_remove_columns", "macro_sql": "{% macro alter_relation_add_remove_columns(relation, add_columns = none, remove_columns = none) -%}\n {{ return(adapter.dispatch('alter_relation_add_remove_columns', 'dbt')(relation, add_columns, remove_columns)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__alter_relation_add_remove_columns"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3917298, "supported_languages": null}, "macro.dbt.default__alter_relation_add_remove_columns": {"name": "default__alter_relation_add_remove_columns", "resource_type": "macro", "package_name": "dbt", "path": "macros/adapters/columns.sql", "original_file_path": "macros/adapters/columns.sql", "unique_id": "macro.dbt.default__alter_relation_add_remove_columns", "macro_sql": "{% macro default__alter_relation_add_remove_columns(relation, add_columns, remove_columns) %}\n\n {% if add_columns is none %}\n {% set add_columns = [] %}\n {% endif %}\n {% if remove_columns is none %}\n {% set remove_columns = [] %}\n {% endif %}\n\n {% set sql -%}\n\n alter {{ relation.type }} {{ relation.render() }}\n\n {% for column in add_columns %}\n add column {{ column.name }} {{ column.data_type }}{{ ',' if not loop.last }}\n {% endfor %}{{ ',' if add_columns and remove_columns }}\n\n {% for column in remove_columns %}\n drop column {{ column.name }}{{ ',' if not loop.last }}\n {% endfor %}\n\n {%- endset -%}\n\n {% do run_query(sql) %}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.run_query"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391735, "supported_languages": null}, "macro.dbt.get_fixture_sql": {"name": "get_fixture_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/unit_test_sql/get_fixture_sql.sql", "original_file_path": "macros/unit_test_sql/get_fixture_sql.sql", "unique_id": "macro.dbt.get_fixture_sql", "macro_sql": "{% macro get_fixture_sql(rows, column_name_to_data_types) %}\n-- Fixture for {{ model.name }}\n{% set default_row = {} %}\n\n{%- if not column_name_to_data_types -%}\n{#-- Use defer_relation IFF it is available in the manifest and 'this' is missing from the database --#}\n{%- set this_or_defer_relation = defer_relation if (defer_relation and not load_relation(this)) else this -%}\n{%- set columns_in_relation = adapter.get_columns_in_relation(this_or_defer_relation) -%}\n\n{%- set column_name_to_data_types = {} -%}\n{%- for column in columns_in_relation -%}\n{#-- This needs to be a case-insensitive comparison --#}\n{%- do column_name_to_data_types.update({column.name|lower: column.data_type}) -%}\n{%- endfor -%}\n{%- endif -%}\n\n{%- if not column_name_to_data_types -%}\n {{ exceptions.raise_compiler_error(\"Not able to get columns for unit test '\" ~ model.name ~ \"' from relation \" ~ this ~ \" because the relation doesn't exist\") }}\n{%- endif -%}\n\n{%- for column_name, column_type in column_name_to_data_types.items() -%}\n {%- do default_row.update({column_name: (safe_cast(\"null\", column_type) | trim )}) -%}\n{%- endfor -%}\n\n{{ validate_fixture_rows(rows, row_number) }}\n\n{%- for row in rows -%}\n{%- set formatted_row = format_row(row, column_name_to_data_types) -%}\n{%- set default_row_copy = default_row.copy() -%}\n{%- do default_row_copy.update(formatted_row) -%}\nselect\n{%- for column_name, column_value in default_row_copy.items() %} {{ column_value }} as {{ column_name }}{% if not loop.last -%}, {%- endif %}\n{%- endfor %}\n{%- if not loop.last %}\nunion all\n{% endif %}\n{%- endfor -%}\n\n{%- if (rows | length) == 0 -%}\n select\n {%- for column_name, column_value in default_row.items() %} {{ column_value }} as {{ column_name }}{% if not loop.last -%},{%- endif %}\n {%- endfor %}\n limit 0\n{%- endif -%}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.load_relation", "macro.dbt.safe_cast", "macro.dbt.validate_fixture_rows", "macro.dbt.format_row"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3917499, "supported_languages": null}, "macro.dbt.get_expected_sql": {"name": "get_expected_sql", "resource_type": "macro", "package_name": "dbt", "path": "macros/unit_test_sql/get_fixture_sql.sql", "original_file_path": "macros/unit_test_sql/get_fixture_sql.sql", "unique_id": "macro.dbt.get_expected_sql", "macro_sql": "{% macro get_expected_sql(rows, column_name_to_data_types) %}\n\n{%- if (rows | length) == 0 -%}\n select * from dbt_internal_unit_test_actual\n limit 0\n{%- else -%}\n{%- for row in rows -%}\n{%- set formatted_row = format_row(row, column_name_to_data_types) -%}\nselect\n{%- for column_name, column_value in formatted_row.items() %} {{ column_value }} as {{ column_name }}{% if not loop.last -%}, {%- endif %}\n{%- endfor %}\n{%- if not loop.last %}\nunion all\n{% endif %}\n{%- endfor -%}\n{%- endif -%}\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.format_row"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391756, "supported_languages": null}, "macro.dbt.format_row": {"name": "format_row", "resource_type": "macro", "package_name": "dbt", "path": "macros/unit_test_sql/get_fixture_sql.sql", "original_file_path": "macros/unit_test_sql/get_fixture_sql.sql", "unique_id": "macro.dbt.format_row", "macro_sql": "\n\n{%- macro format_row(row, column_name_to_data_types) -%}\n {#-- generate case-insensitive formatted row --#}\n {% set formatted_row = {} %}\n {%- for column_name, column_value in row.items() -%}\n {% set column_name = column_name|lower %}\n\n {%- if column_name not in column_name_to_data_types %}\n {#-- if user-provided row contains column name that relation does not contain, raise an error --#}\n {% set fixture_name = \"expected output\" if model.resource_type == 'unit_test' else (\"'\" ~ model.name ~ \"'\") %}\n {{ exceptions.raise_compiler_error(\n \"Invalid column name: '\" ~ column_name ~ \"' in unit test fixture for \" ~ fixture_name ~ \".\"\n \"\\nAccepted columns for \" ~ fixture_name ~ \" are: \" ~ (column_name_to_data_types.keys()|list)\n ) }}\n {%- endif -%}\n\n {%- set column_type = column_name_to_data_types[column_name] %}\n\n {#-- sanitize column_value: wrap yaml strings in quotes, apply cast --#}\n {%- set column_value_clean = column_value -%}\n {%- if column_value is string -%}\n {%- set column_value_clean = dbt.string_literal(dbt.escape_single_quotes(column_value)) -%}\n {%- elif column_value is none -%}\n {%- set column_value_clean = 'null' -%}\n {%- endif -%}\n\n {%- set row_update = {column_name: safe_cast(column_value_clean, column_type) } -%}\n {%- do formatted_row.update(row_update) -%}\n {%- endfor -%}\n {{ return(formatted_row) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.string_literal", "macro.dbt.escape_single_quotes", "macro.dbt.safe_cast"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3917599, "supported_languages": null}, "macro.dbt.validate_fixture_rows": {"name": "validate_fixture_rows", "resource_type": "macro", "package_name": "dbt", "path": "macros/unit_test_sql/get_fixture_sql.sql", "original_file_path": "macros/unit_test_sql/get_fixture_sql.sql", "unique_id": "macro.dbt.validate_fixture_rows", "macro_sql": "{%- macro validate_fixture_rows(rows, row_number) -%}\n {{ return(adapter.dispatch('validate_fixture_rows', 'dbt')(rows, row_number)) }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": ["macro.dbt.default__validate_fixture_rows"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391764, "supported_languages": null}, "macro.dbt.default__validate_fixture_rows": {"name": "default__validate_fixture_rows", "resource_type": "macro", "package_name": "dbt", "path": "macros/unit_test_sql/get_fixture_sql.sql", "original_file_path": "macros/unit_test_sql/get_fixture_sql.sql", "unique_id": "macro.dbt.default__validate_fixture_rows", "macro_sql": "{%- macro default__validate_fixture_rows(rows, row_number) -%}\n {# This is an abstract method for adapter overrides as needed #}\n{%- endmacro -%}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391767, "supported_languages": null}, "macro.dbt.resolve_model_name": {"name": "resolve_model_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/python_model/python.sql", "original_file_path": "macros/python_model/python.sql", "unique_id": "macro.dbt.resolve_model_name", "macro_sql": "{% macro resolve_model_name(input_model_name) %}\n {{ return(adapter.dispatch('resolve_model_name', 'dbt')(input_model_name)) }}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.default__resolve_model_name"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391777, "supported_languages": null}, "macro.dbt.default__resolve_model_name": {"name": "default__resolve_model_name", "resource_type": "macro", "package_name": "dbt", "path": "macros/python_model/python.sql", "original_file_path": "macros/python_model/python.sql", "unique_id": "macro.dbt.default__resolve_model_name", "macro_sql": "\n\n{%- macro default__resolve_model_name(input_model_name) -%}\n {{ input_model_name | string | replace('\"', '\\\"') }}\n{%- endmacro -%}\n\n", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3917818, "supported_languages": null}, "macro.dbt.build_ref_function": {"name": "build_ref_function", "resource_type": "macro", "package_name": "dbt", "path": "macros/python_model/python.sql", "original_file_path": "macros/python_model/python.sql", "unique_id": "macro.dbt.build_ref_function", "macro_sql": "{% macro build_ref_function(model) %}\n\n {%- set ref_dict = {} -%}\n {%- for _ref in model.refs -%}\n {% set _ref_args = [_ref.get('package'), _ref['name']] if _ref.get('package') else [_ref['name'],] %}\n {%- set resolved = ref(*_ref_args, v=_ref.get('version')) -%}\n {%- if _ref.get('version') -%}\n {% do _ref_args.extend([\"v\" ~ _ref['version']]) %}\n {%- endif -%}\n {%- do ref_dict.update({_ref_args | join('.'): resolve_model_name(resolved)}) -%}\n {%- endfor -%}\n\ndef ref(*args, **kwargs):\n refs = {{ ref_dict | tojson }}\n key = '.'.join(args)\n version = kwargs.get(\"v\") or kwargs.get(\"version\")\n if version:\n key += f\".v{version}\"\n dbt_load_df_function = kwargs.get(\"dbt_load_df_function\")\n return dbt_load_df_function(refs[key])\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.resolve_model_name"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391787, "supported_languages": null}, "macro.dbt.build_source_function": {"name": "build_source_function", "resource_type": "macro", "package_name": "dbt", "path": "macros/python_model/python.sql", "original_file_path": "macros/python_model/python.sql", "unique_id": "macro.dbt.build_source_function", "macro_sql": "{% macro build_source_function(model) %}\n\n {%- set source_dict = {} -%}\n {%- for _source in model.sources -%}\n {%- set resolved = source(*_source) -%}\n {%- do source_dict.update({_source | join('.'): resolve_model_name(resolved)}) -%}\n {%- endfor -%}\n\ndef source(*args, dbt_load_df_function):\n sources = {{ source_dict | tojson }}\n key = '.'.join(args)\n return dbt_load_df_function(sources[key])\n\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.resolve_model_name"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391792, "supported_languages": null}, "macro.dbt.build_config_dict": {"name": "build_config_dict", "resource_type": "macro", "package_name": "dbt", "path": "macros/python_model/python.sql", "original_file_path": "macros/python_model/python.sql", "unique_id": "macro.dbt.build_config_dict", "macro_sql": "{% macro build_config_dict(model) %}\n {%- set config_dict = {} -%}\n {% set config_dbt_used = zip(model.config.config_keys_used, model.config.config_keys_defaults) | list %}\n {%- for key, default in config_dbt_used -%}\n {# weird type testing with enum, would be much easier to write this logic in Python! #}\n {%- if key == \"language\" -%}\n {%- set value = \"python\" -%}\n {%- endif -%}\n {%- set value = model.config.get(key, default) -%}\n {%- do config_dict.update({key: value}) -%}\n {%- endfor -%}\nconfig_dict = {{ config_dict }}\n{% endmacro %}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3917959, "supported_languages": null}, "macro.dbt.py_script_postfix": {"name": "py_script_postfix", "resource_type": "macro", "package_name": "dbt", "path": "macros/python_model/python.sql", "original_file_path": "macros/python_model/python.sql", "unique_id": "macro.dbt.py_script_postfix", "macro_sql": "{% macro py_script_postfix(model) %}\n# This part is user provided model code\n# you will need to copy the next section to run the code\n# COMMAND ----------\n# this part is dbt logic for get ref work, do not modify\n\n{{ build_ref_function(model ) }}\n{{ build_source_function(model ) }}\n{{ build_config_dict(model) }}\n\nclass config:\n def __init__(self, *args, **kwargs):\n pass\n\n @staticmethod\n def get(key, default=None):\n return config_dict.get(key, default)\n\nclass this:\n \"\"\"dbt.this() or dbt.this.identifier\"\"\"\n database = \"{{ this.database }}\"\n schema = \"{{ this.schema }}\"\n identifier = \"{{ this.identifier }}\"\n {% set this_relation_name = resolve_model_name(this) %}\n def __repr__(self):\n return '{{ this_relation_name }}'\n\n\nclass dbtObj:\n def __init__(self, load_df_function) -> None:\n self.source = lambda *args: source(*args, dbt_load_df_function=load_df_function)\n self.ref = lambda *args, **kwargs: ref(*args, **kwargs, dbt_load_df_function=load_df_function)\n self.config = config\n self.this = this()\n self.is_incremental = {{ is_incremental() }}\n\n# COMMAND ----------\n{{py_script_comment()}}\n{% endmacro %}", "depends_on": {"macros": ["macro.dbt.build_ref_function", "macro.dbt.build_source_function", "macro.dbt.build_config_dict", "macro.dbt.resolve_model_name", "macro.dbt.is_incremental", "macro.dbt.py_script_comment"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391801, "supported_languages": null}, "macro.dbt.py_script_comment": {"name": "py_script_comment", "resource_type": "macro", "package_name": "dbt", "path": "macros/python_model/python.sql", "original_file_path": "macros/python_model/python.sql", "unique_id": "macro.dbt.py_script_comment", "macro_sql": "{%macro py_script_comment()%}\n{%endmacro%}", "depends_on": {"macros": []}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.391806, "supported_languages": null}, "macro.dbt.test_unique": {"name": "test_unique", "resource_type": "macro", "package_name": "dbt", "path": "tests/generic/builtin.sql", "original_file_path": "tests/generic/builtin.sql", "unique_id": "macro.dbt.test_unique", "macro_sql": "{% test unique(model, column_name) %}\n {% set macro = adapter.dispatch('test_unique', 'dbt') %}\n {{ macro(model, column_name) }}\n{% endtest %}", "depends_on": {"macros": ["macro.dbt.default__test_unique"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3922958, "supported_languages": null}, "macro.dbt.test_not_null": {"name": "test_not_null", "resource_type": "macro", "package_name": "dbt", "path": "tests/generic/builtin.sql", "original_file_path": "tests/generic/builtin.sql", "unique_id": "macro.dbt.test_not_null", "macro_sql": "{% test not_null(model, column_name) %}\n {% set macro = adapter.dispatch('test_not_null', 'dbt') %}\n {{ macro(model, column_name) }}\n{% endtest %}", "depends_on": {"macros": ["macro.dbt.default__test_not_null"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3925362, "supported_languages": null}, "macro.dbt.test_accepted_values": {"name": "test_accepted_values", "resource_type": "macro", "package_name": "dbt", "path": "tests/generic/builtin.sql", "original_file_path": "tests/generic/builtin.sql", "unique_id": "macro.dbt.test_accepted_values", "macro_sql": "{% test accepted_values(model, column_name, values, quote=True) %}\n {% set macro = adapter.dispatch('test_accepted_values', 'dbt') %}\n {{ macro(model, column_name, values, quote) }}\n{% endtest %}", "depends_on": {"macros": ["macro.dbt.default__test_accepted_values"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.3928409, "supported_languages": null}, "macro.dbt.test_relationships": {"name": "test_relationships", "resource_type": "macro", "package_name": "dbt", "path": "tests/generic/builtin.sql", "original_file_path": "tests/generic/builtin.sql", "unique_id": "macro.dbt.test_relationships", "macro_sql": "{% test relationships(model, column_name, to, field) %}\n {% set macro = adapter.dispatch('test_relationships', 'dbt') %}\n {{ macro(model, column_name, to, field) }}\n{% endtest %}", "depends_on": {"macros": ["macro.dbt.default__test_relationships"]}, "description": "", "meta": {}, "docs": {"show": true, "node_color": null}, "patch_path": null, "arguments": [], "created_at": 1729197943.393118, "supported_languages": null}}, "docs": {"doc.test.somedoc": {"name": "somedoc", "resource_type": "doc", "package_name": "test", "path": "somedoc.md", "original_file_path": "models/somedoc.md", "unique_id": "doc.test.somedoc", "block_contents": "Testing, testing"}, "doc.dbt.__overview__": {"name": "__overview__", "resource_type": "doc", "package_name": "dbt", "path": "overview.md", "original_file_path": "docs/overview.md", "unique_id": "doc.dbt.__overview__", "block_contents": "### Welcome!\n\nWelcome to the auto-generated documentation for your dbt project!\n\n### Navigation\n\nYou can use the `Project` and `Database` navigation tabs on the left side of the window to explore the models\nin your project.\n\n#### Project Tab\nThe `Project` tab mirrors the directory structure of your dbt project. In this tab, you can see all of the\nmodels defined in your dbt project, as well as models imported from dbt packages.\n\n#### Database Tab\nThe `Database` tab also exposes your models, but in a format that looks more like a database explorer. This view\nshows relations (tables and views) grouped into database schemas. Note that ephemeral models are _not_ shown\nin this interface, as they do not exist in the database.\n\n### Graph Exploration\nYou can click the blue icon on the bottom-right corner of the page to view the lineage graph of your models.\n\nOn model pages, you'll see the immediate parents and children of the model you're exploring. By clicking the `Expand`\nbutton at the top-right of this lineage pane, you'll be able to see all of the models that are used to build,\nor are built from, the model you're exploring.\n\nOnce expanded, you'll be able to use the `--select` and `--exclude` model selection syntax to filter the\nmodels in the graph. For more information on model selection, check out the [dbt docs](https://docs.getdbt.com/docs/model-selection-syntax).\n\nNote that you can also right-click on models to interactively filter and explore the graph.\n\n---\n\n### More information\n\n- [What is dbt](https://docs.getdbt.com/docs/introduction)?\n- Read the [dbt viewpoint](https://docs.getdbt.com/docs/viewpoint)\n- [Installation](https://docs.getdbt.com/docs/installation)\n- Join the [dbt Community](https://www.getdbt.com/community/) for questions and discussion"}}, "exposures": {"exposure.test.simple_exposure": {"name": "simple_exposure", "resource_type": "exposure", "package_name": "test", "path": "schema.yml", "original_file_path": "models/schema.yml", "unique_id": "exposure.test.simple_exposure", "fqn": ["test", "simple_exposure"], "type": "dashboard", "owner": {"email": "something@example.com", "name": null}, "description": "", "label": null, "maturity": null, "meta": {}, "tags": [], "config": {"enabled": true}, "unrendered_config": {}, "url": null, "depends_on": {"macros": [], "nodes": ["source.test.my_source.my_table", "model.test.my_model"]}, "refs": [{"name": "my_model", "package": null, "version": null}], "sources": [["my_source", "my_table"]], "metrics": [], "created_at": 1729197944.2103732}}, "metrics": {"metric.test.blue_customers_post_2010": {"name": "blue_customers_post_2010", "resource_type": "metric", "package_name": "test", "path": "schema.yml", "original_file_path": "models/schema.yml", "unique_id": "metric.test.blue_customers_post_2010", "fqn": ["test", "blue_customers_post_2010"], "description": "", "label": "Blue Customers since 2010", "type": "simple", "type_params": {"measure": {"name": "customers", "filter": {"where_filters": [{"where_sql_template": "{{ Dimension('id__favorite_color') }} = 'blue'"}]}, "alias": null, "join_to_timespine": false, "fill_nulls_with": null}, "input_measures": [{"name": "customers", "filter": {"where_filters": [{"where_sql_template": "{{ Dimension('id__favorite_color') }} = 'blue'"}]}, "alias": null, "join_to_timespine": false, "fill_nulls_with": null}], "numerator": null, "denominator": null, "expr": null, "window": null, "grain_to_date": null, "metrics": [], "conversion_type_params": null, "cumulative_type_params": null}, "filter": {"where_filters": [{"where_sql_template": "{{ TimeDimension('id__created_at', 'day') }} > '2010-01-01'"}]}, "metadata": null, "time_granularity": null, "meta": {}, "tags": [], "config": {"enabled": true, "group": null, "meta": {}}, "unrendered_config": {}, "sources": [], "depends_on": {"macros": [], "nodes": ["semantic_model.test.semantic_people"]}, "refs": [], "metrics": [], "created_at": 1729197944.252921, "group": null}, "metric.test.customers": {"name": "customers", "resource_type": "metric", "package_name": "test", "path": "schema.yml", "original_file_path": "models/schema.yml", "unique_id": "metric.test.customers", "fqn": ["test", "customers"], "description": "", "label": "Customers Metric", "type": "simple", "type_params": {"measure": {"name": "customers", "filter": null, "alias": null, "join_to_timespine": false, "fill_nulls_with": null}, "input_measures": [{"name": "customers", "filter": null, "alias": null, "join_to_timespine": false, "fill_nulls_with": null}], "numerator": null, "denominator": null, "expr": null, "window": null, "grain_to_date": null, "metrics": [], "conversion_type_params": null, "cumulative_type_params": null}, "filter": null, "metadata": null, "time_granularity": null, "meta": {}, "tags": [], "config": {"enabled": true, "group": null, "meta": {}}, "unrendered_config": {}, "sources": [], "depends_on": {"macros": [], "nodes": ["semantic_model.test.semantic_people"]}, "refs": [], "metrics": [], "created_at": 1729197944.253312, "group": null}, "metric.test.ratio_of_blue_customers_to_red_customers": {"name": "ratio_of_blue_customers_to_red_customers", "resource_type": "metric", "package_name": "test", "path": "schema.yml", "original_file_path": "models/schema.yml", "unique_id": "metric.test.ratio_of_blue_customers_to_red_customers", "fqn": ["test", "ratio_of_blue_customers_to_red_customers"], "description": "", "label": "Very Important Customer Color Ratio", "type": "ratio", "type_params": {"measure": null, "input_measures": [{"name": "customers", "filter": null, "alias": null, "join_to_timespine": false, "fill_nulls_with": null}], "numerator": {"name": "customers", "filter": {"where_filters": [{"where_sql_template": "{{ Dimension('id__favorite_color')}} = 'blue'"}]}, "alias": null, "offset_window": null, "offset_to_grain": null}, "denominator": {"name": "customers", "filter": {"where_filters": [{"where_sql_template": "{{ Dimension('id__favorite_color')}} = 'red'"}]}, "alias": null, "offset_window": null, "offset_to_grain": null}, "expr": null, "window": null, "grain_to_date": null, "metrics": [], "conversion_type_params": null, "cumulative_type_params": null}, "filter": null, "metadata": null, "time_granularity": null, "meta": {}, "tags": [], "config": {"enabled": true, "group": null, "meta": {}}, "unrendered_config": {}, "sources": [], "depends_on": {"macros": [], "nodes": ["metric.test.customers"]}, "refs": [], "metrics": [], "created_at": 1729197944.2556689, "group": null}, "metric.test.doubled_blue_customers": {"name": "doubled_blue_customers", "resource_type": "metric", "package_name": "test", "path": "schema.yml", "original_file_path": "models/schema.yml", "unique_id": "metric.test.doubled_blue_customers", "fqn": ["test", "doubled_blue_customers"], "description": "", "label": "Inflated blue customer numbers", "type": "derived", "type_params": {"measure": null, "input_measures": [{"name": "customers", "filter": null, "alias": null, "join_to_timespine": false, "fill_nulls_with": null}], "numerator": null, "denominator": null, "expr": "customers * 2", "window": null, "grain_to_date": null, "metrics": [{"name": "customers", "filter": {"where_filters": [{"where_sql_template": "{{ Dimension('id__favorite_color')}} = 'blue'"}]}, "alias": null, "offset_window": null, "offset_to_grain": null}], "conversion_type_params": null, "cumulative_type_params": null}, "filter": null, "metadata": null, "time_granularity": null, "meta": {}, "tags": [], "config": {"enabled": true, "group": null, "meta": {}}, "unrendered_config": {}, "sources": [], "depends_on": {"macros": [], "nodes": ["metric.test.customers"]}, "refs": [], "metrics": [], "created_at": 1729197944.256597, "group": null}}, "groups": {}, "selectors": {}, "disabled": {"model.test.disabled_model": [{"database": "dbt", "schema": "test17291979431657508683_test_previous_version_state", "name": "disabled_model", "resource_type": "model", "package_name": "test", "path": "disabled_model.sql", "original_file_path": "models/disabled_model.sql", "unique_id": "model.test.disabled_model", "fqn": ["test", "disabled_model"], "alias": "disabled_model", "checksum": {"name": "sha256", "checksum": "597106d23ce34e3cd2430588e5c1cf474ebdd138fc47e09b925a4ab258a27acc"}, "config": {"enabled": false, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "view", "incremental_strategy": null, "batch_size": null, "lookback": 1, "begin": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": null, "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}, "event_time": null, "access": "protected"}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {"enabled": false}, "created_at": 1729197943.715627, "relation_name": "\"dbt\".\"test17291979431657508683_test_previous_version_state\".\"disabled_model\"", "raw_code": "{{ config(enabled=False) }}\nselect 2 as id", "language": "sql", "refs": [], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": []}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}, "access": "protected", "constraints": [], "version": null, "latest_version": null, "deprecation_date": null, "primary_key": [], "time_spine": null}], "snapshot.test.disabled_snapshot_seed": [{"database": "dbt", "schema": "test17291979431657508683_test_previous_version_state", "name": "disabled_snapshot_seed", "resource_type": "snapshot", "package_name": "test", "path": "disabled_snapshot_seed.sql", "original_file_path": "snapshots/disabled_snapshot_seed.sql", "unique_id": "snapshot.test.disabled_snapshot_seed", "fqn": ["test", "disabled_snapshot_seed", "disabled_snapshot_seed"], "alias": "disabled_snapshot_seed", "checksum": {"name": "sha256", "checksum": "fe76c9dd437341c9e82a0f2a8baf3148f961b768eaa0a4410cd27d3c071bd617"}, "config": {"enabled": false, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "snapshot", "incremental_strategy": null, "batch_size": null, "lookback": 1, "begin": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": "id", "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}, "event_time": null, "strategy": "check", "target_schema": "test17291979431657508683_test_previous_version_state", "target_database": null, "updated_at": null, "check_cols": "all", "snapshot_meta_column_names": {"dbt_valid_to": null, "dbt_valid_from": null, "dbt_scd_id": null, "dbt_updated_at": null}, "dbt_valid_to_current": null}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {"unique_key": "id", "strategy": "check", "check_cols": "all", "target_schema": "test17291979431657508683_test_previous_version_state", "enabled": false}, "created_at": 1729197943.826247, "relation_name": "\"dbt\".\"test17291979431657508683_test_previous_version_state\".\"disabled_snapshot_seed\"", "raw_code": "\n{{\n config(\n unique_key='id',\n strategy='check',\n check_cols='all',\n target_schema=schema,\n enabled=False,\n )\n}}\nselect * from {{ ref('my_seed') }}\n", "language": "sql", "refs": [{"name": "my_seed", "package": null, "version": null}], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": []}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}}], "analysis.test.disabled_al": [{"database": "dbt", "schema": "test17291979431657508683_test_previous_version_state", "name": "disabled_al", "resource_type": "analysis", "package_name": "test", "path": "analysis/disabled_al.sql", "original_file_path": "analyses/disabled_al.sql", "unique_id": "analysis.test.disabled_al", "fqn": ["test", "analysis", "disabled_al"], "alias": "disabled_al", "checksum": {"name": "sha256", "checksum": "32d36ad6cff0786eb562440ba60ef6c9b9a7f4c282dfb7a52eaf19d36370f0e1"}, "config": {"enabled": false, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "view", "incremental_strategy": null, "batch_size": null, "lookback": 1, "begin": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": null, "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}, "event_time": null}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {"enabled": false}, "created_at": 1729197943.8950012, "relation_name": null, "raw_code": "{{ config(enabled=False) }}\nselect 9 as id", "language": "sql", "refs": [], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": []}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}}], "test.test.disabled_just_my": [{"database": "dbt", "schema": "test17291979431657508683_test_previous_version_state_dbt_test__audit", "name": "disabled_just_my", "resource_type": "test", "package_name": "test", "path": "disabled_just_my.sql", "original_file_path": "tests/disabled_just_my.sql", "unique_id": "test.test.disabled_just_my", "fqn": ["test", "disabled_just_my"], "alias": "disabled_just_my", "checksum": {"name": "sha256", "checksum": "4f2268fd89a3b4ef899264ada6d7aa33603671cbc5d5acead7dc2eadf1add985"}, "config": {"enabled": false, "alias": null, "schema": "dbt_test__audit", "database": null, "tags": [], "meta": {}, "group": null, "materialized": "test", "severity": "ERROR", "store_failures": null, "store_failures_as": null, "where": null, "limit": null, "fail_calc": "count(*)", "warn_if": "!= 0", "error_if": "!= 0"}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {"enabled": false}, "created_at": 1729197943.954093, "relation_name": null, "raw_code": "{{ config(enabled=False) }}\n\nselect * from {{ ref('my_model') }}\nwhere false", "language": "sql", "refs": [{"name": "my_model", "package": null, "version": null}], "sources": [], "metrics": [], "depends_on": {"macros": [], "nodes": []}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}}], "test.test.disabled_check_nothing_my_model_.f2c6a72d37": [{"database": "dbt", "schema": "test17291979431657508683_test_previous_version_state_dbt_test__audit", "name": "disabled_check_nothing_my_model_", "resource_type": "test", "package_name": "test", "path": "disabled_check_nothing_my_model_.sql", "original_file_path": "models/schema.yml", "unique_id": "test.test.disabled_check_nothing_my_model_.f2c6a72d37", "fqn": ["test", "disabled_check_nothing_my_model_"], "alias": "disabled_check_nothing_my_model_", "checksum": {"name": "none", "checksum": ""}, "config": {"enabled": false, "alias": null, "schema": "dbt_test__audit", "database": null, "tags": [], "meta": {}, "group": null, "materialized": "test", "severity": "ERROR", "store_failures": null, "store_failures_as": null, "where": null, "limit": null, "fail_calc": "count(*)", "warn_if": "!= 0", "error_if": "!= 0"}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": null, "build_path": null, "unrendered_config": {"enabled": false}, "created_at": 1729197944.136138, "relation_name": null, "raw_code": "{{ test_disabled_check_nothing(**_dbt_generic_test_kwargs) }}", "language": "sql", "refs": [{"name": "my_model", "package": null, "version": null}], "sources": [], "metrics": [], "depends_on": {"macros": ["macro.test.test_disabled_check_nothing", "macro.dbt.get_where_subquery"], "nodes": []}, "compiled_path": null, "contract": {"enforced": false, "alias_types": true, "checksum": null}, "column_name": null, "file_key_name": "models.my_model", "attached_node": "model.test.my_model", "test_metadata": {"name": "disabled_check_nothing", "kwargs": {"model": "{{ get_where_subquery(ref('my_model')) }}"}, "namespace": null}}], "exposure.test.disabled_exposure": [{"name": "disabled_exposure", "resource_type": "exposure", "package_name": "test", "path": "schema.yml", "original_file_path": "models/schema.yml", "unique_id": "exposure.test.disabled_exposure", "fqn": ["test", "disabled_exposure"], "type": "dashboard", "owner": {"email": "something@example.com", "name": null}, "description": "", "label": null, "maturity": null, "meta": {}, "tags": [], "config": {"enabled": false}, "unrendered_config": {"enabled": false}, "url": null, "depends_on": {"macros": [], "nodes": []}, "refs": [{"name": "my_model", "package": null, "version": null}], "sources": [], "metrics": [], "created_at": 1729197944.2111921}], "metric.test.disabled_metric": [{"name": "disabled_metric", "resource_type": "metric", "package_name": "test", "path": "schema.yml", "original_file_path": "models/schema.yml", "unique_id": "metric.test.disabled_metric", "fqn": ["test", "disabled_metric"], "description": "", "label": "Count records", "type": "simple", "type_params": {"measure": {"name": "customers", "filter": null, "alias": null, "join_to_timespine": false, "fill_nulls_with": null}, "input_measures": [], "numerator": null, "denominator": null, "expr": null, "window": null, "grain_to_date": null, "metrics": [], "conversion_type_params": null, "cumulative_type_params": null}, "filter": {"where_filters": [{"where_sql_template": "{{ Dimension('id__favorite_color') }} = 'blue'"}]}, "metadata": null, "time_granularity": null, "meta": {}, "tags": [], "config": {"enabled": false, "group": null, "meta": {}}, "unrendered_config": {"enabled": false}, "sources": [], "depends_on": {"macros": [], "nodes": []}, "refs": [], "metrics": [], "created_at": 1729197944.25384, "group": null}], "seed.test.disabled_seed": [{"database": "dbt", "schema": "test17291979431657508683_test_previous_version_state", "name": "disabled_seed", "resource_type": "seed", "package_name": "test", "path": "disabled_seed.csv", "original_file_path": "seeds/disabled_seed.csv", "unique_id": "seed.test.disabled_seed", "fqn": ["test", "disabled_seed"], "alias": "disabled_seed", "checksum": {"name": "sha256", "checksum": "31fddd8ec40c6aba6a3a8e7d83fedea2fd0a56c47b64ea3df1847ec1b018e2d1"}, "config": {"enabled": false, "alias": null, "schema": null, "database": null, "tags": [], "meta": {}, "group": null, "materialized": "seed", "incremental_strategy": null, "batch_size": null, "lookback": 1, "begin": null, "persist_docs": {}, "post-hook": [], "pre-hook": [], "quoting": {}, "column_types": {}, "full_refresh": null, "unique_key": null, "on_schema_change": "ignore", "on_configuration_change": "apply", "grants": {}, "packages": [], "docs": {"show": true, "node_color": null}, "contract": {"enforced": false, "alias_types": true}, "event_time": null, "delimiter": ",", "quote_columns": null}, "tags": [], "description": "", "columns": {}, "meta": {}, "group": null, "docs": {"show": true, "node_color": null}, "patch_path": "test://models/schema.yml", "build_path": null, "unrendered_config": {"enabled": false}, "created_at": 1729197944.151926, "relation_name": "\"dbt\".\"test17291979431657508683_test_previous_version_state\".\"disabled_seed\"", "raw_code": "", "root_path": "/private/var/folders/79/5290gpvn3lx5jdryk4844rm80000gn/T/pytest-of-quigleymalcolm/pytest-289/project0", "depends_on": {"macros": []}}], "source.test.my_source.disabled_table": [{"database": "dbt", "schema": "my_source", "name": "disabled_table", "resource_type": "source", "package_name": "test", "path": "models/schema.yml", "original_file_path": "models/schema.yml", "unique_id": "source.test.my_source.disabled_table", "fqn": ["test", "my_source", "disabled_table"], "source_name": "my_source", "source_description": "My source", "loader": "a_loader", "identifier": "disabled_table", "quoting": {"database": null, "schema": null, "identifier": null, "column": null}, "loaded_at_field": null, "freshness": {"warn_after": {"count": null, "period": null}, "error_after": {"count": null, "period": null}, "filter": null}, "external": null, "description": "Disabled table", "columns": {}, "meta": {}, "source_meta": {}, "tags": [], "config": {"enabled": false, "event_time": null}, "patch_path": null, "unrendered_config": {"enabled": false}, "relation_name": "\"dbt\".\"my_source\".\"disabled_table\"", "created_at": 1729197944.290741, "unrendered_database": null, "unrendered_schema": null}]}, "parent_map": {"model.test.my_model": [], "model.test.metricflow_time_spine": [], "snapshot.test.snapshot_seed": ["seed.test.my_seed"], "analysis.test.a": [], "test.test.just_my": ["model.test.my_model"], "seed.test.my_seed": [], "test.test.not_null_my_model_id.43e0e9183a": ["model.test.my_model"], "test.test.check_nothing_my_model_.d5a5e66110": ["model.test.my_model"], "source.test.my_source.my_table": [], "exposure.test.simple_exposure": ["model.test.my_model", "source.test.my_source.my_table"], "metric.test.blue_customers_post_2010": ["semantic_model.test.semantic_people"], "metric.test.customers": ["semantic_model.test.semantic_people"], "metric.test.ratio_of_blue_customers_to_red_customers": ["metric.test.customers"], "metric.test.doubled_blue_customers": ["metric.test.customers"], "semantic_model.test.semantic_people": ["model.test.my_model"]}, "child_map": {"model.test.my_model": ["exposure.test.simple_exposure", "semantic_model.test.semantic_people", "test.test.check_nothing_my_model_.d5a5e66110", "test.test.just_my", "test.test.not_null_my_model_id.43e0e9183a"], "model.test.metricflow_time_spine": [], "snapshot.test.snapshot_seed": [], "analysis.test.a": [], "test.test.just_my": [], "seed.test.my_seed": ["snapshot.test.snapshot_seed"], "test.test.not_null_my_model_id.43e0e9183a": [], "test.test.check_nothing_my_model_.d5a5e66110": [], "source.test.my_source.my_table": ["exposure.test.simple_exposure"], "exposure.test.simple_exposure": [], "metric.test.blue_customers_post_2010": [], "metric.test.customers": ["metric.test.doubled_blue_customers", "metric.test.ratio_of_blue_customers_to_red_customers"], "metric.test.ratio_of_blue_customers_to_red_customers": [], "metric.test.doubled_blue_customers": [], "semantic_model.test.semantic_people": ["metric.test.blue_customers_post_2010", "metric.test.customers"]}, "group_map": {}, "saved_queries": {}, "semantic_models": {"semantic_model.test.semantic_people": {"name": "semantic_people", "resource_type": "semantic_model", "package_name": "test", "path": "schema.yml", "original_file_path": "models/schema.yml", "unique_id": "semantic_model.test.semantic_people", "fqn": ["test", "semantic_people"], "model": "ref('my_model')", "node_relation": {"alias": "my_model", "schema_name": "test17291979431657508683_test_previous_version_state", "database": "dbt", "relation_name": "\"dbt\".\"test17291979431657508683_test_previous_version_state\".\"my_model\""}, "description": null, "label": null, "defaults": {"agg_time_dimension": "created_at"}, "entities": [{"name": "id", "type": "primary", "description": null, "label": null, "role": null, "expr": null}], "measures": [{"name": "years_tenure", "agg": "sum", "description": null, "label": null, "create_metric": false, "expr": "tenure", "agg_params": null, "non_additive_dimension": null, "agg_time_dimension": null}, {"name": "people", "agg": "count", "description": null, "label": null, "create_metric": false, "expr": "id", "agg_params": null, "non_additive_dimension": null, "agg_time_dimension": null}, {"name": "customers", "agg": "count", "description": null, "label": null, "create_metric": false, "expr": "id", "agg_params": null, "non_additive_dimension": null, "agg_time_dimension": null}], "dimensions": [{"name": "favorite_color", "type": "categorical", "description": null, "label": null, "is_partition": false, "type_params": null, "expr": null, "metadata": null}, {"name": "created_at", "type": "time", "description": null, "label": null, "is_partition": false, "type_params": {"time_granularity": "day", "validity_params": null}, "expr": null, "metadata": null}], "metadata": null, "depends_on": {"macros": [], "nodes": ["model.test.my_model"]}, "refs": [{"name": "my_model", "package": null, "version": null}], "created_at": 1729197944.286252, "config": {"enabled": true, "group": null, "meta": {}}, "unrendered_config": {}, "primary_entity": null, "group": null}}, "unit_tests": {}} diff --git a/tests/functional/artifacts/expected_manifest.py b/tests/functional/artifacts/expected_manifest.py index 68e7799182b..f8091c1ddd2 100644 --- a/tests/functional/artifacts/expected_manifest.py +++ b/tests/functional/artifacts/expected_manifest.py @@ -39,6 +39,10 @@ def get_rendered_model_config(**updates): "docs": {"node_color": None, "show": True}, "contract": {"enforced": False, "alias_types": True}, "access": "protected", + "event_time": None, + "lookback": 1, + "batch_size": None, + "begin": None, } result.update(updates) return result @@ -74,6 +78,10 @@ def get_rendered_seed_config(**updates): "incremental_strategy": None, "docs": {"node_color": None, "show": True}, "contract": {"enforced": False, "alias_types": True}, + "event_time": None, + "lookback": 1, + "batch_size": None, + "begin": None, } result.update(updates) return result @@ -97,6 +105,13 @@ def get_rendered_snapshot_config(**updates): "post-hook": [], "column_types": {}, "quoting": {}, + "snapshot_meta_column_names": { + "dbt_valid_to": None, + "dbt_valid_from": None, + "dbt_updated_at": None, + "dbt_scd_id": None, + }, + "dbt_valid_to_current": None, "tags": [], "persist_docs": {}, "full_refresh": None, @@ -114,6 +129,10 @@ def get_rendered_snapshot_config(**updates): "incremental_strategy": None, "docs": {"node_color": None, "show": True}, "contract": {"enforced": False, "alias_types": True}, + "event_time": None, + "lookback": 1, + "batch_size": None, + "begin": None, } result.update(updates) return result @@ -292,6 +311,7 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "first_name": { "name": "first_name", @@ -301,6 +321,7 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "email": { "name": "email", @@ -310,6 +331,7 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "ip_address": { "name": "ip_address", @@ -319,6 +341,7 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "updated_at": { "name": "updated_at", @@ -328,6 +351,7 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, }, "contract": {"checksum": None, "enforced": False, "alias_types": True}, @@ -343,6 +367,7 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "access": "protected", "version": None, "latest_version": None, + "time_spine": None, }, "model.test.second_model": { "compiled_path": os.path.join(compiled_model_path, "second_model.sql"), @@ -385,6 +410,7 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "first_name": { "name": "first_name", @@ -394,6 +420,7 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "email": { "name": "email", @@ -403,6 +430,7 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "ip_address": { "name": "ip_address", @@ -412,6 +440,7 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "updated_at": { "name": "updated_at", @@ -421,6 +450,7 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, }, "contract": {"checksum": None, "enforced": False, "alias_types": True}, @@ -436,6 +466,7 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "access": "protected", "version": None, "latest_version": None, + "time_spine": None, }, "seed.test.seed": { "build_path": None, @@ -468,6 +499,7 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "first_name": { "name": "first_name", @@ -477,6 +509,7 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "email": { "name": "email", @@ -486,6 +519,7 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "ip_address": { "name": "ip_address", @@ -495,6 +529,7 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "updated_at": { "name": "updated_at", @@ -504,6 +539,7 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, }, "docs": {"node_color": None, "show": True}, @@ -730,10 +766,12 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "quote": None, "tags": [], "constraints": [], + "granularity": None, } }, "config": { "enabled": True, + "event_time": None, }, "quoting": { "database": None, @@ -770,6 +808,8 @@ def expected_seeded_manifest(project, model_database=None, quote_model=False): "unique_id": "source.test.my_source.my_table", "fqn": ["test", "my_source", "my_table"], "unrendered_config": {}, + "unrendered_database": None, + "unrendered_schema": "{{ var('test_schema') }}", }, }, "exposures": { @@ -957,6 +997,7 @@ def expected_references_manifest(project): "version": None, "latest_version": None, "constraints": [], + "time_spine": None, }, "model.test.ephemeral_summary": { "alias": "ephemeral_summary", @@ -972,6 +1013,7 @@ def expected_references_manifest(project): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "ct": { "description": "The number of instances of the first name", @@ -981,6 +1023,7 @@ def expected_references_manifest(project): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, }, "config": get_rendered_model_config(materialized="table", group="test_group"), @@ -1026,6 +1069,7 @@ def expected_references_manifest(project): "version": None, "latest_version": None, "constraints": [], + "time_spine": None, }, "model.test.view_summary": { "alias": "view_summary", @@ -1041,6 +1085,7 @@ def expected_references_manifest(project): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "ct": { "description": "The number of instances of the first name", @@ -1050,6 +1095,7 @@ def expected_references_manifest(project): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, }, "config": get_rendered_model_config(), @@ -1091,6 +1137,7 @@ def expected_references_manifest(project): "version": None, "latest_version": None, "constraints": [], + "time_spine": None, }, "seed.test.seed": { "alias": "seed", @@ -1105,6 +1152,7 @@ def expected_references_manifest(project): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "first_name": { "name": "first_name", @@ -1114,6 +1162,7 @@ def expected_references_manifest(project): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "email": { "name": "email", @@ -1123,6 +1172,7 @@ def expected_references_manifest(project): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "ip_address": { "name": "ip_address", @@ -1132,6 +1182,7 @@ def expected_references_manifest(project): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "updated_at": { "name": "updated_at", @@ -1141,6 +1192,7 @@ def expected_references_manifest(project): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, }, "config": get_rendered_seed_config(), @@ -1219,10 +1271,12 @@ def expected_references_manifest(project): "quote": None, "tags": [], "constraints": [], + "granularity": None, } }, "config": { "enabled": True, + "event_time": None, }, "quoting": { "database": False, @@ -1258,6 +1312,8 @@ def expected_references_manifest(project): "unique_id": "source.test.my_source.my_table", "fqn": ["test", "my_source", "my_table"], "unrendered_config": {}, + "unrendered_database": None, + "unrendered_schema": "{{ var('test_schema') }}", }, }, "exposures": { @@ -1487,6 +1543,7 @@ def expected_versions_manifest(project): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "ct": { "description": "The number of instances of the first name", @@ -1496,6 +1553,7 @@ def expected_versions_manifest(project): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, }, "config": get_rendered_model_config( @@ -1544,6 +1602,7 @@ def expected_versions_manifest(project): "access": "protected", "version": 1, "latest_version": 2, + "time_spine": None, }, "model.test.versioned_model.v2": { "alias": "versioned_model_v2", @@ -1559,6 +1618,7 @@ def expected_versions_manifest(project): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, "extra": { "description": "", @@ -1568,6 +1628,7 @@ def expected_versions_manifest(project): "quote": None, "tags": [], "constraints": [], + "granularity": None, }, }, "config": get_rendered_model_config( @@ -1612,6 +1673,7 @@ def expected_versions_manifest(project): "access": "protected", "version": 2, "latest_version": 2, + "time_spine": None, }, "model.test.ref_versioned_model": { "alias": "ref_versioned_model", @@ -1669,6 +1731,7 @@ def expected_versions_manifest(project): "access": "protected", "version": None, "latest_version": None, + "time_spine": None, }, "test.test.unique_versioned_model_v1_first_name.6138195dec": { "alias": "unique_versioned_model_v1_first_name", diff --git a/tests/functional/artifacts/expected_run_results.py b/tests/functional/artifacts/expected_run_results.py index 3a3148eba4d..06c1c727a60 100644 --- a/tests/functional/artifacts/expected_run_results.py +++ b/tests/functional/artifacts/expected_run_results.py @@ -21,6 +21,7 @@ def expected_run_results(): "compiled": True, "compiled_code": ANY, "relation_name": ANY, + "batch_results": None, }, { "status": "success", @@ -34,6 +35,7 @@ def expected_run_results(): "compiled": True, "compiled_code": ANY, "relation_name": ANY, + "batch_results": None, }, { "status": "success", @@ -47,6 +49,7 @@ def expected_run_results(): "compiled": None, "compiled_code": ANY, "relation_name": None, + "batch_results": None, }, { "status": "success", @@ -60,6 +63,7 @@ def expected_run_results(): "compiled": True, "compiled_code": ANY, "relation_name": ANY, + "batch_results": None, }, { "status": "success", @@ -73,6 +77,7 @@ def expected_run_results(): "compiled": True, "compiled_code": ANY, "relation_name": None, + "batch_results": None, }, { "status": "success", @@ -86,6 +91,7 @@ def expected_run_results(): "compiled": True, "compiled_code": ANY, "relation_name": None, + "batch_results": None, }, { "status": "success", @@ -99,6 +105,7 @@ def expected_run_results(): "compiled": True, "compiled_code": ANY, "relation_name": None, + "batch_results": None, }, ] @@ -117,6 +124,7 @@ def expected_references_run_results(): "compiled": True, "compiled_code": ANY, "relation_name": ANY, + "batch_results": None, }, { "status": "success", @@ -130,6 +138,7 @@ def expected_references_run_results(): "compiled": True, "compiled_code": ANY, "relation_name": ANY, + "batch_results": None, }, { "status": "success", @@ -143,6 +152,7 @@ def expected_references_run_results(): "compiled": None, "compiled_code": ANY, "relation_name": ANY, + "batch_results": None, }, { "status": "success", @@ -156,6 +166,7 @@ def expected_references_run_results(): "compiled": True, "compiled_code": ANY, "relation_name": ANY, + "batch_results": None, }, ] @@ -174,6 +185,7 @@ def expected_versions_run_results(): "compiled": True, "compiled_code": ANY, "relation_name": ANY, + "batch_results": None, }, { "status": "success", @@ -187,6 +199,7 @@ def expected_versions_run_results(): "compiled": True, "compiled_code": ANY, "relation_name": ANY, + "batch_results": None, }, { "status": "success", @@ -200,6 +213,7 @@ def expected_versions_run_results(): "compiled": True, "compiled_code": ANY, "relation_name": ANY, + "batch_results": None, }, { "status": "success", @@ -213,6 +227,7 @@ def expected_versions_run_results(): "compiled": True, "compiled_code": ANY, "relation_name": ANY, + "batch_results": None, }, { "status": "success", @@ -226,6 +241,7 @@ def expected_versions_run_results(): "compiled": True, "compiled_code": ANY, "relation_name": ANY, + "batch_results": None, }, { "status": "success", @@ -239,5 +255,6 @@ def expected_versions_run_results(): "compiled": True, "compiled_code": ANY, "relation_name": ANY, + "batch_results": None, }, ] diff --git a/tests/functional/artifacts/test_artifacts.py b/tests/functional/artifacts/test_artifacts.py index 4126266b129..92ec8abe196 100644 --- a/tests/functional/artifacts/test_artifacts.py +++ b/tests/functional/artifacts/test_artifacts.py @@ -8,6 +8,7 @@ from dbt.artifacts.schemas.results import RunStatus from dbt.artifacts.schemas.run import RunResultsArtifact from dbt.contracts.graph.manifest import WritableManifest +from dbt.events.types import ArtifactWritten from dbt.tests.util import ( check_datetime_between, get_artifact, @@ -24,6 +25,7 @@ expected_run_results, expected_versions_run_results, ) +from tests.utils import EventCatcher models__schema_yml = """ version: 2 @@ -617,8 +619,9 @@ def models(self): # Test generic "docs generate" command def test_run_and_generate(self, project, manifest_schema_path, run_results_schema_path): + catcher = EventCatcher(ArtifactWritten) start_time = datetime.utcnow() - results = run_dbt(["compile"]) + results = run_dbt(args=["compile"], callbacks=[catcher.catch]) assert len(results) == 7 verify_manifest( project, @@ -627,6 +630,38 @@ def test_run_and_generate(self, project, manifest_schema_path, run_results_schem manifest_schema_path, ) verify_run_results(project, expected_run_results(), start_time, run_results_schema_path) + # manifest written twice, semantic manifest written twice, run results written once + assert len(catcher.caught_events) == 5 + assert ( + len( + [ + event + for event in catcher.caught_events + if event.data.artifact_type == "WritableManifest" + ] + ) + > 0 + ) + assert ( + len( + [ + event + for event in catcher.caught_events + if event.data.artifact_type == "SemanticManifest" + ] + ) + > 0 + ) + assert ( + len( + [ + event + for event in catcher.caught_events + if event.data.artifact_type == "RunExecutionResult" + ] + ) + > 0 + ) class TestVerifyArtifactsReferences(BaseVerifyProject): diff --git a/tests/functional/artifacts/test_run_results.py b/tests/functional/artifacts/test_run_results.py index dea947f342b..a474f80c4c7 100644 --- a/tests/functional/artifacts/test_run_results.py +++ b/tests/functional/artifacts/test_run_results.py @@ -55,7 +55,7 @@ def project_config_update(self): def test_results_serializable(self, project): results = run_dbt(["run"]) - assert len(results.results) == 1 + assert len(results.results) == 2 # This test is failing due to the faulty assumptions that run_results.json would diff --git a/tests/functional/cli/test_option_interaction_validations.py b/tests/functional/cli/test_option_interaction_validations.py new file mode 100644 index 00000000000..3cee244aedd --- /dev/null +++ b/tests/functional/cli/test_option_interaction_validations.py @@ -0,0 +1,49 @@ +import pytest + +from dbt.tests.util import run_dbt + + +class TestEventTimeEndEventTimeStart: + @pytest.mark.parametrize( + "event_time_start,event_time_end,expect_pass", + [ + ("2024-10-01", "2024-10-02", True), + ("2024-10-02", "2024-10-01", False), + ], + ) + def test_option_combo(self, project, event_time_start, event_time_end, expect_pass): + try: + run_dbt( + [ + "build", + "--event-time-start", + event_time_start, + "--event-time-end", + event_time_end, + ] + ) + assert expect_pass + except Exception as e: + assert ( + "Value for `--event-time-start` must be less than `--event-time-end`" + in e.__str__() + ) + assert not expect_pass + + +class TestEventTimeEndEventTimeStartMutuallyRequired: + @pytest.mark.parametrize( + "specified,missing", + [ + ("--event-time-start", "--event-time-end"), + ("--event-time-end", "--event-time-start"), + ], + ) + def test_option_combo(self, project, specified, missing): + try: + run_dbt(["build", specified, "2024-10-01"]) + assert False, f"An error should have been raised for missing `{missing}` flag" + except Exception as e: + assert ( + f"When specifying `{specified}`, `{missing}` must also be present." in e.__str__() + ) diff --git a/tests/functional/compile/fixtures.py b/tests/functional/compile/fixtures.py index e0be7c895bf..97ccd6d16b6 100644 --- a/tests/functional/compile/fixtures.py +++ b/tests/functional/compile/fixtures.py @@ -42,6 +42,15 @@ select sum(n) from t; """ +first_ephemeral_model_with_alias_sql = """ +{{ config(materialized = 'ephemeral', alias = 'first_alias') }} +select 1 as fun +""" + +second_ephemeral_model_with_alias_sql = """ +select * from {{ ref('first_ephemeral_model_with_alias') }} +""" + schema_yml = """ version: 2 diff --git a/tests/functional/compile/test_compile.py b/tests/functional/compile/test_compile.py index c400ca8a751..9976fd7b071 100644 --- a/tests/functional/compile/test_compile.py +++ b/tests/functional/compile/test_compile.py @@ -10,10 +10,12 @@ from tests.functional.assertions.test_runner import dbtTestRunner from tests.functional.compile.fixtures import ( first_ephemeral_model_sql, + first_ephemeral_model_with_alias_sql, first_model_sql, model_multiline_jinja, schema_yml, second_ephemeral_model_sql, + second_ephemeral_model_with_alias_sql, second_model_sql, third_ephemeral_model_sql, with_recursive_model_sql, @@ -128,6 +130,24 @@ def test_with_recursive_cte(self, project): ] +class TestEphemeralModelWithAlias: + @pytest.fixture(scope="class") + def models(self): + return { + "first_ephemeral_model_with_alias.sql": first_ephemeral_model_with_alias_sql, + "second_ephemeral_model_with_alias.sql": second_ephemeral_model_with_alias_sql, + } + + def test_compile(self, project): + run_dbt(["compile"]) + + assert get_lines("second_ephemeral_model_with_alias") == [ + "with __dbt__cte__first_alias as (", + "select 1 as fun", + ") select * from __dbt__cte__first_alias", + ] + + class TestCompile: @pytest.fixture(scope="class") def models(self): diff --git a/tests/functional/configs/test_configs.py b/tests/functional/configs/test_configs.py index 2bbfac85c5c..952b951e507 100644 --- a/tests/functional/configs/test_configs.py +++ b/tests/functional/configs/test_configs.py @@ -2,6 +2,7 @@ import pytest +from dbt.exceptions import SchemaConfigError from dbt.tests.util import ( check_relations_equal, run_dbt, @@ -81,7 +82,7 @@ def test_tests_materialization_proj_config(self, project): tests_dir = os.path.join(project.project_root, "tests") write_file("select * from foo", tests_dir, "test.sql") - with pytest.raises(ValidationError): + with pytest.raises(SchemaConfigError): run_dbt() @@ -93,7 +94,7 @@ def test_seeds_materialization_proj_config(self, project): seeds_dir = os.path.join(project.project_root, "seeds") write_file("id1, id2\n1, 2", seeds_dir, "seed.csv") - with pytest.raises(ValidationError): + with pytest.raises(SchemaConfigError): run_dbt() @@ -107,7 +108,7 @@ def test_seeds_materialization_schema_config(self, project): ) write_file("id1, id2\n1, 2", seeds_dir, "myseed.csv") - with pytest.raises(ValidationError): + with pytest.raises(SchemaConfigError): run_dbt() diff --git a/tests/functional/configs/test_configs_in_schema_files.py b/tests/functional/configs/test_configs_in_schema_files.py index e0b85686895..72d41722a53 100644 --- a/tests/functional/configs/test_configs_in_schema_files.py +++ b/tests/functional/configs/test_configs_in_schema_files.py @@ -110,6 +110,15 @@ class TestSchemaFileConfigs: + @pytest.fixture(scope="class") + def expected_unrendered_config(self): + # my_attr is unrendered when state_modified_compare_more_unrendered_values: True + return { + "materialized": "view", + "meta": {"my_attr": "{{ var('my_var') }}", "owner": "Julie Smith"}, + "tags": ["tag_1_in_model", "tag_2_in_model"], + } + @pytest.fixture(scope="class") def models(self): return { @@ -125,6 +134,9 @@ def seeds(self): @pytest.fixture(scope="class") def project_config_update(self): return { + "flags": { + "state_modified_compare_more_unrendered_values": True, + }, "models": { "+meta": { "company": "NuMade", @@ -160,6 +172,7 @@ def project_config_update(self): def test_config_layering( self, project, + expected_unrendered_config, ): # run seed @@ -219,11 +232,7 @@ def test_config_layering( model_node = manifest.nodes[model_id] assert model_node.config.materialized == "view" - model_unrendered_config = { - "materialized": "view", - "meta": {"my_attr": "TESTING", "owner": "Julie Smith"}, - "tags": ["tag_1_in_model", "tag_2_in_model"], - } + model_unrendered_config = expected_unrendered_config assert model_node.unrendered_config == model_unrendered_config # look for test meta @@ -251,6 +260,57 @@ def test_config_layering( run_dbt(["run"]) +class TestLegacySchemaFileConfigs(TestSchemaFileConfigs): + @pytest.fixture(scope="class") + def expected_unrendered_config(self): + # my_attr is rendered ("TESTING") when state_modified_compare_more_unrendered_values: False + return { + "materialized": "view", + "meta": {"my_attr": "TESTING", "owner": "Julie Smith"}, + "tags": ["tag_1_in_model", "tag_2_in_model"], + } + + @pytest.fixture(scope="class") + def project_config_update(self): + return { + # The uncommented below lines can be removed once the default behaviour is flipped. + # state_modified_compare_more_unrendered_values defaults to false currently + # "flags": { + # "state_modified_compare_more_unrendered_values": False, + # }, + "models": { + "+meta": { + "company": "NuMade", + }, + "test": { + "+meta": { + "project": "test", + }, + "tagged": { + "+meta": { + "team": "Core Team", + }, + "tags": ["tag_in_project"], + "model": { + "materialized": "table", + "+meta": { + "owner": "Julie Dent", + }, + }, + }, + }, + }, + "vars": { + "test": { + "my_var": "TESTING", + } + }, + "seeds": { + "quote_columns": False, + }, + } + + list_schema_yml = """ - name: my_name - name: alt_name diff --git a/tests/functional/configs/test_contract_configs.py b/tests/functional/configs/test_contract_configs.py index 3af9415e3e3..179b0058a8d 100644 --- a/tests/functional/configs/test_contract_configs.py +++ b/tests/functional/configs/test_contract_configs.py @@ -331,7 +331,7 @@ def test__model_contract_true(self, project): assert contract_actual_config.enforced is True - expected_columns = "{'id': ColumnInfo(name='id', description='hello', meta={}, data_type='integer', constraints=[ColumnLevelConstraint(type=, name=None, expression=None, warn_unenforced=True, warn_unsupported=True, to=None, to_columns=[]), ColumnLevelConstraint(type=, name=None, expression=None, warn_unenforced=True, warn_unsupported=True, to=None, to_columns=[]), ColumnLevelConstraint(type=, name=None, expression='(id > 0)', warn_unenforced=True, warn_unsupported=True, to=None, to_columns=[])], quote=True, tags=[], _extra={}), 'color': ColumnInfo(name='color', description='', meta={}, data_type='string', constraints=[], quote=None, tags=[], _extra={}), 'date_day': ColumnInfo(name='date_day', description='', meta={}, data_type='date', constraints=[], quote=None, tags=[], _extra={})}" + expected_columns = "{'id': ColumnInfo(name='id', description='hello', meta={}, data_type='integer', constraints=[ColumnLevelConstraint(type=, name=None, expression=None, warn_unenforced=True, warn_unsupported=True, to=None, to_columns=[]), ColumnLevelConstraint(type=, name=None, expression=None, warn_unenforced=True, warn_unsupported=True, to=None, to_columns=[]), ColumnLevelConstraint(type=, name=None, expression='(id > 0)', warn_unenforced=True, warn_unsupported=True, to=None, to_columns=[])], quote=True, tags=[], _extra={}, granularity=None), 'color': ColumnInfo(name='color', description='', meta={}, data_type='string', constraints=[], quote=None, tags=[], _extra={}, granularity=None), 'date_day': ColumnInfo(name='date_day', description='', meta={}, data_type='date', constraints=[], quote=None, tags=[], _extra={}, granularity=None)}" assert expected_columns == str(my_model_columns) diff --git a/tests/functional/configs/test_disabled_configs.py b/tests/functional/configs/test_disabled_configs.py index d2ee83e801a..f0176788777 100644 --- a/tests/functional/configs/test_disabled_configs.py +++ b/tests/functional/configs/test_disabled_configs.py @@ -88,3 +88,47 @@ def test_conditional_model(self, project): assert len(results) == 2 results = run_dbt(["test"]) assert len(results) == 5 + + +my_analysis_sql = """ +{{ + config(enabled=False) +}} +select 1 as id +""" + + +schema_yml = """ +models: + - name: my_analysis + description: "A Sample model" + config: + meta: + owner: Joe + +analyses: + - name: my_analysis + description: "A sample analysis" + config: + enabled: false +""" + + +class TestDisabledConfigsSameName: + @pytest.fixture(scope="class") + def models(self): + return { + "my_analysis.sql": my_analysis_sql, + "schema.yml": schema_yml, + } + + @pytest.fixture(scope="class") + def analyses(self): + return { + "my_analysis.sql": my_analysis_sql, + } + + def test_disabled_analysis(self, project): + manifest = run_dbt(["parse"]) + assert len(manifest.disabled) == 2 + assert len(manifest.nodes) == 0 diff --git a/tests/functional/configs/test_disabled_model.py b/tests/functional/configs/test_disabled_model.py index 23cf8fde1e0..a918067ac15 100644 --- a/tests/functional/configs/test_disabled_model.py +++ b/tests/functional/configs/test_disabled_model.py @@ -1,8 +1,7 @@ import pytest -from dbt.exceptions import CompilationError, ParsingError +from dbt.exceptions import CompilationError, ParsingError, SchemaConfigError from dbt.tests.util import get_manifest, run_dbt -from dbt_common.dataclass_schema import ValidationError from tests.functional.configs.fixtures import ( my_model, my_model_2, @@ -394,8 +393,8 @@ def models(self): "my_model.sql": my_model, } - def test_invalis_config(self, project): - with pytest.raises(ValidationError) as exc: + def test_invalid_config(self, project): + with pytest.raises(SchemaConfigError) as exc: run_dbt(["parse"]) exc_str = " ".join(str(exc.value).split()) # flatten all whitespace expected_msg = "'True and False' is not of type 'boolean'" diff --git a/tests/functional/conftest.py b/tests/functional/conftest.py new file mode 100644 index 00000000000..1e5ab4fa9f8 --- /dev/null +++ b/tests/functional/conftest.py @@ -0,0 +1,4 @@ +from tests.functional.fixtures.happy_path_fixture import ( # noqa:D + happy_path_project, + happy_path_project_files, +) diff --git a/tests/functional/constraints/fixtures.py b/tests/functional/constraints/fixtures.py new file mode 100644 index 00000000000..de60963bfec --- /dev/null +++ b/tests/functional/constraints/fixtures.py @@ -0,0 +1,115 @@ +model_foreign_key_model_schema_yml = """ +models: + - name: my_model + constraints: + - type: foreign_key + columns: [id] + to: ref('my_model_to') + to_columns: [id] + columns: + - name: id + data_type: integer +""" + + +model_foreign_key_source_schema_yml = """ +sources: + - name: test_source + tables: + - name: test_table + +models: + - name: my_model + constraints: + - type: foreign_key + columns: [id] + to: source('test_source', 'test_table') + to_columns: [id] + columns: + - name: id + data_type: integer +""" + + +model_foreign_key_model_node_not_found_schema_yml = """ +models: + - name: my_model + constraints: + - type: foreign_key + columns: [id] + to: ref('doesnt_exist') + to_columns: [id] + columns: + - name: id + data_type: integer +""" + + +model_foreign_key_model_invalid_syntax_schema_yml = """ +models: + - name: my_model + constraints: + - type: foreign_key + columns: [id] + to: invalid + to_columns: [id] + columns: + - name: id + data_type: integer +""" + + +model_foreign_key_model_column_schema_yml = """ +models: + - name: my_model + columns: + - name: id + data_type: integer + constraints: + - type: foreign_key + to: ref('my_model_to') + to_columns: [id] +""" + + +model_foreign_key_column_invalid_syntax_schema_yml = """ +models: + - name: my_model + columns: + - name: id + data_type: integer + constraints: + - type: foreign_key + to: invalid + to_columns: [id] +""" + + +model_foreign_key_column_node_not_found_schema_yml = """ +models: + - name: my_model + columns: + - name: id + data_type: integer + constraints: + - type: foreign_key + to: ref('doesnt_exist') + to_columns: [id] +""" + +model_column_level_foreign_key_source_schema_yml = """ +sources: + - name: test_source + tables: + - name: test_table + +models: + - name: my_model + columns: + - name: id + data_type: integer + constraints: + - type: foreign_key + to: source('test_source', 'test_table') + to_columns: [id] +""" diff --git a/tests/functional/constraints/test_foreign_key_constraints.py b/tests/functional/constraints/test_foreign_key_constraints.py new file mode 100644 index 00000000000..2c02cfe7ad7 --- /dev/null +++ b/tests/functional/constraints/test_foreign_key_constraints.py @@ -0,0 +1,241 @@ +import pytest + +from dbt.artifacts.resources import RefArgs +from dbt.exceptions import CompilationError, ParsingError +from dbt.tests.util import get_artifact, run_dbt +from dbt_common.contracts.constraints import ( + ColumnLevelConstraint, + ConstraintType, + ModelLevelConstraint, +) +from tests.functional.constraints.fixtures import ( + model_column_level_foreign_key_source_schema_yml, + model_foreign_key_column_invalid_syntax_schema_yml, + model_foreign_key_column_node_not_found_schema_yml, + model_foreign_key_model_column_schema_yml, + model_foreign_key_model_invalid_syntax_schema_yml, + model_foreign_key_model_node_not_found_schema_yml, + model_foreign_key_model_schema_yml, + model_foreign_key_source_schema_yml, +) + + +class TestModelLevelForeignKeyConstraintToRef: + @pytest.fixture(scope="class") + def models(self): + return { + "constraints_schema.yml": model_foreign_key_model_schema_yml, + "my_model.sql": "select 1 as id", + "my_model_to.sql": "select 1 as id", + } + + def test_model_level_fk_to(self, project, unique_schema): + manifest = run_dbt(["parse"]) + node_with_fk_constraint = manifest.nodes["model.test.my_model"] + assert len(node_with_fk_constraint.constraints) == 1 + + parsed_constraint = node_with_fk_constraint.constraints[0] + assert parsed_constraint == ModelLevelConstraint( + type=ConstraintType.foreign_key, + columns=["id"], + to="ref('my_model_to')", + to_columns=["id"], + ) + # Assert column-level constraint source included in node.depends_on + assert node_with_fk_constraint.refs == [RefArgs("my_model_to")] + assert node_with_fk_constraint.depends_on.nodes == ["model.test.my_model_to"] + assert node_with_fk_constraint.sources == [] + + # Assert compilation renders to from 'ref' to relation identifer + run_dbt(["compile"]) + manifest = get_artifact(project.project_root, "target", "manifest.json") + assert len(manifest["nodes"]["model.test.my_model"]["constraints"]) == 1 + + compiled_constraint = manifest["nodes"]["model.test.my_model"]["constraints"][0] + assert compiled_constraint["to"] == f'"dbt"."{unique_schema}"."my_model_to"' + # Other constraint fields should remain as parsed + assert compiled_constraint["to_columns"] == parsed_constraint.to_columns + assert compiled_constraint["columns"] == parsed_constraint.columns + assert compiled_constraint["type"] == parsed_constraint.type + + +class TestModelLevelForeignKeyConstraintToSource: + @pytest.fixture(scope="class") + def models(self): + return { + "constraints_schema.yml": model_foreign_key_source_schema_yml, + "my_model.sql": "select 1 as id", + "my_model_to.sql": "select 1 as id", + } + + def test_model_level_fk_to(self, project, unique_schema): + manifest = run_dbt(["parse"]) + node_with_fk_constraint = manifest.nodes["model.test.my_model"] + assert len(node_with_fk_constraint.constraints) == 1 + + parsed_constraint = node_with_fk_constraint.constraints[0] + assert parsed_constraint == ModelLevelConstraint( + type=ConstraintType.foreign_key, + columns=["id"], + to="source('test_source', 'test_table')", + to_columns=["id"], + ) + # Assert column-level constraint source included in node.depends_on + assert node_with_fk_constraint.refs == [] + assert node_with_fk_constraint.depends_on.nodes == ["source.test.test_source.test_table"] + assert node_with_fk_constraint.sources == [["test_source", "test_table"]] + + # Assert compilation renders to from 'ref' to relation identifer + run_dbt(["compile"]) + manifest = get_artifact(project.project_root, "target", "manifest.json") + assert len(manifest["nodes"]["model.test.my_model"]["constraints"]) == 1 + + compiled_constraint = manifest["nodes"]["model.test.my_model"]["constraints"][0] + assert compiled_constraint["to"] == '"dbt"."test_source"."test_table"' + # Other constraint fields should remain as parsed + assert compiled_constraint["to_columns"] == parsed_constraint.to_columns + assert compiled_constraint["columns"] == parsed_constraint.columns + assert compiled_constraint["type"] == parsed_constraint.type + + +class TestModelLevelForeignKeyConstraintRefNotFoundError: + @pytest.fixture(scope="class") + def models(self): + return { + "constraints_schema.yml": model_foreign_key_model_node_not_found_schema_yml, + "my_model.sql": "select 1 as id", + "my_model_to.sql": "select 1 as id", + } + + def test_model_level_fk_to_doesnt_exist(self, project): + with pytest.raises( + CompilationError, match="depends on a node named 'doesnt_exist' which was not found" + ): + run_dbt(["parse"]) + + +class TestModelLevelForeignKeyConstraintRefSyntaxError: + @pytest.fixture(scope="class") + def models(self): + return { + "constraints_schema.yml": model_foreign_key_model_invalid_syntax_schema_yml, + "my_model.sql": "select 1 as id", + "my_model_to.sql": "select 1 as id", + } + + def test_model_level_fk_to(self, project): + with pytest.raises( + ParsingError, + match="Invalid 'ref' or 'source' syntax on foreign key constraint 'to' on model my_model: invalid", + ): + run_dbt(["parse"]) + + +class TestColumnLevelForeignKeyConstraintToRef: + @pytest.fixture(scope="class") + def models(self): + return { + "constraints_schema.yml": model_foreign_key_model_column_schema_yml, + "my_model.sql": "select 1 as id", + "my_model_to.sql": "select 1 as id", + } + + def test_column_level_fk_to(self, project, unique_schema): + manifest = run_dbt(["parse"]) + node_with_fk_constraint = manifest.nodes["model.test.my_model"] + assert len(node_with_fk_constraint.columns["id"].constraints) == 1 + + parsed_constraint = node_with_fk_constraint.columns["id"].constraints[0] + # Assert column-level constraint parsed + assert parsed_constraint == ColumnLevelConstraint( + type=ConstraintType.foreign_key, to="ref('my_model_to')", to_columns=["id"] + ) + # Assert column-level constraint ref included in node.depends_on + assert node_with_fk_constraint.refs == [RefArgs(name="my_model_to")] + assert node_with_fk_constraint.sources == [] + assert node_with_fk_constraint.depends_on.nodes == ["model.test.my_model_to"] + + # Assert compilation renders to from 'ref' to relation identifer + run_dbt(["compile"]) + manifest = get_artifact(project.project_root, "target", "manifest.json") + assert len(manifest["nodes"]["model.test.my_model"]["columns"]["id"]["constraints"]) == 1 + + compiled_constraint = manifest["nodes"]["model.test.my_model"]["columns"]["id"][ + "constraints" + ][0] + assert compiled_constraint["to"] == f'"dbt"."{unique_schema}"."my_model_to"' + # Other constraint fields should remain as parsed + assert compiled_constraint["to_columns"] == parsed_constraint.to_columns + assert compiled_constraint["type"] == parsed_constraint.type + + +class TestColumnLevelForeignKeyConstraintToSource: + @pytest.fixture(scope="class") + def models(self): + return { + "constraints_schema.yml": model_column_level_foreign_key_source_schema_yml, + "my_model.sql": "select 1 as id", + "my_model_to.sql": "select 1 as id", + } + + def test_model_level_fk_to(self, project, unique_schema): + manifest = run_dbt(["parse"]) + node_with_fk_constraint = manifest.nodes["model.test.my_model"] + assert len(node_with_fk_constraint.columns["id"].constraints) == 1 + + parsed_constraint = node_with_fk_constraint.columns["id"].constraints[0] + assert parsed_constraint == ColumnLevelConstraint( + type=ConstraintType.foreign_key, + to="source('test_source', 'test_table')", + to_columns=["id"], + ) + # Assert column-level constraint source included in node.depends_on + assert node_with_fk_constraint.refs == [] + assert node_with_fk_constraint.depends_on.nodes == ["source.test.test_source.test_table"] + assert node_with_fk_constraint.sources == [["test_source", "test_table"]] + + # Assert compilation renders to from 'ref' to relation identifer + run_dbt(["compile"]) + manifest = get_artifact(project.project_root, "target", "manifest.json") + assert len(manifest["nodes"]["model.test.my_model"]["columns"]["id"]["constraints"]) == 1 + + compiled_constraint = manifest["nodes"]["model.test.my_model"]["columns"]["id"][ + "constraints" + ][0] + assert compiled_constraint["to"] == '"dbt"."test_source"."test_table"' + # # Other constraint fields should remain as parsed + assert compiled_constraint["to_columns"] == parsed_constraint.to_columns + assert compiled_constraint["type"] == parsed_constraint.type + + +class TestColumnLevelForeignKeyConstraintRefNotFoundError: + @pytest.fixture(scope="class") + def models(self): + return { + "constraints_schema.yml": model_foreign_key_column_node_not_found_schema_yml, + "my_model.sql": "select 1 as id", + "my_model_to.sql": "select 1 as id", + } + + def test_model_level_fk_to_doesnt_exist(self, project): + with pytest.raises( + CompilationError, match="depends on a node named 'doesnt_exist' which was not found" + ): + run_dbt(["parse"]) + + +class TestColumnLevelForeignKeyConstraintRefSyntaxError: + @pytest.fixture(scope="class") + def models(self): + return { + "constraints_schema.yml": model_foreign_key_column_invalid_syntax_schema_yml, + "my_model.sql": "select 1 as id", + "my_model_to.sql": "select 1 as id", + } + + def test_model_level_fk_to(self, project): + with pytest.raises( + ParsingError, + match="Invalid 'ref' or 'source' syntax on foreign key constraint 'to' on model my_model: invalid.", + ): + run_dbt(["parse"]) diff --git a/tests/functional/context_methods/test_env_vars.py b/tests/functional/context_methods/test_env_vars.py index 30c56551c09..d1441dcb7e2 100644 --- a/tests/functional/context_methods/test_env_vars.py +++ b/tests/functional/context_methods/test_env_vars.py @@ -193,3 +193,36 @@ def test_env_vars_secrets(self, project): assert not ("secret_variable" in log_output) assert "regular_variable" in log_output del os.environ["DBT_DEBUG"] + + +class TestEnvVarInCreateSchema: + """Test that the env_var() method works in overrides of the create_schema + macro, which is called during a different phase of execution than most + macros, causing problems.""" + + @pytest.fixture(scope="class", autouse=True) + def setup(self): + os.environ["DBT_TEST_ENV_VAR"] = "1" + + @pytest.fixture(scope="class") + def macros(self): + return { + "macros.sql": """ + {% macro create_schema(relation) %} + {%- call statement('create_schema') -%} + SELECT {{ env_var('DBT_TEST_ENV_VAR') }} as TEST + {% endcall %} + {% endmacro %}% + """ + } + + @pytest.fixture(scope="class") + def models(self): + return { + "mymodel.sql": """ + SELECT 1 as TEST -- {%- do adapter.create_schema(this) -%} + """ + } + + def test_env_var_in_create_schema(self, project): + run_dbt(["run"]) diff --git a/tests/functional/data_test_patch/fixtures.py b/tests/functional/data_test_patch/fixtures.py new file mode 100644 index 00000000000..be056f32680 --- /dev/null +++ b/tests/functional/data_test_patch/fixtures.py @@ -0,0 +1,38 @@ +tests__my_singular_test_sql = """ +with my_cte as ( + select 1 as id, 'foo' as name + union all + select 2 as id, 'bar' as name +) +select * from my_cte +""" + +tests__schema_yml = """ +data_tests: + - name: my_singular_test + description: "{{ doc('my_singular_test_documentation') }}" + config: + error_if: ">10" + meta: + some_key: some_val +""" + +tests__doc_block_md = """ +{% docs my_singular_test_documentation %} + +Some docs from a doc block + +{% enddocs %} +""" + +tests__invalid_name_schema_yml = """ +data_tests: + - name: my_double_test + description: documentation, but make it double +""" + +tests__malformed_schema_yml = """ +data_tests: ¬_null + - not_null: + where: some_condition +""" diff --git a/tests/functional/data_test_patch/test_singular_test_patch.py b/tests/functional/data_test_patch/test_singular_test_patch.py new file mode 100644 index 00000000000..df359c5e645 --- /dev/null +++ b/tests/functional/data_test_patch/test_singular_test_patch.py @@ -0,0 +1,65 @@ +from pathlib import Path + +import pytest + +from dbt.tests.util import get_artifact, run_dbt, run_dbt_and_capture +from tests.functional.data_test_patch.fixtures import ( + tests__doc_block_md, + tests__invalid_name_schema_yml, + tests__malformed_schema_yml, + tests__my_singular_test_sql, + tests__schema_yml, +) + + +class TestPatchSingularTest: + @pytest.fixture(scope="class") + def tests(self): + return { + "my_singular_test.sql": tests__my_singular_test_sql, + "schema.yml": tests__schema_yml, + "doc_block.md": tests__doc_block_md, + } + + def test_compile(self, project): + run_dbt(["compile"]) + manifest = get_artifact(project.project_root, "target", "manifest.json") + assert len(manifest["nodes"]) == 1 + + my_singular_test_node = manifest["nodes"]["test.test.my_singular_test"] + assert my_singular_test_node["description"] == "Some docs from a doc block" + assert my_singular_test_node["config"]["error_if"] == ">10" + assert my_singular_test_node["config"]["meta"] == {"some_key": "some_val"} + + +class TestPatchSingularTestInvalidName: + @pytest.fixture(scope="class") + def tests(self): + return { + "my_singular_test.sql": tests__my_singular_test_sql, + "schema_with_invalid_name.yml": tests__invalid_name_schema_yml, + } + + def test_compile(self, project): + _, log_output = run_dbt_and_capture(["compile"]) + + file_path = Path("tests/schema_with_invalid_name.yml") + assert ( + f"Did not find matching node for patch with name 'my_double_test' in the 'data_tests' section of file '{file_path}'" + in log_output + ) + + +class TestPatchSingularTestMalformedYaml: + @pytest.fixture(scope="class") + def tests(self): + return { + "my_singular_test.sql": tests__my_singular_test_sql, + "schema.yml": tests__malformed_schema_yml, + } + + def test_compile(self, project): + _, log_output = run_dbt_and_capture(["compile"]) + file_path = Path("tests/schema.yml") + assert f"Unable to parse 'data_tests' section of file '{file_path}'" in log_output + assert "Entry did not contain a name" in log_output diff --git a/tests/functional/dbt_runner/test_dbt_runner.py b/tests/functional/dbt_runner/test_dbt_runner.py index 0b1607a2eba..b2e52d1237b 100644 --- a/tests/functional/dbt_runner/test_dbt_runner.py +++ b/tests/functional/dbt_runner/test_dbt_runner.py @@ -1,3 +1,4 @@ +import os from unittest import mock import pytest @@ -103,6 +104,21 @@ def test_pass_in_args_variable(self, dbt): dbt.invoke(args) assert args == args_before + def test_directory_does_not_change(self, project, dbt: dbtRunner) -> None: + project_dir = os.getcwd() # The directory where dbt_project.yml exists. + os.chdir("../") + cmd_execution_dir = os.getcwd() # The directory where dbt command will be run + + commands = ["init", "deps", "clean"] + for command in commands: + args = [command, "--project-dir", project_dir] + if command == "init": + args.append("--skip-profile-setup") + res = dbt.invoke(args) + after_dir = os.getcwd() + assert res.success is True + assert cmd_execution_dir == after_dir + class TestDbtRunnerQueryComments: @pytest.fixture(scope="class") diff --git a/tests/functional/defer_state/fixtures.py b/tests/functional/defer_state/fixtures.py index 832bf258f56..28f4fc898a3 100644 --- a/tests/functional/defer_state/fixtures.py +++ b/tests/functional/defer_state/fixtures.py @@ -540,3 +540,118 @@ metricflow_time_spine_sql = """ SELECT to_date('02/20/2023', 'mm/dd/yyyy') as date_day """ + + +model_with_env_var_in_config_sql = """ +{{ config(materialized=env_var('DBT_TEST_STATE_MODIFIED')) }} +select 1 as id +""" + +model_with_no_in_config_sql = """ +select 1 as id +""" + + +schema_model_with_env_var_in_config_yml = """ +models: + - name: model + config: + materialized: "{{ env_var('DBT_TEST_STATE_MODIFIED') }}" + +""" + + +model_with_var_in_config_sql = """ +{{ config(materialized=var('DBT_TEST_STATE_MODIFIED')) }} +select 1 as id +""" + +model_with_jinja_in_config_sql = """ +{{ config( + materialized = ('table' if execute else 'view') +) }} + +select 1 as id +""" + +model_with_updated_jinja_in_config_sql = """ +{{ config( + materialized = ('view' if execute else 'table') +) }} + +select 1 as id +""" + +schema_model_with_jinja_in_config_yml = """ +models: + - name: model + config: + materialized: "{{ ('table' if execute else 'view') }}" +""" + +schema_model_with_updated_jinja_in_config_yml = """ +models: + - name: model + config: + materialized: "{{ ('view' if execute else 'table') }}" +""" + +schema_source_with_env_var_as_database_property_yml = """ +sources: + - name: jaffle_shop + database: "{{ env_var('DBT_TEST_STATE_MODIFIED') }}" + tables: + - name: customers +""" + +schema_source_with_env_var_as_schema_property_yml = """ +sources: + - name: jaffle_shop + database: "test" + schema: "{{ env_var('DBT_TEST_STATE_MODIFIED') }}" + tables: + - name: customers +""" + +schema_source_with_updated_env_var_as_schema_property_yml = """ +sources: + - name: jaffle_shop + database: "test" + schema: "updated" + tables: + - name: customers +""" + +schema_source_with_jinja_as_database_property_yml = """ +sources: + - name: jaffle_shop + database: "{{ ('foo' if execute else 'bar') }}" + tables: + - name: customers +""" + +schema_source_with_updated_jinja_as_database_property_yml = """ +sources: + - name: jaffle_shop + database: "{{ ('bar' if execute else 'foo') }}" + tables: + - name: customers +""" + +schema_source_with_jinja_as_schema_property_yml = """ +sources: + - name: jaffle_shop + database: "test" + schema: "{{ ('foo' if execute else 'bar') }}" + tables: + - name: customers +""" + +schema_source_with_updated_jinja_as_schema_property_yml = """ +sources: + - name: jaffle_shop + database: "test" + schema: "{{ ('bar' if execute else 'foo') }}" + tables: + - name: customers +""" diff --git a/tests/functional/defer_state/test_modified_state_environment_vars.py b/tests/functional/defer_state/test_modified_state_environment_vars.py new file mode 100644 index 00000000000..569a00e5b10 --- /dev/null +++ b/tests/functional/defer_state/test_modified_state_environment_vars.py @@ -0,0 +1,108 @@ +import os + +import pytest + +from dbt.tests.util import run_dbt +from tests.functional.defer_state.fixtures import ( + model_with_env_var_in_config_sql, + model_with_no_in_config_sql, + schema_model_with_env_var_in_config_yml, +) +from tests.functional.defer_state.test_modified_state import BaseModifiedState + + +class BaseTestStateSelectionEnvVarConfig(BaseModifiedState): + @pytest.fixture(scope="class", autouse=True) + def setup(self): + os.environ["DBT_TEST_STATE_MODIFIED"] = "table" + yield + del os.environ["DBT_TEST_STATE_MODIFIED"] + + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "flags": { + "state_modified_compare_more_unrendered_values": True, + } + } + + def test_change_env_var(self, project): + # Generate ./state without changing environment variable value + run_dbt(["run"]) + self.copy_state() + + # Assert no false positive + results = run_dbt(["list", "-s", "state:modified", "--state", "./state"]) + assert len(results) == 0 + + # Change environment variable and assert no false positive + # Environment variables do not have an effect on state:modified + os.environ["DBT_TEST_STATE_MODIFIED"] = "view" + results = run_dbt(["list", "-s", "state:modified", "--state", "./state"]) + assert len(results) == 0 + + +class TestModelNodeWithEnvVarConfigInSqlFile(BaseTestStateSelectionEnvVarConfig): + @pytest.fixture(scope="class") + def models(self): + return { + "model.sql": model_with_env_var_in_config_sql, + } + + +class TestModelNodeWithEnvVarConfigInSchemaYml(BaseTestStateSelectionEnvVarConfig): + @pytest.fixture(scope="class") + def models(self): + return { + "model.sql": model_with_no_in_config_sql, + "schema.yml": schema_model_with_env_var_in_config_yml, + } + + +class TestModelNodeWithEnvVarConfigInProjectYml(BaseTestStateSelectionEnvVarConfig): + @pytest.fixture(scope="class") + def models(self): + return { + "model.sql": model_with_no_in_config_sql, + } + + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "models": { + "test": { + "+materialized": "{{ env_var('DBT_TEST_STATE_MODIFIED') }}", + } + } + } + + +class TestModelNodeWithEnvVarConfigInProjectYmlAndSchemaYml(BaseTestStateSelectionEnvVarConfig): + @pytest.fixture(scope="class") + def models(self): + return { + "model.sql": model_with_no_in_config_sql, + "schema.yml": schema_model_with_env_var_in_config_yml, + } + + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "flags": { + "state_modified_compare_more_unrendered_values": True, + }, + "models": { + "test": { + "+materialized": "{{ env_var('DBT_TEST_STATE_MODIFIED') }}", + } + }, + } + + +class TestModelNodeWithEnvVarConfigInSqlAndSchemaYml(BaseTestStateSelectionEnvVarConfig): + @pytest.fixture(scope="class") + def models(self): + return { + "model.sql": model_with_env_var_in_config_sql, + "schema.yml": schema_model_with_env_var_in_config_yml, + } diff --git a/tests/functional/defer_state/test_modified_state_jinja.py b/tests/functional/defer_state/test_modified_state_jinja.py new file mode 100644 index 00000000000..2489eeedde6 --- /dev/null +++ b/tests/functional/defer_state/test_modified_state_jinja.py @@ -0,0 +1,144 @@ +import pytest + +from dbt.tests.util import ( + get_artifact, + get_project_config, + run_dbt, + update_config_file, + write_file, +) +from tests.functional.defer_state.fixtures import ( + model_with_jinja_in_config_sql, + model_with_no_in_config_sql, + model_with_updated_jinja_in_config_sql, + schema_model_with_jinja_in_config_yml, + schema_model_with_updated_jinja_in_config_yml, +) +from tests.functional.defer_state.test_modified_state import BaseModifiedState + + +class BaseTestStateSelectionJinjaInConfig(BaseModifiedState): + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "flags": { + "state_modified_compare_more_unrendered_values": True, + } + } + + def update_jinja_expression_in_config(self, project): + pass + + def test_change_jinja_if(self, project): + run_dbt(["run"]) + self.copy_state() + # Model is table when execute = True + manifest_json = get_artifact(project.project_root, "target", "manifest.json") + assert manifest_json["nodes"]["model.test.model"]["config"]["materialized"] == "view" + + # Assert no false positive (execute = False) + results = run_dbt(["list", "-s", "state:modified", "--state", "./state"]) + assert len(results) == 0 + + # Update unrendered config (change jinja expression) + self.update_jinja_expression_in_config(project) + # Assert no false negatives (jinja expression has changed) + results = run_dbt(["list", "-s", "state:modified", "--state", "./state"]) + assert len(results) == 1 + + +class TestModelNodeWithJinjaConfigInSqlFile(BaseTestStateSelectionJinjaInConfig): + def update_jinja_expression_in_config(self, project): + write_file( + model_with_updated_jinja_in_config_sql, project.project_root, "models", "model.sql" + ) + + @pytest.fixture(scope="class") + def models(self): + return { + "model.sql": model_with_jinja_in_config_sql, + } + + +class TestModelNodeWithEnvVarConfigInSchemaYml(BaseTestStateSelectionJinjaInConfig): + def update_jinja_expression_in_config(self, project): + write_file( + schema_model_with_updated_jinja_in_config_yml, + project.project_root, + "models", + "schema.yml", + ) + + @pytest.fixture(scope="class") + def models(self): + return { + "model.sql": model_with_no_in_config_sql, + "schema.yml": schema_model_with_jinja_in_config_yml, + } + + +class TestModelNodeWithJinjaConfigInProjectYml(BaseTestStateSelectionJinjaInConfig): + def update_jinja_expression_in_config(self, project): + config = get_project_config(project) + config["models"]["test"]["+materialized"] = "{{ ('view' if execute else 'table') }}" + update_config_file(config, "dbt_project.yml") + + @pytest.fixture(scope="class") + def models(self): + return { + "model.sql": model_with_no_in_config_sql, + } + + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "models": { + "test": { + "+materialized": "{{ ('table' if execute else 'view') }}", + } + } + } + + +class TestModelNodeWithJinjaConfigInProjectYmlAndSchemaYml(BaseTestStateSelectionJinjaInConfig): + def update_jinja_expression_in_config(self, project): + write_file( + schema_model_with_updated_jinja_in_config_yml, + project.project_root, + "models", + "schema.yml", + ) + + @pytest.fixture(scope="class") + def models(self): + return { + "model.sql": model_with_no_in_config_sql, + "schema.yml": schema_model_with_jinja_in_config_yml, + } + + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "flags": { + "state_modified_compare_more_unrendered_values": True, + }, + "models": { + "test": { + "+materialized": "{{ ('view' if execute else 'table') }}", + } + }, + } + + +class TestModelNodeWithJinjaConfigInSqlAndSchemaYml(BaseTestStateSelectionJinjaInConfig): + def update_jinja_expression_in_config(self, project): + write_file( + model_with_updated_jinja_in_config_sql, project.project_root, "models", "model.sql" + ) + + @pytest.fixture(scope="class") + def models(self): + return { + "model.sql": model_with_jinja_in_config_sql, + "schema.yml": schema_model_with_jinja_in_config_yml, + } diff --git a/tests/functional/defer_state/test_modified_state_sources_unrendered.py b/tests/functional/defer_state/test_modified_state_sources_unrendered.py new file mode 100644 index 00000000000..e2e2f68db1b --- /dev/null +++ b/tests/functional/defer_state/test_modified_state_sources_unrendered.py @@ -0,0 +1,163 @@ +import os + +import pytest + +from dbt.tests.util import get_artifact, run_dbt, write_file +from tests.functional.defer_state.fixtures import ( + schema_source_with_env_var_as_database_property_yml, + schema_source_with_env_var_as_schema_property_yml, + schema_source_with_jinja_as_database_property_yml, + schema_source_with_jinja_as_schema_property_yml, + schema_source_with_updated_env_var_as_schema_property_yml, + schema_source_with_updated_jinja_as_database_property_yml, + schema_source_with_updated_jinja_as_schema_property_yml, +) +from tests.functional.defer_state.test_modified_state import BaseModifiedState +from tests.functional.defer_state.test_modified_state_environment_vars import ( + BaseTestStateSelectionEnvVarConfig, +) + + +class TestSourceNodeWithEnvVarConfigInDatabase(BaseTestStateSelectionEnvVarConfig): + @pytest.fixture(scope="class") + def models(self): + return { + "schema.yml": schema_source_with_env_var_as_database_property_yml, + } + + +class TestSourceNodeWithEnvVarConfigInSchema(BaseTestStateSelectionEnvVarConfig): + @pytest.fixture(scope="class") + def models(self): + return { + "schema.yml": schema_source_with_env_var_as_schema_property_yml, + } + + def test_change_env_var(self, project): + # Generate ./state without changing environment variable value + run_dbt(["run"]) + self.copy_state() + + manifest_json = get_artifact(project.project_root, "target", "manifest.json") + assert manifest_json["sources"]["source.test.jaffle_shop.customers"]["schema"] == "table" + assert ( + manifest_json["sources"]["source.test.jaffle_shop.customers"]["unrendered_schema"] + == "{{ env_var('DBT_TEST_STATE_MODIFIED') }}" + ) + + # Assert no false positive + results = run_dbt(["list", "-s", "state:modified", "--state", "./state"]) + assert len(results) == 0 + + # Change environment variable and assert no false positive + # Environment variables do not have an effect on state:modified + os.environ["DBT_TEST_STATE_MODIFIED"] = "view" + results = run_dbt(["list", "-s", "state:modified", "--state", "./state"]) + assert len(results) == 0 + + # Confirm env change changed schema, but not unrendered_schema + manifest_json = get_artifact(project.project_root, "target", "manifest.json") + assert manifest_json["sources"]["source.test.jaffle_shop.customers"]["schema"] == "view" + assert ( + manifest_json["sources"]["source.test.jaffle_shop.customers"]["unrendered_schema"] + == "{{ env_var('DBT_TEST_STATE_MODIFIED') }}" + ) + + # Assert no false negative after actual change to schema + write_file( + schema_source_with_updated_env_var_as_schema_property_yml, + project.project_root, + "models", + "schema.yml", + ) + results = run_dbt(["list", "-s", "state:modified", "--state", "./state"]) + assert len(results) == 1 + + manifest_json = get_artifact(project.project_root, "target", "manifest.json") + assert manifest_json["sources"]["source.test.jaffle_shop.customers"]["schema"] == "updated" + assert ( + manifest_json["sources"]["source.test.jaffle_shop.customers"]["unrendered_schema"] + == "updated" + ) + + +class TestSourceNodeWithJinjaInDatabase(BaseModifiedState): + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "flags": { + "state_modified_compare_more_unrendered_values": True, + } + } + + def update_jinja_expression_in_config(self, project): + write_file( + schema_source_with_updated_jinja_as_database_property_yml, + project.project_root, + "models", + "schema.yml", + ) + + @pytest.fixture(scope="class") + def models(self): + return { + "schema.yml": schema_source_with_jinja_as_database_property_yml, + } + + def test_change_jinja_if(self, project): + run_dbt(["run"]) + self.copy_state() + # source database is 'bar' when execute = False + manifest_json = get_artifact(project.project_root, "target", "manifest.json") + assert manifest_json["sources"]["source.test.jaffle_shop.customers"]["database"] == "bar" + assert ( + manifest_json["sources"]["source.test.jaffle_shop.customers"]["unrendered_database"] + == "{{ ('foo' if execute else 'bar') }}" + ) + + # Assert no false positive (execute = False) + results = run_dbt(["list", "-s", "state:modified", "--state", "./state"]) + assert len(results) == 0 + + # Update unrendered config (change jinja expression) + self.update_jinja_expression_in_config(project) + # Assert no false negatives (jinja expression has changed) + results = run_dbt(["list", "-s", "state:modified", "--state", "./state"]) + assert len(results) == 1 + + +class TestSourceNodeWithJinjaInSchema(BaseModifiedState): + def update_jinja_expression_in_config(self, project): + write_file( + schema_source_with_updated_jinja_as_schema_property_yml, + project.project_root, + "models", + "schema.yml", + ) + + @pytest.fixture(scope="class") + def models(self): + return { + "schema.yml": schema_source_with_jinja_as_schema_property_yml, + } + + def test_change_jinja_if(self, project): + run_dbt(["run"]) + self.copy_state() + # source database is 'bar' when execute = False + manifest_json = get_artifact(project.project_root, "target", "manifest.json") + assert manifest_json["sources"]["source.test.jaffle_shop.customers"]["schema"] == "bar" + assert ( + manifest_json["sources"]["source.test.jaffle_shop.customers"]["unrendered_schema"] + == "{{ ('foo' if execute else 'bar') }}" + ) + + # Assert no false positive (execute = False) + results = run_dbt(["list", "-s", "state:modified", "--state", "./state"]) + assert len(results) == 0 + + # Update unrendered config (change jinja expression) + self.update_jinja_expression_in_config(project) + # Assert no false negatives (jinja expression has changed) + results = run_dbt(["list", "-s", "state:modified", "--state", "./state"]) + assert len(results) == 1 diff --git a/tests/functional/defer_state/test_modified_state_vars.py b/tests/functional/defer_state/test_modified_state_vars.py new file mode 100644 index 00000000000..01068f0130f --- /dev/null +++ b/tests/functional/defer_state/test_modified_state_vars.py @@ -0,0 +1,103 @@ +import pytest + +from dbt.tests.util import run_dbt +from tests.functional.defer_state.fixtures import model_with_var_in_config_sql +from tests.functional.defer_state.test_modified_state import BaseModifiedState + + +class TestStateSelectionVarConfigLegacy(BaseModifiedState): + @pytest.fixture(scope="class") + def models(self): + return { + "model.sql": model_with_var_in_config_sql, + } + + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "flags": { + "state_modified_compare_more_unrendered_values": False, + } + } + + def test_change_var(self, project): + # Generate ./state without changing variable value + run_dbt(["run", "--vars", "DBT_TEST_STATE_MODIFIED: view"]) + self.copy_state() + + # Assert no false positive + results = run_dbt( + [ + "list", + "-s", + "state:modified", + "--state", + "./state", + "--vars", + "DBT_TEST_STATE_MODIFIED: view", + ] + ) + assert len(results) == 0 + + # Change var and assert no false negative - legacy behaviour + results = run_dbt( + [ + "list", + "-s", + "state:modified", + "--state", + "./state", + "--vars", + "DBT_TEST_STATE_MODIFIED: table", + ] + ) + assert len(results) == 1 + + +class TestStateSelectionVarConfig(BaseModifiedState): + @pytest.fixture(scope="class") + def models(self): + return { + "model.sql": model_with_var_in_config_sql, + } + + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "flags": { + "state_modified_compare_more_unrendered_values": True, + } + } + + def test_change_var(self, project): + # Generate ./state without changing variable value + run_dbt(["run", "--vars", "DBT_TEST_STATE_MODIFIED: view"]) + self.copy_state() + + # Assert no false positive + results = run_dbt( + [ + "list", + "-s", + "state:modified", + "--state", + "./state", + "--vars", + "DBT_TEST_STATE_MODIFIED: view", + ] + ) + assert len(results) == 0 + + # Change var and assert no sensitivity to var changes -- new behaviour until state:modified.vars included in state:modified by default + results = run_dbt( + [ + "list", + "-s", + "state:modified", + "--state", + "./state", + "--vars", + "DBT_TEST_STATE_MODIFIED: table", + ] + ) + assert len(results) == 0 diff --git a/tests/functional/dependencies/test_dependency_options.py b/tests/functional/dependencies/test_dependency_options.py index bf176831aaa..66fcc96109a 100644 --- a/tests/functional/dependencies/test_dependency_options.py +++ b/tests/functional/dependencies/test_dependency_options.py @@ -33,16 +33,14 @@ def test_deps_lock(self, clean_start): assert os.path.exists("package-lock.yml") with open("package-lock.yml") as fp: contents = fp.read() - assert ( - contents - == """packages: - - package: fivetran/fivetran_utils - version: 0.4.7 - - package: dbt-labs/dbt_utils - version: 1.2.0 -sha1_hash: 71304bca2138cf8004070b3573a1e17183c0c1a8 -""" - ) + + fivetran_package = "- package: fivetran/fivetran_utils\n version: 0.4.7" + # dbt-utils is a dep in fivetran so we can't check for a specific version or this test fails everytime a new dbt-utils version comes out + dbt_labs_package = "- package: dbt-labs/dbt_utils" + package_sha = "sha1_hash: 71304bca2138cf8004070b3573a1e17183c0c1a8" + assert fivetran_package in contents + assert dbt_labs_package in contents + assert package_sha in contents def test_deps_default(self, clean_start): run_dbt(["deps"]) @@ -50,16 +48,13 @@ def test_deps_default(self, clean_start): assert os.path.exists("package-lock.yml") with open("package-lock.yml") as fp: contents = fp.read() - assert ( - contents - == """packages: - - package: fivetran/fivetran_utils - version: 0.4.7 - - package: dbt-labs/dbt_utils - version: 1.2.0 -sha1_hash: 71304bca2138cf8004070b3573a1e17183c0c1a8 -""" - ) + fivetran_package = "- package: fivetran/fivetran_utils\n version: 0.4.7" + # dbt-utils is a dep in fivetran so we can't check for a specific version or this test fails everytime a new dbt-utils version comes out + dbt_labs_package = "- package: dbt-labs/dbt_utils" + package_sha = "sha1_hash: 71304bca2138cf8004070b3573a1e17183c0c1a8" + assert fivetran_package in contents + assert dbt_labs_package in contents + assert package_sha in contents def test_deps_add(self, clean_start): run_dbt(["deps", "--add-package", "dbt-labs/audit_helper@0.9.0"]) diff --git a/tests/functional/dependencies/test_local_dependency.py b/tests/functional/dependencies/test_local_dependency.py index d26345f2da6..e5bf1de5323 100644 --- a/tests/functional/dependencies/test_local_dependency.py +++ b/tests/functional/dependencies/test_local_dependency.py @@ -328,7 +328,7 @@ def test_hook_dependency(self, prepare_dependencies, project): run_dbt(["deps", "--vars", cli_vars]) results = run_dbt(["run", "--vars", cli_vars]) - assert len(results) == 2 + assert len(results) == 8 check_relations_equal(project.adapter, ["actual", "expected"]) diff --git a/tests/functional/deprecations/fixtures.py b/tests/functional/deprecations/fixtures.py index d9ae97603db..85f45d78f47 100644 --- a/tests/functional/deprecations/fixtures.py +++ b/tests/functional/deprecations/fixtures.py @@ -30,6 +30,23 @@ email: something@example.com """ + +deprecated_model_exposure_yaml = """ +version: 2 + +models: + - name: model + deprecation_date: 1999-01-01 00:00:00.00+00:00 + +exposures: + - name: simple_exposure + type: dashboard + depends_on: + - ref('model') + owner: + email: something@example.com +""" + # deprecated test config fixtures data_tests_yaml = """ models: diff --git a/tests/functional/deprecations/test_config_deprecations.py b/tests/functional/deprecations/test_config_deprecations.py index d4e965a8451..077dd7da103 100644 --- a/tests/functional/deprecations/test_config_deprecations.py +++ b/tests/functional/deprecations/test_config_deprecations.py @@ -32,7 +32,7 @@ def test_project_tests_config(self, project): deprecations.reset_deprecations() assert deprecations.active_deprecations == set() run_dbt(["parse"]) - expected = {"project-test-config"} + expected = set() assert expected == deprecations.active_deprecations def test_project_tests_config_fail(self, project): @@ -41,7 +41,7 @@ def test_project_tests_config_fail(self, project): with pytest.raises(CompilationError) as exc: run_dbt(["--warn-error", "--no-partial-parse", "parse"]) exc_str = " ".join(str(exc.value).split()) # flatten all whitespace - expected_msg = "The `tests` config has been renamed to `data_tests`" + expected_msg = "Configuration paths exist in your dbt_project.yml file which do not apply to any resources. There are 1 unused configuration paths: - data_tests" assert expected_msg in exc_str @@ -62,17 +62,13 @@ def test_generic_tests_config(self, project): deprecations.reset_deprecations() assert deprecations.active_deprecations == set() run_dbt(["parse"]) - expected = {"project-test-config"} + expected = set() assert expected == deprecations.active_deprecations def test_generic_tests_fail(self, project): deprecations.reset_deprecations() assert deprecations.active_deprecations == set() - with pytest.raises(CompilationError) as exc: - run_dbt(["--warn-error", "--no-partial-parse", "parse"]) - exc_str = " ".join(str(exc.value).split()) # flatten all whitespace - expected_msg = "The `tests` config has been renamed to `data_tests`" - assert expected_msg in exc_str + run_dbt(["--warn-error", "--no-partial-parse", "parse"]) def test_generic_data_test_parsing(self, project): results = run_dbt(["list", "--resource-type", "test"]) @@ -98,7 +94,7 @@ def test_source_tests_config(self, project): deprecations.reset_deprecations() assert deprecations.active_deprecations == set() run_dbt(["parse"]) - expected = {"project-test-config"} + expected = set() assert expected == deprecations.active_deprecations def test_generic_data_tests(self, project): diff --git a/tests/functional/deprecations/test_deprecations.py b/tests/functional/deprecations/test_deprecations.py index 8082be7b911..2cb51935da4 100644 --- a/tests/functional/deprecations/test_deprecations.py +++ b/tests/functional/deprecations/test_deprecations.py @@ -3,9 +3,11 @@ import dbt_common from dbt import deprecations -from dbt.tests.util import run_dbt, write_file +from dbt.tests.util import run_dbt, run_dbt_and_capture, write_file +from dbt_common.exceptions import EventCompilationError from tests.functional.deprecations.fixtures import ( bad_name_yaml, + deprecated_model_exposure_yaml, models_trivial__model_sql, ) @@ -98,6 +100,18 @@ def test_package_redirect_fail(self, project): assert expected_msg in exc_str +class TestDeprecatedModelExposure: + @pytest.fixture(scope="class") + def models(self): + return { + "model.sql": models_trivial__model_sql, + "exposure.yml": deprecated_model_exposure_yaml, + } + + def test_exposure_with_deprecated_model(self, project): + run_dbt(["parse"]) + + class TestExposureNameDeprecation: @pytest.fixture(scope="class") def models(self): @@ -143,6 +157,45 @@ def models(self): def test_profile_config_deprecation(self, project): deprecations.reset_deprecations() assert deprecations.active_deprecations == set() - run_dbt(["parse"]) - expected = {"project-flags-moved"} - assert expected == deprecations.active_deprecations + + _, logs = run_dbt_and_capture(["parse"]) + + assert ( + "User config should be moved from the 'config' key in profiles.yml to the 'flags' key in dbt_project.yml." + in logs + ) + assert deprecations.active_deprecations == {"project-flags-moved"} + + +class TestProjectFlagsMovedDeprecationQuiet(TestProjectFlagsMovedDeprecation): + def test_profile_config_deprecation(self, project): + deprecations.reset_deprecations() + assert deprecations.active_deprecations == set() + + _, logs = run_dbt_and_capture(["--quiet", "parse"]) + + assert ( + "User config should be moved from the 'config' key in profiles.yml to the 'flags' key in dbt_project.yml." + not in logs + ) + assert deprecations.active_deprecations == {"project-flags-moved"} + + +class TestProjectFlagsMovedDeprecationWarnErrorOptions(TestProjectFlagsMovedDeprecation): + def test_profile_config_deprecation(self, project): + deprecations.reset_deprecations() + with pytest.raises(EventCompilationError): + run_dbt(["--warn-error-options", "{'include': 'all'}", "parse"]) + + with pytest.raises(EventCompilationError): + run_dbt( + ["--warn-error-options", "{'include': ['ProjectFlagsMovedDeprecation']}", "parse"] + ) + + _, logs = run_dbt_and_capture( + ["--warn-error-options", "{'silence': ['ProjectFlagsMovedDeprecation']}", "parse"] + ) + assert ( + "User config should be moved from the 'config' key in profiles.yml to the 'flags' key in dbt_project.yml." + not in logs + ) diff --git a/tests/functional/deprecations/model_deprecations.py b/tests/functional/deprecations/test_model_deprecations.py similarity index 97% rename from tests/functional/deprecations/model_deprecations.py rename to tests/functional/deprecations/test_model_deprecations.py index 03e38b1220e..1318562c10f 100644 --- a/tests/functional/deprecations/model_deprecations.py +++ b/tests/functional/deprecations/test_model_deprecations.py @@ -1,8 +1,8 @@ import pytest from dbt.cli.main import dbtRunner -from dbt.exceptions import EventCompilationError from dbt.tests.util import run_dbt +from dbt_common.exceptions import EventCompilationError deprecated_model__yml = """ version: 2 @@ -52,7 +52,7 @@ def test_deprecation_warning_error_options(self, project): run_dbt(["--warn-error-options", '{"include": ["DeprecatedModel"]}', "parse"]) -class TestUpcomingReferenceDeprecatingWarning: +class TestUpcomingReferenceDeprecationWarning: @pytest.fixture(scope="class") def models(self): return { diff --git a/tests/functional/docs/test_good_docs_blocks.py b/tests/functional/docs/test_good_docs_blocks.py index 768e6201275..e1ed96c5eb7 100644 --- a/tests/functional/docs/test_good_docs_blocks.py +++ b/tests/functional/docs/test_good_docs_blocks.py @@ -91,6 +91,7 @@ def test_valid_doc_ref(self, project): "meta": {}, "quote": None, "tags": [], + "granularity": None, } == model_data["columns"]["id"] assert { @@ -101,6 +102,7 @@ def test_valid_doc_ref(self, project): "meta": {}, "quote": None, "tags": [], + "granularity": None, } == model_data["columns"]["first_name"] assert { @@ -111,6 +113,7 @@ def test_valid_doc_ref(self, project): "meta": {}, "quote": None, "tags": [], + "granularity": None, } == model_data["columns"]["last_name"] assert len(model_data["columns"]) == 3 @@ -152,6 +155,7 @@ def test_alternative_docs_path(self, project): "meta": {}, "quote": None, "tags": [], + "granularity": None, } == model_data["columns"]["id"] assert { @@ -162,6 +166,7 @@ def test_alternative_docs_path(self, project): "meta": {}, "quote": None, "tags": [], + "granularity": None, } == model_data["columns"]["first_name"] assert { @@ -172,6 +177,7 @@ def test_alternative_docs_path(self, project): "meta": {}, "quote": None, "tags": [], + "granularity": None, } == model_data["columns"]["last_name"] assert len(model_data["columns"]) == 3 diff --git a/tests/functional/fixtures/happy_path_project/models/metricflow_time_spine_second.sql b/tests/functional/fixtures/happy_path_project/models/metricflow_time_spine_second.sql new file mode 100644 index 00000000000..656724dbe97 --- /dev/null +++ b/tests/functional/fixtures/happy_path_project/models/metricflow_time_spine_second.sql @@ -0,0 +1,2 @@ +select + {{ dbt.date_trunc('second', dbt.current_timestamp()) }} as ts_second diff --git a/tests/functional/fixtures/happy_path_project/models/schema.yml b/tests/functional/fixtures/happy_path_project/models/schema.yml index ef6addf5b91..3d75adab62d 100644 --- a/tests/functional/fixtures/happy_path_project/models/schema.yml +++ b/tests/functional/fixtures/happy_path_project/models/schema.yml @@ -8,6 +8,16 @@ models: data_tests: - unique - not_null + - name: metricflow_time_spine + description: Day time spine + columns: + - name: date_day + granularity: day + - name: metricflow_time_spine_second + description: Second time spine + columns: + - name: ts_second + granularity: second sources: - name: my_source diff --git a/tests/functional/generic_test_description/fixtures.py b/tests/functional/generic_test_description/fixtures.py new file mode 100644 index 00000000000..2be2f2313ba --- /dev/null +++ b/tests/functional/generic_test_description/fixtures.py @@ -0,0 +1,34 @@ +models__my_model_sql = """ +with my_cte as ( + select 1 as id, 'blue' as color + union all + select 2 as id, 'green' as red + union all + select 3 as id, 'red' as red +) +select * from my_cte +""" + +models__schema_yml = """ +models: + - name: my_model + columns: + - name: id + tests: + - unique: + description: "id must be unique" + - not_null + - name: color + tests: + - accepted_values: + values: ['blue', 'green', 'red'] + description: "{{ doc('color_accepted_values') }}" +""" + +models__doc_block_md = """ +{% docs color_accepted_values %} + +The `color` column must be one of 'blue', 'green', or 'red'. + +{% enddocs %} +""" diff --git a/tests/functional/generic_test_description/test_generic_test_description.py b/tests/functional/generic_test_description/test_generic_test_description.py new file mode 100644 index 00000000000..cf68ff1c33c --- /dev/null +++ b/tests/functional/generic_test_description/test_generic_test_description.py @@ -0,0 +1,32 @@ +import pytest + +from dbt.tests.util import get_artifact, run_dbt +from tests.functional.generic_test_description.fixtures import ( + models__doc_block_md, + models__my_model_sql, + models__schema_yml, +) + + +class TestBuiltinGenericTestDescription: + @pytest.fixture(scope="class") + def models(self): + return { + "my_model.sql": models__my_model_sql, + "schema.yml": models__schema_yml, + "doc_block.md": models__doc_block_md, + } + + def test_compile(self, project): + run_dbt(["compile"]) + manifest = get_artifact(project.project_root, "target", "manifest.json") + assert len(manifest["nodes"]) == 4 + + nodes = {node["alias"]: node for node in manifest["nodes"].values()} + + assert nodes["unique_my_model_id"]["description"] == "id must be unique" + assert nodes["not_null_my_model_id"]["description"] == "" + assert ( + nodes["accepted_values_my_model_color__blue__green__red"]["description"] + == "The `color` column must be one of 'blue', 'green', or 'red'." + ) diff --git a/tests/functional/invalid_model_tests/test_model_logging.py b/tests/functional/invalid_model_tests/test_model_logging.py new file mode 100644 index 00000000000..731a0980e27 --- /dev/null +++ b/tests/functional/invalid_model_tests/test_model_logging.py @@ -0,0 +1,40 @@ +import pytest + +from dbt.tests.util import run_dbt_and_capture + +warnings_sql = """ +{{ config(group='my_group') }} +{% do exceptions.warn('warning: everything is terrible but not that terrible') %} +{{ exceptions.warn("warning: everything is terrible but not that terrible") }} +select 1 as id +""" + +schema_yml = """ +version: 2 +groups: + - name: my_group + owner: + name: group_owner +""" + + +class TestModelLogging: + @pytest.fixture(scope="class") + def models(self): + return { + "warnings.sql": warnings_sql, + "schema.yml": schema_yml, + } + + def test_warn(self, project): + results, log_output = run_dbt_and_capture(["run", "--log-format", "json"]) + log_lines = log_output.split("\n") + + log_lines_with_warning = [line for line in log_lines if "JinjaLogWarning" in line] + assert len(log_lines_with_warning) == 4 + assert all("everything is terrible" in line for line in log_lines_with_warning) + + log_lines_with_group = [line for line in log_lines if "LogModelResult" in line] + assert len(log_lines_with_group) == 1 + assert "group_owner" in log_lines_with_group[0] + assert "my_group" in log_lines_with_group[0] diff --git a/tests/functional/invalid_model_tests/test_model_warning.py b/tests/functional/invalid_model_tests/test_model_warning.py deleted file mode 100644 index 700e0aaa3f0..00000000000 --- a/tests/functional/invalid_model_tests/test_model_warning.py +++ /dev/null @@ -1,18 +0,0 @@ -import pytest - -from dbt.tests.util import run_dbt - -warnings_sql = """ -{% do exceptions.warn('warning: everything is terrible but not that terrible') %} -{{ exceptions.warn("warning: everything is terrible but not that terrible") }} -select 1 as id -""" - - -class TestEmitWarning: - @pytest.fixture(scope="class") - def models(self): - return {"warnings.sql": warnings_sql} - - def test_warn(self, project): - run_dbt(["run"], expect_pass=True) diff --git a/tests/functional/list/test_commands.py b/tests/functional/list/test_commands.py new file mode 100644 index 00000000000..70739e1c4f7 --- /dev/null +++ b/tests/functional/list/test_commands.py @@ -0,0 +1,105 @@ +import shutil + +import pytest + +from dbt.artifacts.resources.types import NodeType +from dbt.cli.main import dbtRunner +from dbt.cli.types import Command +from dbt.events.types import NoNodesSelected +from dbt.tests.util import run_dbt +from tests.utils import EventCatcher + +""" +Testing different commands against the happy path fixture + +The general flow +1. Declare the commands to be tested +2. Write a paramaterized test ensure a given command reaches causes and associated desired state. +""" + +# These are commands we're skipping as they don't make sense or don't work with the +# happy path fixture currently +commands_to_skip = { + "clone", + "generate", + "server", + "init", + "list", + "run-operation", + "show", + "snapshot", + "freshness", +} + +# Commands to happy path test +commands = [command.value for command in Command if command.value not in commands_to_skip] + + +class TestRunCommands: + @pytest.fixture(scope="class", autouse=True) + def drop_snapshots(self, happy_path_project, project_root: str) -> None: + """The snapshots are erroring out, so lets drop them. + + Seems to be database related. Ideally snapshots should work in these tests. It's a bad sign that they don't. That + may have more to do with our fixture setup than the source code though. + + Note: that the `happy_path_fixture_files` are a _class_ based fixture. Thus although this fixture _modifies_ the + files available to the happy path project, it doesn't affect that fixture for tests in other test classes. + """ + + shutil.rmtree(f"{project_root}/snapshots") + + @pytest.mark.parametrize("dbt_command", [(command,) for command in commands]) + def test_run_commmand( + self, + happy_path_project, + dbt_command, + ): + run_dbt([dbt_command]) + + +""" +Testing command interactions with specific node types + +The general flow +1. Declare resource (node) types to be tested +2. Write a parameterized test that ensures commands interact successfully with each resource type +""" + +# TODO: Figure out which of these are just missing from the happy path fixture vs which ones aren't selectable +skipped_resource_types = { + "analysis", + "operation", + "rpc", + "sql_operation", + "doc", + "macro", + "exposure", + "group", + "unit_test", + "fixture", +} +resource_types = [ + node_type.value for node_type in NodeType if node_type.value not in skipped_resource_types +] + + +class TestSelectResourceType: + @pytest.fixture(scope="function") + def catcher(self) -> EventCatcher: + return EventCatcher(event_to_catch=NoNodesSelected) + + @pytest.fixture(scope="function") + def runner(self, catcher: EventCatcher) -> dbtRunner: + return dbtRunner(callbacks=[catcher.catch]) + + @pytest.mark.parametrize("resource_type", resource_types) + def test_select_by_resource_type( + self, + resource_type: str, + happy_path_project, + runner: dbtRunner, + catcher: EventCatcher, + ) -> None: + runner.invoke(["list", "--select", f"resource_type:{resource_type}"]) + assert len(catcher.caught_events) == 0 diff --git a/tests/functional/list/test_list.py b/tests/functional/list/test_list.py index 653021c608b..0775687788c 100644 --- a/tests/functional/list/test_list.py +++ b/tests/functional/list/test_list.py @@ -63,6 +63,13 @@ def expect_snapshot_output(self, happy_path_project): # noqa: F811 "persist_docs": {}, "target_database": happy_path_project.database, "target_schema": happy_path_project.test_schema, + "dbt_valid_to_current": None, + "snapshot_meta_column_names": { + "dbt_scd_id": None, + "dbt_updated_at": None, + "dbt_valid_from": None, + "dbt_valid_to": None, + }, "unique_key": "id", "strategy": "timestamp", "updated_at": "updated_at", @@ -79,6 +86,10 @@ def expect_snapshot_output(self, happy_path_project): # noqa: F811 "incremental_strategy": None, "docs": {"node_color": None, "show": True}, "contract": {"enforced": False, "alias_types": True}, + "event_time": None, + "lookback": 1, + "batch_size": None, + "begin": None, }, "unique_id": "snapshot.test.my_snapshot", "original_file_path": normalize("snapshots/snapshot.sql"), @@ -121,6 +132,10 @@ def expect_analyses_output(self): "incremental_strategy": None, "docs": {"node_color": None, "show": True}, "contract": {"enforced": False, "alias_types": True}, + "event_time": None, + "lookback": 1, + "batch_size": None, + "begin": None, }, "unique_id": "analysis.test.a", "original_file_path": normalize("analyses/a.sql"), @@ -133,12 +148,20 @@ def expect_analyses_output(self): def expect_model_output(self): expectations = { - "name": ("ephemeral", "incremental", "inner", "metricflow_time_spine", "outer"), + "name": ( + "ephemeral", + "incremental", + "inner", + "metricflow_time_spine", + "metricflow_time_spine_second", + "outer", + ), "selector": ( "test.ephemeral", "test.incremental", "test.sub.inner", "test.metricflow_time_spine", + "test.metricflow_time_spine_second", "test.outer", ), "json": ( @@ -174,6 +197,10 @@ def expect_model_output(self): "docs": {"node_color": None, "show": True}, "contract": {"enforced": False, "alias_types": True}, "access": "protected", + "event_time": None, + "lookback": 1, + "batch_size": None, + "begin": None, }, "original_file_path": normalize("models/ephemeral.sql"), "unique_id": "model.test.ephemeral", @@ -212,6 +239,10 @@ def expect_model_output(self): "docs": {"node_color": None, "show": True}, "contract": {"enforced": False, "alias_types": True}, "access": "protected", + "event_time": None, + "lookback": 1, + "batch_size": None, + "begin": None, }, "original_file_path": normalize("models/incremental.sql"), "unique_id": "model.test.incremental", @@ -250,6 +281,10 @@ def expect_model_output(self): "docs": {"node_color": None, "show": True}, "contract": {"enforced": False, "alias_types": True}, "access": "protected", + "event_time": None, + "lookback": 1, + "batch_size": None, + "begin": None, }, "original_file_path": normalize("models/sub/inner.sql"), "unique_id": "model.test.inner", @@ -288,12 +323,58 @@ def expect_model_output(self): "docs": {"node_color": None, "show": True}, "contract": {"enforced": False, "alias_types": True}, "access": "protected", + "event_time": None, + "lookback": 1, + "batch_size": None, + "begin": None, }, "original_file_path": normalize("models/metricflow_time_spine.sql"), "unique_id": "model.test.metricflow_time_spine", "alias": "metricflow_time_spine", "resource_type": "model", }, + { + "name": "metricflow_time_spine_second", + "package_name": "test", + "depends_on": { + "nodes": [], + "macros": ["macro.dbt.current_timestamp", "macro.dbt.date_trunc"], + }, + "tags": [], + "config": { + "enabled": True, + "group": None, + "materialized": "view", + "post-hook": [], + "tags": [], + "pre-hook": [], + "quoting": {}, + "column_types": {}, + "persist_docs": {}, + "full_refresh": None, + "unique_key": None, + "on_schema_change": "ignore", + "on_configuration_change": "apply", + "database": None, + "schema": None, + "alias": None, + "meta": {}, + "grants": {}, + "packages": [], + "incremental_strategy": None, + "docs": {"node_color": None, "show": True}, + "contract": {"enforced": False, "alias_types": True}, + "access": "protected", + "event_time": None, + "lookback": 1, + "batch_size": None, + "begin": None, + }, + "original_file_path": normalize("models/metricflow_time_spine_second.sql"), + "unique_id": "model.test.metricflow_time_spine_second", + "alias": "metricflow_time_spine_second", + "resource_type": "model", + }, { "name": "outer", "package_name": "test", @@ -326,6 +407,10 @@ def expect_model_output(self): "docs": {"node_color": None, "show": True}, "contract": {"enforced": False, "alias_types": True}, "access": "protected", + "event_time": None, + "lookback": 1, + "batch_size": None, + "begin": None, }, "original_file_path": normalize("models/outer.sql"), "unique_id": "model.test.outer", @@ -338,6 +423,7 @@ def expect_model_output(self): self.dir("models/incremental.sql"), self.dir("models/sub/inner.sql"), self.dir("models/metricflow_time_spine.sql"), + self.dir("models/metricflow_time_spine_second.sql"), self.dir("models/outer.sql"), ), } @@ -393,6 +479,7 @@ def expect_source_output(self): "json": { "config": { "enabled": True, + "event_time": None, }, "unique_id": "source.test.my_source.my_table", "original_file_path": normalize("models/schema.yml"), @@ -443,6 +530,10 @@ def expect_seed_output(self): "incremental_strategy": None, "docs": {"node_color": None, "show": True}, "contract": {"enforced": False, "alias_types": True}, + "event_time": None, + "lookback": 1, + "batch_size": None, + "begin": None, }, "depends_on": {"macros": []}, "unique_id": "seed.test.seed", @@ -573,6 +664,7 @@ def expect_all_output(self): "test.not_null_outer_id", "test.unique_outer_id", "test.metricflow_time_spine", + "test.metricflow_time_spine_second", "test.t", "semantic_model:test.my_sm", "metric:test.total_outer", @@ -618,6 +710,7 @@ def expect_select(self): "test.ephemeral", "test.outer", "test.metricflow_time_spine", + "test.metricflow_time_spine_second", "test.incremental", } @@ -638,6 +731,7 @@ def expect_resource_type_multiple(self): "test.outer", "test.sub.inner", "test.metricflow_time_spine", + "test.metricflow_time_spine_second", "test.t", "test.unique_outer_id", } @@ -658,6 +752,7 @@ def expect_resource_type_multiple(self): "test.not_null_outer_id", "test.outer", "test.metricflow_time_spine", + "test.metricflow_time_spine_second", "test.sub.inner", "test.t", } @@ -693,6 +788,7 @@ def expect_resource_type_env_var(self): "test.outer", "test.sub.inner", "test.metricflow_time_spine", + "test.metricflow_time_spine_second", "test.t", "test.unique_outer_id", } @@ -707,6 +803,7 @@ def expect_resource_type_env_var(self): "test.outer", "test.sub.inner", "test.metricflow_time_spine", + "test.metricflow_time_spine_second", } del os.environ["DBT_EXCLUDE_RESOURCE_TYPES"] diff --git a/tests/functional/logging/test_logging.py b/tests/functional/logging/test_logging.py index 9205a8faab4..9de46a84a95 100644 --- a/tests/functional/logging/test_logging.py +++ b/tests/functional/logging/test_logging.py @@ -5,6 +5,7 @@ from dbt.events.types import InvalidOptionYAML from dbt.tests.util import get_manifest, read_file, run_dbt +from dbt_common.events import EventLevel from dbt_common.events.functions import fire_event my_model_sql = """ @@ -100,40 +101,179 @@ def test_invalid_event_value(project, logs_dir): with pytest.raises(Exception) as excinfo: fire_event(InvalidOptionYAML(option_name=1)) - assert str(excinfo.value) == "[InvalidOptionYAML]: Unable to parse dict {'option_name': 1}" + assert "[InvalidOptionYAML]: Unable to parse logging event dictionary." in str(excinfo.value) -class TestNodeInfo: +groups_yml = """ +groups: + - name: my_group_with_owner_metadata + owner: + name: my_name + email: my.email@gmail.com + slack: my_slack + other_property: something_else + +models: + - name: my_model + group: my_group_with_owner_metadata + access: public +""" + + +class TestRunResultErrorNodeInfo: @pytest.fixture(scope="class") def models(self): - return {"my_model.sql": "select not_found as id"} + return { + "my_model.sql": "select not_found as id", + } def test_node_info_on_results(self, project, logs_dir): results = run_dbt(["--log-format=json", "run"], expect_pass=False) assert len(results) == 1 - # get log file + log_file = read_file(logs_dir, "dbt.log") - task_printer_events = [ - "RunResultWarning", - "RunResultFailure", - "RunResultWarningMessage", - "RunResultError", - "RunResultErrorNoMessage", - "SQLCompiledPath", - "CheckNodeTestFailure", - ] - count = 0 + for log_line in log_file.split("\n"): - # skip empty lines - if len(log_line) == 0: + if not log_line: continue - # The adapter logging also shows up, so skip non-json lines - if "[debug]" in log_line: + + log_json = json.loads(log_line) + if log_json["info"]["level"] == EventLevel.DEBUG: continue - log_dct = json.loads(log_line) - log_data = log_dct["data"] - log_event = log_dct["info"]["name"] - if log_event in task_printer_events: - assert "node_info" in log_data - count += 1 - assert count > 0 + + if log_json["info"]["name"] == "RunResultError": + assert "node_info" in log_json["data"] + assert log_json["data"]["node_info"]["unique_id"] == "model.test.my_model" + assert "Database Error" in log_json["data"]["msg"] + + +def assert_group_data(group_data): + assert group_data["name"] == "my_group_with_owner_metadata" + assert group_data["owner"] == { + "name": "my_name", + "email": "my.email@gmail.com", + "slack": "my_slack", + "other_property": "something_else", + } + + +class TestRunResultErrorGroup: + @pytest.fixture(scope="class") + def models(self): + return { + "my_model.sql": "select not_found as id", + "groups.yml": groups_yml, + } + + def test_node_info_on_results(self, project, logs_dir): + results = run_dbt(["--log-format=json", "run"], expect_pass=False) + assert len(results) == 1 + + log_file = read_file(logs_dir, "dbt.log") + run_result_error_count = 0 + + for log_line in log_file.split("\n"): + if not log_line: + continue + + log_json = json.loads(log_line) + if log_json["info"]["level"] == EventLevel.DEBUG: + continue + + if log_json["info"]["name"] == "RunResultError": + assert "group" in log_json["data"] + assert_group_data(log_json["data"]["group"]) + run_result_error_count += 1 + + assert run_result_error_count == 1 + + +class TestRunResultFailureGroup: + @pytest.fixture(scope="class") + def models(self): + schema_yml = ( + groups_yml + + """ + columns: + - name: my_column + tests: + - not_null +""" + ) + print(schema_yml) + return { + "my_model.sql": "select 1 as id, null as my_column", + "groups.yml": schema_yml, + } + + def test_node_info_on_results(self, project, logs_dir): + results = run_dbt(["--log-format=json", "build"], expect_pass=False) + assert len(results) == 2 + + log_file = read_file(logs_dir, "dbt.log") + run_result_error_count = 0 + run_result_failure_count = 0 + + for log_line in log_file.split("\n"): + if not log_line: + continue + + log_json = json.loads(log_line) + if log_json["info"]["level"] == EventLevel.DEBUG: + continue + + if log_json["info"]["name"] == "RunResultError": + assert "group" in log_json["data"] + assert_group_data(log_json["data"]["group"]) + run_result_error_count += 1 + + if log_json["info"]["name"] == "RunResultFailure": + assert "group" in log_json["data"] + assert_group_data(log_json["data"]["group"]) + run_result_failure_count += 1 + + assert run_result_error_count == 1 + assert run_result_failure_count == 1 + + +class TestRunResultWarningGroup: + @pytest.fixture(scope="class") + def models(self): + schema_yml = ( + groups_yml + + """ + columns: + - name: my_column + tests: + - not_null: + config: + severity: warn +""" + ) + print(schema_yml) + return { + "my_model.sql": "select 1 as id, null as my_column", + "groups.yml": schema_yml, + } + + def test_node_info_on_results(self, project, logs_dir): + results = run_dbt(["--log-format=json", "build"]) + assert len(results) == 2 + + log_file = read_file(logs_dir, "dbt.log") + run_result_warning_count = 0 + + for log_line in log_file.split("\n"): + if not log_line: + continue + + log_json = json.loads(log_line) + if log_json["info"]["level"] == EventLevel.DEBUG: + continue + + if log_json["info"]["name"] == "RunResultWarning": + assert "group" in log_json["data"] + assert_group_data(log_json["data"]["group"]) + run_result_warning_count += 1 + + assert run_result_warning_count == 1 diff --git a/tests/functional/materializations/test_incremental_with_contract.py b/tests/functional/materializations/test_incremental_with_contract.py new file mode 100644 index 00000000000..a9321f10118 --- /dev/null +++ b/tests/functional/materializations/test_incremental_with_contract.py @@ -0,0 +1,131 @@ +import pytest + +from dbt.tests.util import ( + check_relations_equal, + get_relation_columns, + relation_from_name, + run_dbt, +) + +seeds_base_csv = """ +id,name_xxx,some_date +1,Easton,1981-05-20T06:46:51 +2,Lillian,1978-09-03T18:10:33 +3,Jeremiah,1982-03-11T03:59:51 +4,Nolan,1976-05-06T20:21:35 +5,Hannah,1982-06-23T05:41:26 +6,Eleanor,1991-08-10T23:12:21 +7,Lily,1971-03-29T14:58:02 +8,Jonathan,1988-02-26T02:55:24 +9,Adrian,1994-02-09T13:14:23 +10,Nora,1976-03-01T16:51:39 +""".lstrip() + + +seeds_added_csv = ( + seeds_base_csv + + """ +11,Mateo,2014-09-07T17:04:27 +12,Julian,2000-02-04T11:48:30 +13,Gabriel,2001-07-10T07:32:52 +14,Isaac,2002-11-24T03:22:28 +15,Levi,2009-11-15T11:57:15 +16,Elizabeth,2005-04-09T03:50:11 +17,Grayson,2019-08-06T19:28:17 +18,Dylan,2014-03-01T11:50:41 +19,Jayden,2009-06-06T07:12:49 +20,Luke,2003-12-05T21:42:18 +""".lstrip() +) + +incremental_not_schema_change_sql = """ +{{ config(materialized="incremental", unique_key="user_id_current_time",on_schema_change="sync_all_columns") }} +select + 1 || '-' || current_timestamp as user_id_current_time, + {% if is_incremental() %} + 'thisis18characters' as platform + {% else %} + 'okthisis20characters' as platform + {% endif %} +""" + +incremental_sql = """ + {{ config(materialized="incremental") }} + select * from {{ source('raw', 'seed') }} + {% if is_incremental() %} + where id > (select max(id) from {{ this }}) + {% endif %} +""" + +schema_base_yml = """ +sources: + - name: raw + schema: "{{ target.schema }}" + tables: + - name: seed + identifier: "{{ var('seed_name', 'base') }}" + +models: + - name: incremental + config: + contract: + enforced: true + on_schema_change: append_new_columns + columns: + - name: id + data_type: int + - name: name_xxx + data_type: character varying(10) + - name: some_date + data_type: timestamp +""" + + +class TestIncremental: + @pytest.fixture(scope="class") + def project_config_update(self): + return {"name": "incremental"} + + @pytest.fixture(scope="class") + def models(self): + return {"incremental.sql": incremental_sql, "schema.yml": schema_base_yml} + + @pytest.fixture(scope="class") + def seeds(self): + return {"base.csv": seeds_base_csv, "added.csv": seeds_added_csv} + + def test_incremental(self, project): + # seed command + results = run_dbt(["seed"]) + assert len(results) == 2 + + # base table rowcount + relation = relation_from_name(project.adapter, "base") + result = project.run_sql(f"select count(*) as num_rows from {relation}", fetch="one") + assert result[0] == 10 + + # added table rowcount + relation = relation_from_name(project.adapter, "added") + result = project.run_sql(f"select count(*) as num_rows from {relation}", fetch="one") + assert result[0] == 20 + + # run command + # the "seed_name" var changes the seed identifier in the schema file + results = run_dbt(["run", "--vars", "seed_name: base"]) + assert len(results) == 1 + + # check relations equal + check_relations_equal(project.adapter, ["base", "incremental"]) + + # change seed_name var + # the "seed_name" var changes the seed identifier in the schema file + results = run_dbt(["run", "--debug", "--vars", "seed_name: added"]) + assert len(results) == 1 + + # Error before fix: Changing col type from character varying(10) to character varying(256) in table: + # "dbt"."test<...>_test_incremental_with_contract"."incremental" + columns = get_relation_columns(project.adapter, "incremental") + # [('id', 'integer', None), ('name_xxx', 'character varying', 10), ('some_date', 'timestamp without time zone', None)] + for column in columns: + if column[0] == "name_xxx": + assert column[2] == 10 diff --git a/tests/functional/microbatch/test_microbatch.py b/tests/functional/microbatch/test_microbatch.py new file mode 100644 index 00000000000..bd13f9cff24 --- /dev/null +++ b/tests/functional/microbatch/test_microbatch.py @@ -0,0 +1,758 @@ +import os +from unittest import mock + +import pytest + +from dbt.events.types import LogModelResult, MicrobatchModelNoEventTimeInputs +from dbt.tests.util import ( + get_artifact, + patch_microbatch_end_time, + read_file, + relation_from_name, + run_dbt, + run_dbt_and_capture, + write_file, +) +from tests.utils import EventCatcher + +input_model_sql = """ +{{ config(materialized='table', event_time='event_time') }} + +select 1 as id, TIMESTAMP '2020-01-01 00:00:00-0' as event_time +union all +select 2 as id, TIMESTAMP '2020-01-02 00:00:00-0' as event_time +union all +select 3 as id, TIMESTAMP '2020-01-03 00:00:00-0' as event_time +""" + +input_model_without_event_time_sql = """ +{{ config(materialized='table') }} + +select 1 as id, TIMESTAMP '2020-01-01 00:00:00-0' as event_time +union all +select 2 as id, TIMESTAMP '2020-01-02 00:00:00-0' as event_time +union all +select 3 as id, TIMESTAMP '2020-01-03 00:00:00-0' as event_time +""" + +microbatch_model_sql = """ +{{ config(materialized='incremental', incremental_strategy='microbatch', unique_key='id', event_time='event_time', batch_size='day', begin=modules.datetime.datetime(2020, 1, 1, 0, 0, 0)) }} +select * from {{ ref('input_model') }} +""" + +microbatch_yearly_model_sql = """ +{{ config(materialized='incremental', incremental_strategy='microbatch', unique_key='id', event_time='event_time', batch_size='year', begin=modules.datetime.datetime(2020, 1, 1, 0, 0, 0)) }} +select * from {{ ref('input_model') }} +""" + +microbatch_yearly_model_downstream_sql = """ +{{ config(materialized='incremental', incremental_strategy='microbatch', unique_key='id', event_time='event_time', batch_size='year', begin=modules.datetime.datetime(2020, 1, 1, 0, 0, 0)) }} +select * from {{ ref('microbatch_model') }} +""" + +invalid_batch_context_macro_sql = """ +{% macro check_invalid_batch_context() %} + +{% if model is not mapping %} + {{ exceptions.raise_compiler_error("`model` is invalid: expected mapping type") }} +{% elif compiled_code and compiled_code is not string %} + {{ exceptions.raise_compiler_error("`compiled_code` is invalid: expected string type") }} +{% elif sql and sql is not string %} + {{ exceptions.raise_compiler_error("`sql` is invalid: expected string type") }} +{% elif is_incremental is not callable %} + {{ exceptions.raise_compiler_error("`is_incremental()` is invalid: expected callable type") }} +{% elif should_full_refresh is not callable %} + {{ exceptions.raise_compiler_error("`should_full_refresh()` is invalid: expected callable type") }} +{% endif %} + +{% endmacro %} +""" + +microbatch_model_with_context_checks_sql = """ +{{ config(pre_hook="{{ check_invalid_batch_context() }}", materialized='incremental', incremental_strategy='microbatch', unique_key='id', event_time='event_time', batch_size='day', begin=modules.datetime.datetime(2020, 1, 1, 0, 0, 0)) }} + +{{ check_invalid_batch_context() }} +select * from {{ ref('input_model') }} +""" + +microbatch_model_downstream_sql = """ +{{ config(materialized='incremental', incremental_strategy='microbatch', unique_key='id', event_time='event_time', batch_size='day', begin=modules.datetime.datetime(2020, 1, 1, 0, 0, 0)) }} +select * from {{ ref('microbatch_model') }} +""" + +microbatch_model_ref_render_sql = """ +{{ config(materialized='incremental', incremental_strategy='microbatch', unique_key='id', event_time='event_time', batch_size='day', begin=modules.datetime.datetime(2020, 1, 1, 0, 0, 0)) }} +select * from {{ ref('input_model').render() }} +""" + +seed_csv = """id,event_time +1,'2020-01-01 00:00:00-0' +2,'2020-01-02 00:00:00-0' +3,'2020-01-03 00:00:00-0' +""" + +seeds_yaml = """ +seeds: + - name: raw_source + config: + column_types: + event_time: TIMESTAMP +""" + +sources_yaml = """ +sources: + - name: seed_sources + schema: "{{ target.schema }}" + tables: + - name: raw_source + config: + event_time: event_time +""" + +microbatch_model_calling_source_sql = """ +{{ config(materialized='incremental', incremental_strategy='microbatch', unique_key='id', event_time='event_time', batch_size='day', begin=modules.datetime.datetime(2020, 1, 1, 0, 0, 0)) }} +select * from {{ source('seed_sources', 'raw_source') }} +""" + +custom_microbatch_strategy = """ +{% macro get_incremental_microbatch_sql(arg_dict) %} + {% do log('custom microbatch strategy', info=True) %} + + {%- set dest_cols_csv = get_quoted_csv(arg_dict["dest_columns"] | map(attribute="name")) -%} + + insert into {{ arg_dict["target_relation"] }} ({{ dest_cols_csv }}) + ( + select {{ dest_cols_csv }} + from {{ arg_dict["temp_relation"] }} + ) + +{% endmacro %} +""" + + +downstream_model_of_microbatch_sql = """ +SELECT * FROM {{ ref('microbatch_model') }} +""" + +microbatch_model_full_refresh_false_sql = """ +{{ config(materialized='incremental', incremental_strategy='microbatch', unique_key='id', event_time='event_time', batch_size='day', begin=modules.datetime.datetime(2020, 1, 1, 0, 0, 0), full_refresh=False) }} +select * from {{ ref('input_model') }} +""" + + +class BaseMicrobatchCustomUserStrategy: + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": input_model_sql, + "microbatch_model.sql": microbatch_model_sql, + } + + @pytest.fixture(scope="class") + def macros(self): + return {"microbatch.sql": custom_microbatch_strategy} + + +class TestMicrobatchCustomUserStrategyDefault(BaseMicrobatchCustomUserStrategy): + def test_use_custom_microbatch_strategy_by_default(self, project): + with mock.patch.object( + type(project.adapter), "valid_incremental_strategies", lambda _: [] + ): + # Initial run + run_dbt(["run"]) + + # Incremental run uses custom strategy + _, logs = run_dbt_and_capture(["run"]) + assert "custom microbatch strategy" in logs + + +class TestMicrobatchCustomUserStrategyEnvVarTrueValid(BaseMicrobatchCustomUserStrategy): + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_use_custom_microbatch_strategy_env_var_true_invalid_incremental_strategy( + self, project + ): + with mock.patch.object( + type(project.adapter), "valid_incremental_strategies", lambda _: ["microbatch"] + ): + # Initial run + with patch_microbatch_end_time("2020-01-03 13:57:00"): + run_dbt(["run"]) + + # Incremental run uses custom strategy + with patch_microbatch_end_time("2020-01-03 13:57:00"): + _, logs = run_dbt_and_capture(["run"]) + assert "custom microbatch strategy" in logs + + +# TODO: Consider a behaviour flag here if DBT_EXPERIMENTAL_MICROBATCH is removed +# Since this causes an exception prior to using an override +class TestMicrobatchCustomUserStrategyEnvVarTrueInvalid(BaseMicrobatchCustomUserStrategy): + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_use_custom_microbatch_strategy_env_var_true_invalid_incremental_strategy( + self, project + ): + with mock.patch.object( + type(project.adapter), "valid_incremental_strategies", lambda _: [] + ): + # Run of microbatch model while adapter doesn't have a "valid" + # microbatch strategy causes an error to be raised + with patch_microbatch_end_time("2020-01-03 13:57:00"): + _, logs = run_dbt_and_capture(["run"], expect_pass=False) + assert "'microbatch' is not valid" in logs + + +class BaseMicrobatchTest: + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": input_model_sql, + "microbatch_model.sql": microbatch_model_sql, + } + + def assert_row_count(self, project, relation_name: str, expected_row_count: int): + relation = relation_from_name(project.adapter, relation_name) + result = project.run_sql(f"select count(*) as num_rows from {relation}", fetch="one") + + if result[0] != expected_row_count: + # running show for debugging + run_dbt(["show", "--inline", f"select * from {relation}"]) + + assert result[0] == expected_row_count + + +class TestMicrobatchCLI(BaseMicrobatchTest): + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_run_with_event_time(self, project): + # run without --event-time-start or --event-time-end - 3 expected rows in output + catcher = EventCatcher(event_to_catch=LogModelResult) + + with patch_microbatch_end_time("2020-01-03 13:57:00"): + run_dbt(["run"], callbacks=[catcher.catch]) + self.assert_row_count(project, "microbatch_model", 3) + + assert len(catcher.caught_events) == 5 + batch_creation_events = 0 + for caught_event in catcher.caught_events: + if "batch 2020" in caught_event.data.description: + batch_creation_events += 1 + assert caught_event.data.execution_time > 0 + # 3 batches should have been run, so there should be 3 batch + # creation events + assert batch_creation_events == 3 + + # build model between 2020-01-02 >= event_time < 2020-01-03 + run_dbt( + [ + "run", + "--event-time-start", + "2020-01-02", + "--event-time-end", + "2020-01-03", + "--full-refresh", + ] + ) + self.assert_row_count(project, "microbatch_model", 1) + + +class TestMicroBatchBoundsDefault(BaseMicrobatchTest): + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_run_with_event_time(self, project): + # initial run -- backfills all data + with patch_microbatch_end_time("2020-01-03 13:57:00"): + run_dbt(["run"]) + self.assert_row_count(project, "microbatch_model", 3) + + # our partition grain is "day" so running the same day without new data should produce the same results + with patch_microbatch_end_time("2020-01-03 14:57:00"): + run_dbt(["run"]) + self.assert_row_count(project, "microbatch_model", 3) + + # add next two days of data + test_schema_relation = project.adapter.Relation.create( + database=project.database, schema=project.test_schema + ) + project.run_sql( + f"insert into {test_schema_relation}.input_model(id, event_time) values (4, TIMESTAMP '2020-01-04 00:00:00-0'), (5, TIMESTAMP '2020-01-05 00:00:00-0')" + ) + self.assert_row_count(project, "input_model", 5) + + # re-run without changing current time => no insert + with patch_microbatch_end_time("2020-01-03 14:57:00"): + run_dbt(["run", "--select", "microbatch_model"]) + self.assert_row_count(project, "microbatch_model", 3) + + # re-run by advancing time by one day changing current time => insert 1 row + with patch_microbatch_end_time("2020-01-04 14:57:00"): + run_dbt(["run", "--select", "microbatch_model"]) + self.assert_row_count(project, "microbatch_model", 4) + + # re-run by advancing time by one more day changing current time => insert 1 more row + with patch_microbatch_end_time("2020-01-05 14:57:00"): + run_dbt(["run", "--select", "microbatch_model"]) + self.assert_row_count(project, "microbatch_model", 5) + + +class TestMicrobatchWithSource(BaseMicrobatchTest): + @pytest.fixture(scope="class") + def seeds(self): + return { + "raw_source.csv": seed_csv, + } + + @pytest.fixture(scope="class") + def models(self): + return { + "microbatch_model.sql": microbatch_model_calling_source_sql, + "sources.yml": sources_yaml, + "seeds.yml": seeds_yaml, + } + + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_run_with_event_time(self, project): + # ensure seed is created for source + run_dbt(["seed"]) + + # initial run -- backfills all data + catcher = EventCatcher(event_to_catch=MicrobatchModelNoEventTimeInputs) + with patch_microbatch_end_time("2020-01-03 13:57:00"): + run_dbt(["run"], callbacks=[catcher.catch]) + self.assert_row_count(project, "microbatch_model", 3) + assert len(catcher.caught_events) == 0 + + # our partition grain is "day" so running the same day without new data should produce the same results + with patch_microbatch_end_time("2020-01-03 14:57:00"): + run_dbt(["run"]) + self.assert_row_count(project, "microbatch_model", 3) + + # add next two days of data + test_schema_relation = project.adapter.Relation.create( + database=project.database, schema=project.test_schema + ) + project.run_sql( + f"insert into {test_schema_relation}.raw_source(id, event_time) values (4, TIMESTAMP '2020-01-04 00:00:00-0'), (5, TIMESTAMP '2020-01-05 00:00:00-0')" + ) + self.assert_row_count(project, "raw_source", 5) + + # re-run without changing current time => no insert + with patch_microbatch_end_time("2020-01-03 14:57:00"): + run_dbt(["run", "--select", "microbatch_model"]) + self.assert_row_count(project, "microbatch_model", 3) + + # re-run by advancing time by one day changing current time => insert 1 row + with patch_microbatch_end_time("2020-01-04 14:57:00"): + run_dbt(["run", "--select", "microbatch_model"]) + self.assert_row_count(project, "microbatch_model", 4) + + # re-run by advancing time by one more day changing current time => insert 1 more row + with patch_microbatch_end_time("2020-01-05 14:57:00"): + run_dbt(["run", "--select", "microbatch_model"]) + self.assert_row_count(project, "microbatch_model", 5) + + +class TestMicrobatchJinjaContext(BaseMicrobatchTest): + + @pytest.fixture(scope="class") + def macros(self): + return {"check_batch_context.sql": invalid_batch_context_macro_sql} + + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": input_model_sql, + "microbatch_model.sql": microbatch_model_with_context_checks_sql, + } + + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_run_with_event_time(self, project): + # initial run -- backfills all data + with patch_microbatch_end_time("2020-01-03 13:57:00"): + run_dbt(["run"]) + self.assert_row_count(project, "microbatch_model", 3) + + +class TestMicrobatchWithInputWithoutEventTime(BaseMicrobatchTest): + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": input_model_without_event_time_sql, + "microbatch_model.sql": microbatch_model_sql, + } + + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_run_with_event_time(self, project): + catcher = EventCatcher(event_to_catch=MicrobatchModelNoEventTimeInputs) + + # initial run -- backfills all data + with patch_microbatch_end_time("2020-01-03 13:57:00"): + run_dbt(["run"], callbacks=[catcher.catch]) + self.assert_row_count(project, "microbatch_model", 3) + assert len(catcher.caught_events) == 1 + + # our partition grain is "day" so running the same day without new data should produce the same results + with patch_microbatch_end_time("2020-01-03 14:57:00"): + run_dbt(["run"]) + self.assert_row_count(project, "microbatch_model", 3) + + # add next two days of data + test_schema_relation = project.adapter.Relation.create( + database=project.database, schema=project.test_schema + ) + project.run_sql( + f"insert into {test_schema_relation}.input_model(id, event_time) values (4, TIMESTAMP '2020-01-04 00:00:00-0'), (5, TIMESTAMP '2020-01-05 00:00:00-0')" + ) + self.assert_row_count(project, "input_model", 5) + + # re-run without changing current time => INSERT BECAUSE INPUT MODEL ISN'T BEING FILTERED + with patch_microbatch_end_time("2020-01-03 14:57:00"): + run_dbt(["run", "--select", "microbatch_model"]) + self.assert_row_count(project, "microbatch_model", 5) + + +class TestMicrobatchUsingRefRenderSkipsFilter(BaseMicrobatchTest): + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_run_with_event_time(self, project): + # initial run -- backfills all data + with patch_microbatch_end_time("2020-01-03 13:57:00"): + run_dbt(["run"]) + self.assert_row_count(project, "microbatch_model", 3) + + # our partition grain is "day" so running the same day without new data should produce the same results + with patch_microbatch_end_time("2020-01-03 14:57:00"): + run_dbt(["run"]) + self.assert_row_count(project, "microbatch_model", 3) + + # add next two days of data + test_schema_relation = project.adapter.Relation.create( + database=project.database, schema=project.test_schema + ) + project.run_sql( + f"insert into {test_schema_relation}.input_model(id, event_time) values (4, TIMESTAMP '2020-01-04 00:00:00-0'), (5, TIMESTAMP '2020-01-05 00:00:00-0')" + ) + self.assert_row_count(project, "input_model", 5) + + # re-run without changing current time => no insert + with patch_microbatch_end_time("2020-01-03 14:57:00"): + run_dbt(["run", "--select", "microbatch_model"]) + self.assert_row_count(project, "microbatch_model", 3) + + # Update microbatch model to call .render() on ref('input_model') + write_file( + microbatch_model_ref_render_sql, project.project_root, "models", "microbatch_model.sql" + ) + + # re-run without changing current time => INSERT because .render() skips filtering + with patch_microbatch_end_time("2020-01-03 14:57:00"): + run_dbt(["run", "--select", "microbatch_model"]) + self.assert_row_count(project, "microbatch_model", 5) + + +microbatch_model_context_vars = """ +{{ config(materialized='incremental', incremental_strategy='microbatch', unique_key='id', event_time='event_time', batch_size='day', begin=modules.datetime.datetime(2020, 1, 1, 0, 0, 0)) }} +{{ log("start: "~ model.config.__dbt_internal_microbatch_event_time_start, info=True)}} +{{ log("end: "~ model.config.__dbt_internal_microbatch_event_time_end, info=True)}} +select * from {{ ref('input_model') }} +""" + + +class TestMicrobatchJinjaContextVarsAvailable(BaseMicrobatchTest): + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": input_model_sql, + "microbatch_model.sql": microbatch_model_context_vars, + } + + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_run_with_event_time_logs(self, project): + with patch_microbatch_end_time("2020-01-03 13:57:00"): + _, logs = run_dbt_and_capture(["run"]) + + assert "start: 2020-01-01 00:00:00+00:00" in logs + assert "end: 2020-01-02 00:00:00+00:00" in logs + + assert "start: 2020-01-02 00:00:00+00:00" in logs + assert "end: 2020-01-03 00:00:00+00:00" in logs + + assert "start: 2020-01-03 00:00:00+00:00" in logs + assert "end: 2020-01-03 13:57:00+00:00" in logs + + +microbatch_model_failing_incremental_partition_sql = """ +{{ config(materialized='incremental', incremental_strategy='microbatch', unique_key='id', event_time='event_time', batch_size='day', begin=modules.datetime.datetime(2020, 1, 1, 0, 0, 0)) }} +{% if '2020-01-02' in (model.config.__dbt_internal_microbatch_event_time_start | string) %} + invalid_sql +{% endif %} +select * from {{ ref('input_model') }} +""" + + +class TestMicrobatchIncrementalPartitionFailure(BaseMicrobatchTest): + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": input_model_sql, + "microbatch_model.sql": microbatch_model_failing_incremental_partition_sql, + "downstream_model.sql": downstream_model_of_microbatch_sql, + } + + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_run_with_event_time(self, project): + # run all partitions from start - 2 expected rows in output, one failed + with patch_microbatch_end_time("2020-01-03 13:57:00"): + run_dbt(["run"], expect_pass=False) + self.assert_row_count(project, "microbatch_model", 2) + + run_results = get_artifact(project.project_root, "target", "run_results.json") + microbatch_run_result = run_results["results"][1] + assert microbatch_run_result["status"] == "partial success" + batch_results = microbatch_run_result["batch_results"] + assert batch_results is not None + assert len(batch_results["successful"]) == 2 + assert len(batch_results["failed"]) == 1 + assert run_results["results"][2]["status"] == "skipped" + + +class TestMicrobatchRetriesPartialSuccesses(BaseMicrobatchTest): + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": input_model_sql, + "microbatch_model.sql": microbatch_model_failing_incremental_partition_sql, + } + + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_run_with_event_time(self, project): + # run all partitions from start - 2 expected rows in output, one failed + with patch_microbatch_end_time("2020-01-03 13:57:00"): + _, console_output = run_dbt_and_capture(["run"]) + + assert "PARTIAL SUCCESS (2/3)" in console_output + assert "Completed with 1 partial success" in console_output + + self.assert_row_count(project, "microbatch_model", 2) + + run_results = get_artifact(project.project_root, "target", "run_results.json") + microbatch_run_result = run_results["results"][1] + assert microbatch_run_result["status"] == "partial success" + batch_results = microbatch_run_result["batch_results"] + assert batch_results is not None + assert len(batch_results["successful"]) == 2 + assert len(batch_results["failed"]) == 1 + + # update the microbatch model so that it no longer fails + write_file(microbatch_model_sql, project.project_root, "models", "microbatch_model.sql") + + with patch_microbatch_end_time("2020-01-03 13:57:00"): + _, console_output = run_dbt_and_capture(["retry"]) + + assert "PARTIAL SUCCESS" not in console_output + assert "Completed with 1 partial success" not in console_output + assert "Completed successfully" in console_output + + self.assert_row_count(project, "microbatch_model", 3) + + +class TestMicrobatchMultipleRetries(BaseMicrobatchTest): + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": input_model_sql, + "microbatch_model.sql": microbatch_model_failing_incremental_partition_sql, + } + + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_run_with_event_time(self, project): + # run all partitions from start - 2 expected rows in output, one failed + with patch_microbatch_end_time("2020-01-03 13:57:00"): + _, console_output = run_dbt_and_capture(["run"]) + + assert "PARTIAL SUCCESS (2/3)" in console_output + assert "Completed with 1 partial success" in console_output + + self.assert_row_count(project, "microbatch_model", 2) + + with patch_microbatch_end_time("2020-01-03 13:57:00"): + _, console_output = run_dbt_and_capture(["retry"], expect_pass=False) + + assert "PARTIAL SUCCESS" not in console_output + assert "ERROR" in console_output + assert "Completed with 1 error, 0 partial successs, and 0 warnings" in console_output + + self.assert_row_count(project, "microbatch_model", 2) + + with patch_microbatch_end_time("2020-01-03 13:57:00"): + _, console_output = run_dbt_and_capture(["retry"], expect_pass=False) + + assert "PARTIAL SUCCESS" not in console_output + assert "ERROR" in console_output + assert "Completed with 1 error, 0 partial successs, and 0 warnings" in console_output + + self.assert_row_count(project, "microbatch_model", 2) + + +microbatch_model_first_partition_failing_sql = """ +{{ config(materialized='incremental', incremental_strategy='microbatch', unique_key='id', event_time='event_time', batch_size='day', begin=modules.datetime.datetime(2020, 1, 1, 0, 0, 0)) }} +{% if '2020-01-01' in (model.config.__dbt_internal_microbatch_event_time_start | string) %} + invalid_sql +{% endif %} +select * from {{ ref('input_model') }} +""" + + +class TestMicrobatchInitialPartitionFailure(BaseMicrobatchTest): + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": input_model_sql, + "microbatch_model.sql": microbatch_model_first_partition_failing_sql, + } + + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_run_with_event_time(self, project): + # run all partitions from start - 2 expected rows in output, one failed + with patch_microbatch_end_time("2020-01-03 13:57:00"): + run_dbt(["run"]) + self.assert_row_count(project, "microbatch_model", 2) + + +class TestMicrobatchCompiledRunPaths(BaseMicrobatchTest): + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_run_with_event_time(self, project): + # run all partitions from start - 2 expected rows in output, one failed + with patch_microbatch_end_time("2020-01-03 13:57:00"): + run_dbt(["run"]) + + # Compiled paths - compiled model without filter only + assert read_file( + project.project_root, + "target", + "compiled", + "test", + "models", + "microbatch_model.sql", + ) + + # Compiled paths - batch compilations + assert read_file( + project.project_root, + "target", + "compiled", + "test", + "models", + "microbatch_model", + "microbatch_model_2020-01-01.sql", + ) + assert read_file( + project.project_root, + "target", + "compiled", + "test", + "models", + "microbatch_model", + "microbatch_model_2020-01-02.sql", + ) + assert read_file( + project.project_root, + "target", + "compiled", + "test", + "models", + "microbatch_model", + "microbatch_model_2020-01-03.sql", + ) + + assert read_file( + project.project_root, + "target", + "run", + "test", + "models", + "microbatch_model", + "microbatch_model_2020-01-01.sql", + ) + assert read_file( + project.project_root, + "target", + "run", + "test", + "models", + "microbatch_model", + "microbatch_model_2020-01-02.sql", + ) + assert read_file( + project.project_root, + "target", + "run", + "test", + "models", + "microbatch_model", + "microbatch_model_2020-01-03.sql", + ) + + +class TestMicrobatchFullRefreshConfigFalse(BaseMicrobatchTest): + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": input_model_sql, + "microbatch_model.sql": microbatch_model_full_refresh_false_sql, + "downstream_model.sql": downstream_model_of_microbatch_sql, + } + + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_run_with_event_time(self, project): + # run all partitions from 2020-01-02 to spoofed "now" - 2 expected rows in output + with patch_microbatch_end_time("2020-01-03 13:57:00"): + run_dbt( + [ + "run", + "--event-time-start", + "2020-01-02", + "--event-time-end", + "2020-01-03 13:57:00", + ] + ) + self.assert_row_count(project, "microbatch_model", 2) + + # re-running shouldn't change what it's in the data set because there is nothing new + with patch_microbatch_end_time("2020-01-03 13:57:00"): + run_dbt(["run"]) + self.assert_row_count(project, "microbatch_model", 2) + + # running with --full-refresh shouldn't pick up 2020-01-01 BECAUSE the model has + # full_refresh = false + with patch_microbatch_end_time("2020-01-03 13:57:00"): + run_dbt(["run", "--full-refresh"]) + self.assert_row_count(project, "microbatch_model", 2) + + # update the microbatch model to no longer have full_refresh=False config + write_file(microbatch_model_sql, project.project_root, "models", "microbatch_model.sql") + + # running with full refresh should now pick up the 2020-01-01 data + with patch_microbatch_end_time("2020-01-03 13:57:00"): + run_dbt(["run", "--full-refresh"]) + self.assert_row_count(project, "microbatch_model", 3) + + +class TestMicrbobatchModelsRunWithSameCurrentTime(BaseMicrobatchTest): + + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": input_model_sql, + "microbatch_model.sql": microbatch_yearly_model_sql, + "second_microbatch_model.sql": microbatch_yearly_model_downstream_sql, + } + + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_microbatch(self, project) -> None: + run_dbt(["run"]) + + run_results = get_artifact(project.project_root, "target", "run_results.json") + microbatch_model_last_batch = run_results["results"][1]["batch_results"]["successful"][-1] + second_microbatch_model_last_batch = run_results["results"][2]["batch_results"][ + "successful" + ][-1] + + # they should have the same last batch because they are using the _same_ "current_time" + assert microbatch_model_last_batch == second_microbatch_model_last_batch diff --git a/tests/functional/microbatch/test_microbatch_config_validation.py b/tests/functional/microbatch/test_microbatch_config_validation.py new file mode 100644 index 00000000000..cdebd3a791b --- /dev/null +++ b/tests/functional/microbatch/test_microbatch_config_validation.py @@ -0,0 +1,185 @@ +import os +from unittest import mock + +import pytest + +from dbt.exceptions import ParsingError +from dbt.tests.util import run_dbt + +valid_microbatch_model_sql = """ +{{ config(materialized='incremental', incremental_strategy='microbatch', batch_size='day', event_time='event_time') }} +select * from {{ ref('input_model') }} +""" + +valid_microbatch_model_no_config_sql = """ +select * from {{ ref('input_model') }} +""" + +valid_microbatch_model_config_yml = """ +models: + - name: microbatch + config: + materialized: incremental + incremental_strategy: microbatch + batch_size: day + event_time: event_time + begin: 2020-01-01 +""" + +invalid_microbatch_model_config_yml = """ +models: + - name: microbatch + config: + materialized: incremental + incremental_strategy: microbatch + batch_size: day + event_time: event_time + begin: 2020-01-01 11 PM +""" + +missing_event_time_microbatch_model_sql = """ +{{ config(materialized='incremental', incremental_strategy='microbatch', batch_size='day') }} +select * from {{ ref('input_model') }} +""" + +invalid_event_time_microbatch_model_sql = """ +{{ config(materialized='incremental', incremental_strategy='microbatch', batch_size='day', event_time=2) }} +select * from {{ ref('input_model') }} +""" + +missing_begin_microbatch_model_sql = """ +{{ config(materialized='incremental', incremental_strategy='microbatch', batch_size='day', event_time='event_time') }} +select * from {{ ref('input_model') }} +""" + +invalid_begin_microbatch_model_sql = """ +{{ config(materialized='incremental', incremental_strategy='microbatch', batch_size='day', event_time='event_time', begin=2) }} +select * from {{ ref('input_model') }} +""" + + +missing_batch_size_microbatch_model_sql = """ +{{ config(materialized='incremental', incremental_strategy='microbatch', event_time='event_time') }} +select * from {{ ref('input_model') }} +""" + +invalid_batch_size_microbatch_model_sql = """ +{{ config(materialized='incremental', incremental_strategy='microbatch', batch_size='invalid', event_time='event_time') }} +select * from {{ ref('input_model') }} +""" + +invalid_event_time_input_model_sql = """ +{{ config(materialized='table', event_time=1) }} + +select 1 as id, TIMESTAMP '2020-01-01 00:00:00-0' as event_time +""" + +valid_input_model_sql = """ +{{ config(materialized='table') }} + +select 1 as id, TIMESTAMP '2020-01-01 00:00:00-0' as event_time +""" + + +class BaseMicrobatchTestParseError: + @pytest.fixture(scope="class") + def models(self): + return {} + + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_parsing_error_raised(self, project): + with pytest.raises(ParsingError): + run_dbt(["parse"]) + + +class BaseMicrobatchTestNoError: + @pytest.fixture(scope="class") + def models(self): + return {} + + @mock.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + def test_parsing_error_not_raised(self, project): + run_dbt(["parse"]) + + +class TestMissingEventTimeMicrobatch(BaseMicrobatchTestParseError): + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": valid_input_model_sql, + "microbatch.sql": missing_event_time_microbatch_model_sql, + } + + +class TestInvalidEventTimeMicrobatch(BaseMicrobatchTestParseError): + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": valid_input_model_sql, + "microbatch.sql": invalid_event_time_microbatch_model_sql, + } + + +class TestMissingBeginMicrobatch(BaseMicrobatchTestParseError): + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": valid_input_model_sql, + "microbatch.sql": missing_begin_microbatch_model_sql, + } + + +class TestInvaliBeginTypeMicrobatch(BaseMicrobatchTestParseError): + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": valid_input_model_sql, + "microbatch.sql": invalid_begin_microbatch_model_sql, + } + + +class TestInvaliBegiFormatMicrobatch(BaseMicrobatchTestParseError): + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": valid_input_model_sql, + "microbatch.sql": valid_microbatch_model_no_config_sql, + "microbatch.yml": invalid_microbatch_model_config_yml, + } + + +class TestMissingBatchSizeMicrobatch(BaseMicrobatchTestParseError): + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": valid_input_model_sql, + "microbatch.sql": missing_batch_size_microbatch_model_sql, + } + + +class TestInvalidBatchSizeMicrobatch(BaseMicrobatchTestParseError): + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": valid_input_model_sql, + "microbatch.sql": invalid_batch_size_microbatch_model_sql, + } + + +class TestInvalidInputEventTimeMicrobatch(BaseMicrobatchTestParseError): + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": invalid_event_time_input_model_sql, + "microbatch.sql": valid_microbatch_model_sql, + } + + +class TestValidBeginMicrobatch(BaseMicrobatchTestNoError): + @pytest.fixture(scope="class") + def models(self): + return { + "input_model.sql": valid_input_model_sql, + "microbatch.sql": valid_microbatch_model_no_config_sql, + "schema.yml": valid_microbatch_model_config_yml, + } diff --git a/tests/functional/relation_quoting/test_relation_quoting.py b/tests/functional/relation_quoting/test_relation_quoting.py new file mode 100644 index 00000000000..55f492cc080 --- /dev/null +++ b/tests/functional/relation_quoting/test_relation_quoting.py @@ -0,0 +1,64 @@ +import pytest + +from dbt.tests.util import read_file, run_dbt + +_SOURCES_YML = """ +sources: + - name: source_name + database: source_database + schema: source_schema + tables: + - name: customers +""" + + +class TestSourceQuotingGlobalConfigs: + @pytest.fixture(scope="class") + def project_config_update(self): + # Postgres quoting configs are True by default -- turn them all to False to show they are not respected during source rendering + return { + "quoting": { + "database": False, + "schema": False, + "identifier": False, + }, + } + + @pytest.fixture(scope="class") + def models(self): + return { + "sources.yml": _SOURCES_YML, + "model.sql": "select * from {{ source('source_name', 'customers') }}", + } + + def test_sources_ignore_global_quoting_configs(self, project): + run_dbt(["compile"]) + + generated_sql = read_file("target", "compiled", "test", "models", "model.sql") + assert generated_sql == 'select * from "source_database"."source_schema"."customers"' + + +class TestModelQuoting: + @pytest.fixture(scope="class") + def project_config_update(self): + # Postgres quoting configs are True by default -- turn them all to False to show they are respected during model rendering + return { + "quoting": { + "database": False, + "schema": False, + "identifier": False, + }, + } + + @pytest.fixture(scope="class") + def models(self): + return { + "model.sql": "select 1 as id", + "model_downstream.sql": "select * from {{ ref('model') }}", + } + + def test_models_respect_global_quoting_configs(self, project): + run_dbt(["compile"]) + + generated_sql = read_file("target", "compiled", "test", "models", "model_downstream.sql") + assert generated_sql == f"select * from dbt.{project.test_schema}.model" diff --git a/tests/functional/retry/test_retry.py b/tests/functional/retry/test_retry.py index 012db25e42f..0a9e94f2e06 100644 --- a/tests/functional/retry/test_retry.py +++ b/tests/functional/retry/test_retry.py @@ -5,7 +5,7 @@ from dbt.contracts.results import RunStatus, TestStatus from dbt.exceptions import DbtRuntimeError, TargetNotFoundError -from dbt.tests.util import rm_file, run_dbt, write_file +from dbt.tests.util import rm_file, run_dbt, update_config_file, write_file from tests.functional.retry.fixtures import ( macros__alter_timezone_sql, models__sample_model, @@ -365,3 +365,92 @@ def test_retry_target_path_flag(self, project): results = run_dbt(["retry", "--state", "artifacts", "--target-path", "my_target_path"]) assert len(results) == 1 assert Path("my_target_path").is_dir() + + +class TestRetryHooksAlwaysRun: + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "on-run-start": ["select 1;"], + "on-run-end": ["select 2;"], + } + + @pytest.fixture(scope="class") + def models(self): + return { + "sample_model.sql": models__sample_model, + } + + def test_retry_hooks_always_run(self, project): + res = run_dbt(["run", "--target-path", "target"], expect_pass=False) + assert len(res) == 3 + + write_file(models__second_model, "models", "sample_model.sql") + res = run_dbt(["retry", "--state", "target"]) + assert len(res) == 3 + + +class TestFixRetryHook: + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "flags": { + "skip_nodes_if_on_run_start_fails": True, + }, + "on-run-start": [ + "select 1 as id", + "select column_does_not_exist", + "select 2 as id", + ], + } + + @pytest.fixture(scope="class") + def models(self): + return { + "sample_model.sql": "select 1 as id, 1 as foo", + "second_model.sql": models__second_model, + "union_model.sql": models__union_model, + } + + def test_fix_retry_hook(self, project): + res = run_dbt(["run"], expect_pass=False) + assert {r.node.unique_id: r.status for r in res.results} == { + "operation.test.test-on-run-start-0": RunStatus.Success, + "operation.test.test-on-run-start-1": RunStatus.Error, + "operation.test.test-on-run-start-2": RunStatus.Skipped, + "model.test.sample_model": RunStatus.Skipped, + "model.test.second_model": RunStatus.Skipped, + "model.test.union_model": RunStatus.Skipped, + } + + res = run_dbt(["retry"], expect_pass=False) + assert {r.node.unique_id: r.status for r in res.results} == { + "operation.test.test-on-run-start-0": RunStatus.Success, + "operation.test.test-on-run-start-1": RunStatus.Error, + "operation.test.test-on-run-start-2": RunStatus.Skipped, + "model.test.sample_model": RunStatus.Skipped, + "model.test.second_model": RunStatus.Skipped, + "model.test.union_model": RunStatus.Skipped, + } + + new_dbt_project_yml = { + "flags": { + "skip_nodes_if_on_run_start_fails": True, + }, + "on-run-start": [ + "select 1 as id", + "select 3 as id", + "select 2 as id", + ], + } + + update_config_file(new_dbt_project_yml, project.project_root, "dbt_project.yml") + res = run_dbt(["retry"]) + assert {r.node.unique_id: r.status for r in res.results} == { + "operation.test.test-on-run-start-0": RunStatus.Success, + "operation.test.test-on-run-start-1": RunStatus.Success, + "operation.test.test-on-run-start-2": RunStatus.Success, + "model.test.sample_model": RunStatus.Success, + "model.test.second_model": RunStatus.Success, + "model.test.union_model": RunStatus.Success, + } diff --git a/tests/functional/run_operations/test_run_operations.py b/tests/functional/run_operations/test_run_operations.py index 064c98b3a51..258ed679d7e 100644 --- a/tests/functional/run_operations/test_run_operations.py +++ b/tests/functional/run_operations/test_run_operations.py @@ -3,6 +3,7 @@ import pytest import yaml +from dbt.artifacts.schemas.results import RunStatus from dbt.tests.util import ( check_table_does_exist, mkdir, @@ -135,9 +136,25 @@ def test_run_operation_local_macro(self, project): run_dbt(["deps"]) results, log_output = run_dbt_and_capture(["run-operation", "something_cool"]) + + for result in results: + if result.status == RunStatus.Skipped: + continue + + timing_keys = [timing.name for timing in result.timing] + assert timing_keys == ["compile", "execute"] + assert "something cool" in log_output results, log_output = run_dbt_and_capture(["run-operation", "pkg.something_cool"]) + + for result in results: + if result.status == RunStatus.Skipped: + continue + + timing_keys = [timing.name for timing in result.timing] + assert timing_keys == ["compile", "execute"] + assert "something cool" in log_output rm_dir("pkg") diff --git a/tests/functional/saved_queries/fixtures.py b/tests/functional/saved_queries/fixtures.py index 77735c9d1a9..58ed73c81b0 100644 --- a/tests/functional/saved_queries/fixtures.py +++ b/tests/functional/saved_queries/fixtures.py @@ -11,11 +11,15 @@ metrics: - simple_metric group_by: - - "Dimension('user__ds')" + - "Dimension('id__ds')" where: - - "{{ Dimension('user__ds', 'DAY') }} <= now()" - - "{{ Dimension('user__ds', 'DAY') }} >= '2023-01-01'" + - "{{ TimeDimension('id__ds', 'DAY') }} <= now()" + - "{{ TimeDimension('id__ds', 'DAY') }} >= '2023-01-01'" - "{{ Metric('txn_revenue', ['id']) }} > 1" + order_by: + - "Metric('simple_metric')" + - "Dimension('id__ds')" + limit: 10 exports: - name: my_export config: @@ -33,10 +37,10 @@ metrics: - simple_metric group_by: - - "Dimension('user__ds')" + - "Dimension('id__ds')" where: - - "{{ Dimension('user__ds', 'DAY') }} <= now()" - - "{{ Dimension('user__ds', 'DAY') }} >= '2023-01-01'" + - "{{ TimeDimension('id__ds', 'DAY') }} <= now()" + - "{{ TimeDimension('id__ds', 'DAY') }} >= '2023-01-01'" - "{{ Metric('txn_revenue', ['id']) }} > 1" exports: - name: my_export @@ -54,10 +58,10 @@ metrics: - simple_metric group_by: - - "Dimension('user__ds')" + - "Dimension('id__ds')" where: - - "{{ Dimension('user__ds', 'DAY') }} <= now()" - - "{{ Dimension('user__ds', 'DAY') }} >= '2023-01-01'" + - "{{ TimeDimension('id__ds', 'DAY') }} <= now()" + - "{{ TimeDimension('id__ds', 'DAY') }} >= '2023-01-01'" exports: - name: my_export config: @@ -72,8 +76,8 @@ metrics: - simple_metric group_by: - - "Dimension('user__ds')" - where: "{{ Dimension('user__ds', 'DAY') }} <= now()" + - "Dimension('id__ds')" + where: "{{ TimeDimension('id__ds', 'DAY') }} <= now()" """ saved_query_with_extra_config_attributes_yml = """ @@ -85,10 +89,10 @@ metrics: - simple_metric group_by: - - "Dimension('user__ds')" + - "Dimension('id__ds')" where: - - "{{ Dimension('user__ds', 'DAY') }} <= now()" - - "{{ Dimension('user__ds', 'DAY') }} >= '2023-01-01'" + - "{{ TimeDimension('id__ds', 'DAY') }} <= now()" + - "{{ TimeDimension('id__ds', 'DAY') }} >= '2023-01-01'" exports: - name: my_export config: @@ -108,10 +112,10 @@ metrics: - simple_metric group_by: - - "Dimension('user__ds')" + - "Dimension('id__ds')" where: - - "{{ Dimension('user__ds', 'DAY') }} <= now()" - - "{{ Dimension('user__ds', 'DAY') }} >= '2023-01-01'" + - "{{ TimeDimension('id__ds', 'DAY') }} <= now()" + - "{{ TimeDimension('id__ds', 'DAY') }} >= '2023-01-01'" exports: - name: my_export config: @@ -129,10 +133,10 @@ metrics: - simple_metric group_by: - - "Dimension('user__ds')" + - "Dimension('id__ds')" where: - - "{{ Dimension('user__ds', 'DAY') }} <= now()" - - "{{ Dimension('user__ds', 'DAY') }} >= '2023-01-01'" + - "{{ TimeDimension('id__ds', 'DAY') }} <= now()" + - "{{ TimeDimension('id__ds', 'DAY') }} >= '2023-01-01'" exports: - name: my_export """ @@ -149,10 +153,10 @@ metrics: - simple_metric group_by: - - "Dimension('user__ds')" + - "Dimension('id__ds')" where: - - "{{ Dimension('user__ds', 'DAY') }} <= now()" - - "{{ Dimension('user__ds', 'DAY') }} >= '2023-01-01'" + - "{{ TimeDimension('id__ds', 'DAY') }} <= now()" + - "{{ TimeDimension('id__ds', 'DAY') }} >= '2023-01-01'" exports: - name: my_export config: diff --git a/tests/functional/saved_queries/test_saved_query_build.py b/tests/functional/saved_queries/test_saved_query_build.py index dc89521f380..e9c2bbda3f8 100644 --- a/tests/functional/saved_queries/test_saved_query_build.py +++ b/tests/functional/saved_queries/test_saved_query_build.py @@ -12,7 +12,7 @@ ) -class TestSavedQueryBuildNoOp: +class TestSavedQueryBuild: @pytest.fixture(scope="class") def models(self): return { @@ -31,7 +31,8 @@ def packages(self): version: 1.1.1 """ - def test_build_saved_queries(self, project): + def test_build_saved_queries_no_op(self, project) -> None: + """Test building saved query exports with no flag, so should be no-op.""" run_dbt(["deps"]) result = run_dbt(["build"]) assert len(result.results) == 3 diff --git a/tests/functional/saved_queries/test_saved_query_parsing.py b/tests/functional/saved_queries/test_saved_query_parsing.py index b5333bb770b..f00f7366562 100644 --- a/tests/functional/saved_queries/test_saved_query_parsing.py +++ b/tests/functional/saved_queries/test_saved_query_parsing.py @@ -1,5 +1,6 @@ import os import shutil +from copy import deepcopy from typing import List import pytest @@ -36,6 +37,17 @@ def models(self): "docs.md": saved_query_description, } + @pytest.fixture(scope="class") + def other_schema(self, unique_schema): + return unique_schema + "_other" + + @pytest.fixture(scope="class") + def profiles_config_update(self, dbt_profile_target, unique_schema, other_schema): + outputs = {"default": dbt_profile_target, "prod": deepcopy(dbt_profile_target)} + outputs["default"]["schema"] = unique_schema + outputs["prod"]["schema"] = other_schema + return {"test": {"outputs": outputs, "target": "default"}} + def copy_state(self): if not os.path.exists("state"): os.makedirs("state") @@ -54,12 +66,21 @@ def test_semantic_model_parsing(self, project): assert len(saved_query.query_params.group_by) == 1 assert len(saved_query.query_params.where.where_filters) == 3 assert len(saved_query.depends_on.nodes) == 1 + + assert len(saved_query.query_params.order_by) == 2 + assert saved_query.query_params.limit is not None + assert saved_query.description == "My SavedQuery Description" assert len(saved_query.exports) == 1 assert saved_query.exports[0].name == "my_export" assert saved_query.exports[0].config.alias == "my_export_alias" assert saved_query.exports[0].config.export_as == ExportDestinationType.TABLE assert saved_query.exports[0].config.schema_name == "my_export_schema_name" + assert saved_query.exports[0].unrendered_config == { + "alias": "my_export_alias", + "export_as": "table", + "schema": "my_export_schema_name", + } # Save state self.copy_state() @@ -86,6 +107,86 @@ def test_semantic_model_parsing(self, project): results = run_dbt(["ls", "--select", "state:modified", "--state", "./state"]) assert len(results) == 1 + def test_semantic_model_parsing_change_export(self, project, other_schema): + runner = dbtTestRunner() + result = runner.invoke(["parse", "--no-partial-parse"]) + assert result.success + assert isinstance(result.result, Manifest) + manifest = result.result + assert len(manifest.saved_queries) == 1 + saved_query = manifest.saved_queries["saved_query.test.test_saved_query"] + assert saved_query.name == "test_saved_query" + assert saved_query.exports[0].name == "my_export" + + # Save state + self.copy_state() + # Nothing has changed, so no state:modified results + results = run_dbt(["ls", "--select", "state:modified", "--state", "./state"]) + assert len(results) == 0 + + # Change export name + write_file( + saved_queries_yml.replace("name: my_export", "name: my_expor2"), + project.project_root, + "models", + "saved_queries.yml", + ) + # State modified finds changed saved_query + results = run_dbt(["ls", "--select", "state:modified", "--state", "./state"]) + assert len(results) == 1 + + # Change export schema + write_file( + saved_queries_yml.replace( + "schema: my_export_schema_name", "schema: my_export_schema_name2" + ), + project.project_root, + "models", + "saved_queries.yml", + ) + # State modified finds changed saved_query + results = run_dbt(["ls", "--select", "state:modified", "--state", "./state"]) + assert len(results) == 1 + + def test_semantic_model_parsing_with_default_schema(self, project, other_schema): + write_file( + saved_queries_with_defaults_yml, project.project_root, "models", "saved_queries.yml" + ) + runner = dbtTestRunner() + result = runner.invoke(["parse", "--no-partial-parse", "--target", "prod"]) + assert result.success + assert isinstance(result.result, Manifest) + manifest = result.result + assert len(manifest.saved_queries) == 1 + saved_query = manifest.saved_queries["saved_query.test.test_saved_query"] + assert saved_query.name == "test_saved_query" + assert len(saved_query.query_params.metrics) == 1 + assert len(saved_query.query_params.group_by) == 1 + assert len(saved_query.query_params.where.where_filters) == 3 + assert len(saved_query.depends_on.nodes) == 1 + assert saved_query.description == "My SavedQuery Description" + assert len(saved_query.exports) == 1 + assert saved_query.exports[0].name == "my_export" + assert saved_query.exports[0].config.alias == "my_export_alias" + assert saved_query.exports[0].config.export_as == ExportDestinationType.TABLE + assert saved_query.exports[0].config.schema_name == other_schema + assert saved_query.exports[0].unrendered_config == { + "alias": "my_export_alias", + "export_as": "table", + } + + # Save state + self.copy_state() + # Nothing has changed, so no state:modified results + results = run_dbt( + ["ls", "--select", "state:modified", "--state", "./state", "--target", "prod"] + ) + assert len(results) == 0 + + # There should also be no state:modified results when using the default schema + results = run_dbt(["ls", "--select", "state:modified", "--state", "./state"]) + assert len(results) == 0 + def test_saved_query_error(self, project): error_schema_yml = saved_queries_yml.replace("simple_metric", "metric_not_found") write_file(error_schema_yml, project.project_root, "models", "saved_queries.yml") @@ -125,14 +226,14 @@ def test_saved_query_filter_types(self, project): where_filter.where_sql_template for where_filter in saved_query1.query_params.where.where_filters } == { - "{{ Dimension('user__ds', 'DAY') }} <= now()", - "{{ Dimension('user__ds', 'DAY') }} >= '2023-01-01'", + "{{ TimeDimension('id__ds', 'DAY') }} <= now()", + "{{ TimeDimension('id__ds', 'DAY') }} >= '2023-01-01'", } # String filter assert len(saved_query2.query_params.where.where_filters) == 1 assert ( saved_query2.query_params.where.where_filters[0].where_sql_template - == "{{ Dimension('user__ds', 'DAY') }} <= now()" + == "{{ TimeDimension('id__ds', 'DAY') }} <= now()" ) def test_saved_query_metrics_changed(self, project): diff --git a/tests/functional/schema_tests/data_test_config.py b/tests/functional/schema_tests/data_test_config.py index 377f14aac04..a6a16818000 100644 --- a/tests/functional/schema_tests/data_test_config.py +++ b/tests/functional/schema_tests/data_test_config.py @@ -2,6 +2,8 @@ import pytest +from dbt.contracts.graph.manifest import Manifest +from dbt.contracts.graph.nodes import TestNode from dbt.exceptions import CompilationError from dbt.tests.util import get_manifest, run_dbt from tests.functional.schema_tests.fixtures import ( @@ -10,14 +12,44 @@ same_key_error_yml, seed_csv, table_sql, + test_custom_color_from_config, ) +def _select_test_node(manifest: Manifest, pattern: re.Pattern[str]): + # Find the test_id dynamically + test_id = None + for node_id in manifest.nodes: + if pattern.match(node_id): + test_id = node_id + break + + # Ensure the test_id was found + assert test_id is not None, "Test ID matching the pattern was not found in the manifest nodes" + return manifest.nodes[test_id] + + +def get_table_persistence(project, table_name): + sql = f""" + SELECT + relpersistence + FROM pg_class + WHERE relname like '%{table_name}%' + """ + result = project.run_sql(sql, fetch="one") + assert len(result) == 1 + return result[0] + + class BaseDataTestsConfig: @pytest.fixture(scope="class") def seeds(self): return {"seed.csv": seed_csv} + @pytest.fixture(scope="class") + def macros(self): + return {"custom_color_from_config.sql": test_custom_color_from_config} + @pytest.fixture(scope="class") def project_config_update(self): return { @@ -35,29 +67,24 @@ def models(self): return {"table.sql": table_sql, "custom_config.yml": custom_config_yml} def test_custom_config(self, project): - run_dbt(["parse"]) - manifest = get_manifest(project.project_root) + run_dbt(["run"]) + run_dbt(["test", "--log-level", "debug"], expect_pass=False) + manifest = get_manifest(project.project_root) # Pattern to match the test_id without the specific suffix pattern = re.compile(r"test\.test\.accepted_values_table_color__blue__red\.\d+") - # Find the test_id dynamically - test_id = None - for node_id in manifest.nodes: - if pattern.match(node_id): - test_id = node_id - break - - # Ensure the test_id was found - assert ( - test_id is not None - ), "Test ID matching the pattern was not found in the manifest nodes" - + test_node: TestNode = _select_test_node(manifest, pattern) # Proceed with the assertions - test_node = manifest.nodes[test_id] assert "custom_config_key" in test_node.config assert test_node.config["custom_config_key"] == "some_value" + # pattern = re.compile(r"test\.test\.custom_color_from_config.*") + # test_node = _select_test_node(manifest, pattern) + persistence = get_table_persistence(project, "custom_color_from_config_table_color") + + assert persistence == "u" + class TestMixedDataTestConfig(BaseDataTestsConfig): @pytest.fixture(scope="class") diff --git a/tests/functional/schema_tests/fixtures.py b/tests/functional/schema_tests/fixtures.py index bf16148e0c7..ac266545712 100644 --- a/tests/functional/schema_tests/fixtures.py +++ b/tests/functional/schema_tests/fixtures.py @@ -1285,6 +1285,12 @@ values: ['blue', 'red'] config: custom_config_key: some_value + - custom_color_from_config: + severity: error + config: + test_color: orange + store_failures: true + unlogged: True """ mixed_config_yml = """ @@ -1296,9 +1302,13 @@ data_tests: - accepted_values: values: ['blue', 'red'] - severity: warn config: custom_config_key: some_value + severity: warn + - custom_color_from_config: + severity: error + config: + test_color: blue """ same_key_error_yml = """ @@ -1327,9 +1337,16 @@ 8,green,80 9,yellow,90 10,blue,100 -""" +""".strip() table_sql = """ -- content of the table.sql select * from {{ ref('seed') }} """ + +test_custom_color_from_config = """ +{% test custom_color_from_config(model, column_name) %} + select * from {{ model }} + where color = '{{ config.get('test_color') }}' +{% endtest %} +""" diff --git a/tests/functional/schema_tests/test_schema_v2_tests.py b/tests/functional/schema_tests/test_schema_v2_tests.py index ea33e62bce3..519ff64e853 100644 --- a/tests/functional/schema_tests/test_schema_v2_tests.py +++ b/tests/functional/schema_tests/test_schema_v2_tests.py @@ -474,9 +474,9 @@ def test_hooks_do_run_for_tests( ): # This passes now that hooks run, a behavior we changed in v1.0 results = run_dbt(["test", "--model", "ephemeral"]) - assert len(results) == 1 + assert len(results) == 3 for result in results: - assert result.status == "pass" + assert result.status in ("pass", "success") assert not result.skipped assert result.failures == 0, "test {} failed".format(result.node.name) @@ -507,9 +507,9 @@ def test_these_hooks_dont_run_for_tests( ): # This would fail if the hooks ran results = run_dbt(["test", "--model", "ephemeral"]) - assert len(results) == 1 + assert len(results) == 3 for result in results: - assert result.status == "pass" + assert result.status in ("pass", "success") assert not result.skipped assert result.failures == 0, "test {} failed".format(result.node.name) diff --git a/tests/functional/semantic_models/fixtures.py b/tests/functional/semantic_models/fixtures.py index 46b7fd86d06..7788067e91d 100644 --- a/tests/functional/semantic_models/fixtures.py +++ b/tests/functional/semantic_models/fixtures.py @@ -240,6 +240,12 @@ agg: sum agg_time_dimension: ds create_metric: true + - name: txn_revenue_with_label + label: "Transaction Revenue with label" + expr: revenue + agg: sum + agg_time_dimension: ds + create_metric: true - name: sum_of_things expr: 2 agg: sum diff --git a/tests/functional/semantic_models/test_semantic_model_parsing.py b/tests/functional/semantic_models/test_semantic_model_parsing.py index e89c41f9d8d..607caafa697 100644 --- a/tests/functional/semantic_models/test_semantic_model_parsing.py +++ b/tests/functional/semantic_models/test_semantic_model_parsing.py @@ -38,11 +38,14 @@ def test_semantic_model_parsing(self, project): semantic_model.node_relation.relation_name == f'"dbt"."{project.test_schema}"."fct_revenue"' ) - assert len(semantic_model.measures) == 6 - # manifest should have one metric (that was created from a measure) - assert len(manifest.metrics) == 2 + assert len(semantic_model.measures) == 7 + # manifest should have two metrics created from measures + assert len(manifest.metrics) == 3 metric = manifest.metrics["metric.test.txn_revenue"] assert metric.name == "txn_revenue" + metric_with_label = manifest.metrics["metric.test.txn_revenue_with_label"] + assert metric_with_label.name == "txn_revenue_with_label" + assert metric_with_label.label == "Transaction Revenue with label" def test_semantic_model_error(self, project): # Next, modify the default schema.yml to remove the semantic model. @@ -107,6 +110,7 @@ def test_semantic_model_deleted_partial_parsing(self, project): def test_semantic_model_flipping_create_metric_partial_parsing(self, project): generated_metric = "metric.test.txn_revenue" + generated_metric_with_label = "metric.test.txn_revenue_with_label" # First, use the default schema.yml to define our semantic model, and # run the dbt parse command write_file(schema_yml, project.project_root, "models", "schema.yml") @@ -117,6 +121,11 @@ def test_semantic_model_flipping_create_metric_partial_parsing(self, project): # Verify the metric created by `create_metric: true` exists metric = result.result.metrics[generated_metric] assert metric.name == "txn_revenue" + assert metric.label == "txn_revenue" + + metric_with_label = result.result.metrics[generated_metric_with_label] + assert metric_with_label.name == "txn_revenue_with_label" + assert metric_with_label.label == "Transaction Revenue with label" # --- Next, modify the default schema.yml to have no `create_metric: true` --- no_create_metric_schema_yml = schema_yml.replace( diff --git a/tests/functional/show/test_show.py b/tests/functional/show/test_show.py index d35291458ee..e795e92ffac 100644 --- a/tests/functional/show/test_show.py +++ b/tests/functional/show/test_show.py @@ -155,6 +155,33 @@ def test_inline_fail_database_error(self, project): run_dbt(["show", "--inline", "slect asdlkjfsld;j"]) +class TestShowInlineDirect(ShowBase): + + def test_inline_direct_pass(self, project): + query = f"select * from {project.test_schema}.sample_seed" + (_, log_output) = run_dbt_and_capture(["show", "--inline-direct", query]) + assert "Previewing inline node" in log_output + assert "sample_num" in log_output + assert "sample_bool" in log_output + + # This is a bit of a hack. Unfortunately, the test teardown code + # expects that dbt loaded an adapter with a macro context the last + # time it was called. The '--inline-direct' parameter used on the + # previous run explicitly disables macros. So now we call 'dbt seed', + # which will load the adapter fully and satisfy the teardown code. + run_dbt(["seed"]) + + +class TestShowInlineDirectFail(ShowBase): + + def test_inline_fail_database_error(self, project): + with pytest.raises(DbtRuntimeError, match="Database Error"): + run_dbt(["show", "--inline-direct", "slect asdlkjfsld;j"]) + + # See prior test for explanation of why this is here + run_dbt(["seed"]) + + class TestShowEphemeral(ShowBase): def test_ephemeral_model(self, project): run_dbt(["build"]) diff --git a/tests/functional/snapshots/data/seed_cn.sql b/tests/functional/snapshots/data/seed_cn.sql new file mode 100644 index 00000000000..089200afa47 --- /dev/null +++ b/tests/functional/snapshots/data/seed_cn.sql @@ -0,0 +1,82 @@ +create table {database}.{schema}.seed ( + id INTEGER, + first_name VARCHAR(50), + last_name VARCHAR(50), + email VARCHAR(50), + gender VARCHAR(50), + ip_address VARCHAR(20), + updated_at TIMESTAMP WITHOUT TIME ZONE +); + +create table {database}.{schema}.snapshot_expected ( + id INTEGER, + first_name VARCHAR(50), + last_name VARCHAR(50), + email VARCHAR(50), + gender VARCHAR(50), + ip_address VARCHAR(20), + + -- snapshotting fields + updated_at TIMESTAMP WITHOUT TIME ZONE, + test_valid_from TIMESTAMP WITHOUT TIME ZONE, + test_valid_to TIMESTAMP WITHOUT TIME ZONE, + test_scd_id TEXT, + test_updated_at TIMESTAMP WITHOUT TIME ZONE +); + + +-- seed inserts +-- use the same email for two users to verify that duplicated check_cols values +-- are handled appropriately +insert into {database}.{schema}.seed (id, first_name, last_name, email, gender, ip_address, updated_at) values +(1, 'Judith', 'Kennedy', '(not provided)', 'Female', '54.60.24.128', '2015-12-24 12:19:28'), +(2, 'Arthur', 'Kelly', '(not provided)', 'Male', '62.56.24.215', '2015-10-28 16:22:15'), +(3, 'Rachel', 'Moreno', 'rmoreno2@msu.edu', 'Female', '31.222.249.23', '2016-04-05 02:05:30'), +(4, 'Ralph', 'Turner', 'rturner3@hp.com', 'Male', '157.83.76.114', '2016-08-08 00:06:51'), +(5, 'Laura', 'Gonzales', 'lgonzales4@howstuffworks.com', 'Female', '30.54.105.168', '2016-09-01 08:25:38'), +(6, 'Katherine', 'Lopez', 'klopez5@yahoo.co.jp', 'Female', '169.138.46.89', '2016-08-30 18:52:11'), +(7, 'Jeremy', 'Hamilton', 'jhamilton6@mozilla.org', 'Male', '231.189.13.133', '2016-07-17 02:09:46'), +(8, 'Heather', 'Rose', 'hrose7@goodreads.com', 'Female', '87.165.201.65', '2015-12-29 22:03:56'), +(9, 'Gregory', 'Kelly', 'gkelly8@trellian.com', 'Male', '154.209.99.7', '2016-03-24 21:18:16'), +(10, 'Rachel', 'Lopez', 'rlopez9@themeforest.net', 'Female', '237.165.82.71', '2016-08-20 15:44:49'), +(11, 'Donna', 'Welch', 'dwelcha@shutterfly.com', 'Female', '103.33.110.138', '2016-02-27 01:41:48'), +(12, 'Russell', 'Lawrence', 'rlawrenceb@qq.com', 'Male', '189.115.73.4', '2016-06-11 03:07:09'), +(13, 'Michelle', 'Montgomery', 'mmontgomeryc@scientificamerican.com', 'Female', '243.220.95.82', '2016-06-18 16:27:19'), +(14, 'Walter', 'Castillo', 'wcastillod@pagesperso-orange.fr', 'Male', '71.159.238.196', '2016-10-06 01:55:44'), +(15, 'Robin', 'Mills', 'rmillse@vkontakte.ru', 'Female', '172.190.5.50', '2016-10-31 11:41:21'), +(16, 'Raymond', 'Holmes', 'rholmesf@usgs.gov', 'Male', '148.153.166.95', '2016-10-03 08:16:38'), +(17, 'Gary', 'Bishop', 'gbishopg@plala.or.jp', 'Male', '161.108.182.13', '2016-08-29 19:35:20'), +(18, 'Anna', 'Riley', 'arileyh@nasa.gov', 'Female', '253.31.108.22', '2015-12-11 04:34:27'), +(19, 'Sarah', 'Knight', 'sknighti@foxnews.com', 'Female', '222.220.3.177', '2016-09-26 00:49:06'), +(20, 'Phyllis', 'Fox', null, 'Female', '163.191.232.95', '2016-08-21 10:35:19'); + + +-- populate snapshot table +insert into {database}.{schema}.snapshot_expected ( + id, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + test_valid_from, + test_valid_to, + test_updated_at, + test_scd_id +) + +select + id, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + -- fields added by snapshotting + updated_at as test_valid_from, + null::timestamp as test_valid_to, + updated_at as test_updated_at, + md5(id || '-' || first_name || '|' || updated_at::text) as test_scd_id +from {database}.{schema}.seed; diff --git a/tests/functional/snapshots/data/seed_dbt_valid_to.sql b/tests/functional/snapshots/data/seed_dbt_valid_to.sql new file mode 100644 index 00000000000..9627151042c --- /dev/null +++ b/tests/functional/snapshots/data/seed_dbt_valid_to.sql @@ -0,0 +1,82 @@ +create table {database}.{schema}.seed ( + id INTEGER, + first_name VARCHAR(50), + last_name VARCHAR(50), + email VARCHAR(50), + gender VARCHAR(50), + ip_address VARCHAR(20), + updated_at TIMESTAMP WITHOUT TIME ZONE +); + +create table {database}.{schema}.snapshot_expected ( + id INTEGER, + first_name VARCHAR(50), + last_name VARCHAR(50), + email VARCHAR(50), + gender VARCHAR(50), + ip_address VARCHAR(20), + + -- snapshotting fields + updated_at TIMESTAMP WITHOUT TIME ZONE, + test_valid_from TIMESTAMP WITHOUT TIME ZONE, + test_valid_to TIMESTAMP WITHOUT TIME ZONE, + test_scd_id TEXT, + test_updated_at TIMESTAMP WITHOUT TIME ZONE +); + + +-- seed inserts +-- use the same email for two users to verify that duplicated check_cols values +-- are handled appropriately +insert into {database}.{schema}.seed (id, first_name, last_name, email, gender, ip_address, updated_at) values +(1, 'Judith', 'Kennedy', '(not provided)', 'Female', '54.60.24.128', '2015-12-24 12:19:28'), +(2, 'Arthur', 'Kelly', '(not provided)', 'Male', '62.56.24.215', '2015-10-28 16:22:15'), +(3, 'Rachel', 'Moreno', 'rmoreno2@msu.edu', 'Female', '31.222.249.23', '2016-04-05 02:05:30'), +(4, 'Ralph', 'Turner', 'rturner3@hp.com', 'Male', '157.83.76.114', '2016-08-08 00:06:51'), +(5, 'Laura', 'Gonzales', 'lgonzales4@howstuffworks.com', 'Female', '30.54.105.168', '2016-09-01 08:25:38'), +(6, 'Katherine', 'Lopez', 'klopez5@yahoo.co.jp', 'Female', '169.138.46.89', '2016-08-30 18:52:11'), +(7, 'Jeremy', 'Hamilton', 'jhamilton6@mozilla.org', 'Male', '231.189.13.133', '2016-07-17 02:09:46'), +(8, 'Heather', 'Rose', 'hrose7@goodreads.com', 'Female', '87.165.201.65', '2015-12-29 22:03:56'), +(9, 'Gregory', 'Kelly', 'gkelly8@trellian.com', 'Male', '154.209.99.7', '2016-03-24 21:18:16'), +(10, 'Rachel', 'Lopez', 'rlopez9@themeforest.net', 'Female', '237.165.82.71', '2016-08-20 15:44:49'), +(11, 'Donna', 'Welch', 'dwelcha@shutterfly.com', 'Female', '103.33.110.138', '2016-02-27 01:41:48'), +(12, 'Russell', 'Lawrence', 'rlawrenceb@qq.com', 'Male', '189.115.73.4', '2016-06-11 03:07:09'), +(13, 'Michelle', 'Montgomery', 'mmontgomeryc@scientificamerican.com', 'Female', '243.220.95.82', '2016-06-18 16:27:19'), +(14, 'Walter', 'Castillo', 'wcastillod@pagesperso-orange.fr', 'Male', '71.159.238.196', '2016-10-06 01:55:44'), +(15, 'Robin', 'Mills', 'rmillse@vkontakte.ru', 'Female', '172.190.5.50', '2016-10-31 11:41:21'), +(16, 'Raymond', 'Holmes', 'rholmesf@usgs.gov', 'Male', '148.153.166.95', '2016-10-03 08:16:38'), +(17, 'Gary', 'Bishop', 'gbishopg@plala.or.jp', 'Male', '161.108.182.13', '2016-08-29 19:35:20'), +(18, 'Anna', 'Riley', 'arileyh@nasa.gov', 'Female', '253.31.108.22', '2015-12-11 04:34:27'), +(19, 'Sarah', 'Knight', 'sknighti@foxnews.com', 'Female', '222.220.3.177', '2016-09-26 00:49:06'), +(20, 'Phyllis', 'Fox', null, 'Female', '163.191.232.95', '2016-08-21 10:35:19'); + + +-- populate snapshot table +insert into {database}.{schema}.snapshot_expected ( + id, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + test_valid_from, + test_valid_to, + test_updated_at, + test_scd_id +) + +select + id, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + -- fields added by snapshotting + updated_at as test_valid_from, + date('2099-12-31') as test_valid_to, + updated_at as test_updated_at, + md5(id || '-' || first_name || '|' || updated_at::text) as test_scd_id +from {database}.{schema}.seed; diff --git a/tests/functional/snapshots/fixtures.py b/tests/functional/snapshots/fixtures.py index a94f0c04875..562fcb4b10f 100644 --- a/tests/functional/snapshots/fixtures.py +++ b/tests/functional/snapshots/fixtures.py @@ -291,6 +291,30 @@ {% endsnapshot %} """ +snapshots_pg__snapshot_yml = """ +snapshots: + - name: snapshot_actual + relation: "ref('seed')" + config: + unique_key: "id || '-' || first_name" + strategy: timestamp + updated_at: updated_at + meta: + owner: 'a_owner' +""" + +snapshots_pg__snapshot_mod_yml = """ +snapshots: + - name: snapshot_actual + relation: "ref('seed')" + config: + unique_key: "id || '-' || first_name" + strategy: timestamp + updated_at: updated_at + meta: + owner: 'b_owner' +""" + snapshots_pg__snapshot_no_target_schema_sql = """ {% snapshot snapshot_actual %} diff --git a/tests/functional/snapshots/test_basic_snapshot.py b/tests/functional/snapshots/test_basic_snapshot.py index ac6c3831642..5300f921971 100644 --- a/tests/functional/snapshots/test_basic_snapshot.py +++ b/tests/functional/snapshots/test_basic_snapshot.py @@ -8,6 +8,7 @@ check_relations_equal, relation_from_name, run_dbt, + update_config_file, write_file, ) from tests.functional.snapshots.fixtures import ( @@ -18,8 +19,10 @@ models__schema_yml, seeds__seed_csv, seeds__seed_newcol_csv, + snapshots_pg__snapshot_mod_yml, snapshots_pg__snapshot_no_target_schema_sql, snapshots_pg__snapshot_sql, + snapshots_pg__snapshot_yml, snapshots_pg_custom__snapshot_sql, snapshots_pg_custom_namespaced__snapshot_sql, ) @@ -372,3 +375,55 @@ def test_updated_at_snapshot(self, project): class TestRefUpdatedAtCheckCols(UpdatedAtCheckCols): def test_updated_at_ref(self, project): ref_setup(project, num_snapshot_models=2) + + +class BasicYaml(Basic): + @pytest.fixture(scope="class") + def snapshots(self): + """Overrides the same function in Basic to use the YAML method of + defining a snapshot.""" + return {"snapshot.yml": snapshots_pg__snapshot_yml} + + @pytest.fixture(scope="class") + def models(self): + """Overrides the same function in Basic to use a modified version of + schema.yml without snapshot config.""" + return { + "ref_snapshot.sql": models__ref_snapshot_sql, + } + + +class TestBasicSnapshotYaml(BasicYaml): + def test_basic_snapshot_yaml(self, project): + snapshot_setup(project, num_snapshot_models=1) + + +class TestYamlSnapshotPartialParsing(BasicYaml): + def test_snapshot_partial_parsing(self, project): + manifest = run_dbt(["parse"]) + snapshot_id = "snapshot.test.snapshot_actual" + assert snapshot_id in manifest.nodes + snapshot = manifest.nodes[snapshot_id] + assert snapshot.meta["owner"] == "a_owner" + + # change snapshot yml file and re-parse + write_file(snapshots_pg__snapshot_mod_yml, "snapshots", "snapshot.yml") + manifest = run_dbt(["parse"]) + snapshot = manifest.nodes[snapshot_id] + assert snapshot.meta["owner"] == "b_owner" + + # modify dbt_project.yml and re-parse + config_updates = { + "snapshots": { + "test": { + "+snapshot_meta_column_names": { + "dbt_valid_to": "test_valid_to", + "dbt_valid_from": "test_valid_from", + }, + } + } + } + update_config_file(config_updates, "dbt_project.yml") + manifest = run_dbt(["parse"]) + snapshot = manifest.nodes[snapshot_id] + assert snapshot.config.snapshot_meta_column_names.dbt_valid_to == "test_valid_to" diff --git a/tests/functional/snapshots/test_snapshot_column_names.py b/tests/functional/snapshots/test_snapshot_column_names.py new file mode 100644 index 00000000000..85e9f425765 --- /dev/null +++ b/tests/functional/snapshots/test_snapshot_column_names.py @@ -0,0 +1,234 @@ +import os + +import pytest + +from dbt.tests.util import ( + check_relations_equal, + get_manifest, + run_dbt, + run_dbt_and_capture, + update_config_file, +) + +snapshot_actual_sql = """ +{% snapshot snapshot_actual %} + + {{ + config( + unique_key='id || ' ~ "'-'" ~ ' || first_name', + ) + }} + + select * from {{target.database}}.{{target.schema}}.seed + +{% endsnapshot %} +""" + +snapshots_yml = """ +snapshots: + - name: snapshot_actual + config: + strategy: timestamp + updated_at: updated_at + snapshot_meta_column_names: + dbt_valid_to: test_valid_to + dbt_valid_from: test_valid_from + dbt_scd_id: test_scd_id + dbt_updated_at: test_updated_at +""" + +snapshots_no_column_names_yml = """ +snapshots: + - name: snapshot_actual + config: + strategy: timestamp + updated_at: updated_at +""" + +ref_snapshot_sql = """ +select * from {{ ref('snapshot_actual') }} +""" + + +invalidate_sql = """ +-- update records 11 - 21. Change email and updated_at field +update {schema}.seed set + updated_at = updated_at + interval '1 hour', + email = case when id = 20 then 'pfoxj@creativecommons.org' else 'new_' || email end +where id >= 10 and id <= 20; + + +-- invalidate records 11 - 21 +update {schema}.snapshot_expected set + test_valid_to = updated_at + interval '1 hour' +where id >= 10 and id <= 20; + +""" + +update_sql = """ +-- insert v2 of the 11 - 21 records + +insert into {database}.{schema}.snapshot_expected ( + id, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + test_valid_from, + test_valid_to, + test_updated_at, + test_scd_id +) + +select + id, + first_name, + last_name, + email, + gender, + ip_address, + updated_at, + -- fields added by snapshotting + updated_at as test_valid_from, + null::timestamp as test_valid_to, + updated_at as test_updated_at, + md5(id || '-' || first_name || '|' || updated_at::text) as test_scd_id +from {database}.{schema}.seed +where id >= 10 and id <= 20; +""" + + +class TestSnapshotColumnNames: + @pytest.fixture(scope="class") + def snapshots(self): + return {"snapshot.sql": snapshot_actual_sql} + + @pytest.fixture(scope="class") + def models(self): + return { + "snapshots.yml": snapshots_yml, + "ref_snapshot.sql": ref_snapshot_sql, + } + + def test_snapshot_column_names(self, project): + path = os.path.join(project.test_data_dir, "seed_cn.sql") + project.run_sql_file(path) + results = run_dbt(["snapshot"]) + assert len(results) == 1 + + project.run_sql(invalidate_sql) + project.run_sql(update_sql) + + results = run_dbt(["snapshot"]) + assert len(results) == 1 + + # run_dbt(["test"]) + check_relations_equal(project.adapter, ["snapshot_actual", "snapshot_expected"]) + + +class TestSnapshotColumnNamesFromDbtProject: + @pytest.fixture(scope="class") + def snapshots(self): + return {"snapshot.sql": snapshot_actual_sql} + + @pytest.fixture(scope="class") + def models(self): + return { + "snapshots.yml": snapshots_no_column_names_yml, + "ref_snapshot.sql": ref_snapshot_sql, + } + + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "snapshots": { + "test": { + "+snapshot_meta_column_names": { + "dbt_valid_to": "test_valid_to", + "dbt_valid_from": "test_valid_from", + "dbt_scd_id": "test_scd_id", + "dbt_updated_at": "test_updated_at", + } + } + } + } + + def test_snapshot_column_names_from_project(self, project): + path = os.path.join(project.test_data_dir, "seed_cn.sql") + project.run_sql_file(path) + results = run_dbt(["snapshot"]) + assert len(results) == 1 + + project.run_sql(invalidate_sql) + project.run_sql(update_sql) + + results = run_dbt(["snapshot"]) + assert len(results) == 1 + + # run_dbt(["test"]) + check_relations_equal(project.adapter, ["snapshot_actual", "snapshot_expected"]) + + +class TestSnapshotInvalidColumnNames: + @pytest.fixture(scope="class") + def snapshots(self): + return {"snapshot.sql": snapshot_actual_sql} + + @pytest.fixture(scope="class") + def models(self): + return { + "snapshots.yml": snapshots_no_column_names_yml, + "ref_snapshot.sql": ref_snapshot_sql, + } + + @pytest.fixture(scope="class") + def project_config_update(self): + return { + "snapshots": { + "test": { + "+snapshot_meta_column_names": { + "dbt_valid_to": "test_valid_to", + "dbt_valid_from": "test_valid_from", + "dbt_scd_id": "test_scd_id", + "dbt_updated_at": "test_updated_at", + } + } + } + } + + def test_snapshot_invalid_column_names(self, project): + path = os.path.join(project.test_data_dir, "seed_cn.sql") + project.run_sql_file(path) + results = run_dbt(["snapshot"]) + assert len(results) == 1 + manifest = get_manifest(project.project_root) + snapshot_node = manifest.nodes["snapshot.test.snapshot_actual"] + snapshot_node.config.snapshot_meta_column_names == { + "dbt_valid_to": "test_valid_to", + "dbt_valid_from": "test_valid_from", + "dbt_scd_id": "test_scd_id", + "dbt_updated_at": "test_updated_at", + } + + project.run_sql(invalidate_sql) + project.run_sql(update_sql) + + # Change snapshot_meta_columns and look for an error + different_columns = { + "snapshots": { + "test": { + "+snapshot_meta_column_names": { + "dbt_valid_to": "test_valid_to", + "dbt_updated_at": "test_updated_at", + } + } + } + } + update_config_file(different_columns, "dbt_project.yml") + + results, log_output = run_dbt_and_capture(["snapshot"], expect_pass=False) + assert len(results) == 1 + assert "Compilation Error in snapshot snapshot_actual" in log_output + assert "Snapshot target is missing configured columns" in log_output diff --git a/tests/functional/snapshots/test_snapshot_empty.py b/tests/functional/snapshots/test_snapshot_empty.py new file mode 100644 index 00000000000..a3f648a968f --- /dev/null +++ b/tests/functional/snapshots/test_snapshot_empty.py @@ -0,0 +1,40 @@ +import pytest + +from dbt.tests.util import run_dbt + +my_model_sql = """ +select 1 as id, {{ dbt.current_timestamp() }} as updated_at +""" + +snapshots_yml = """ +snapshots: + - name: my_snapshot + relation: "ref('my_model')" + config: + unique_key: id + strategy: check + check_cols: all + dbt_valid_to_current: "date('9999-12-31')" +""" + + +class TestSnapshotEmpty: + @pytest.fixture(scope="class") + def models(self): + return { + "my_model.sql": my_model_sql, + "snapshots.yml": snapshots_yml, + } + + def test_check(self, project): + run_dbt(["run"]) + run_dbt(["snapshot", "--empty"]) + + query = "select id, updated_at, dbt_valid_from, dbt_valid_to from {database}.{schema}.my_snapshot order by updated_at asc" + snapshot_out1 = project.run_sql(query, fetch="all") + assert snapshot_out1 == [] + + run_dbt(["run"]) + run_dbt(["snapshot", "--empty"]) + snapshot_out2 = project.run_sql(query, fetch="all") + assert snapshot_out2 == [] diff --git a/tests/functional/snapshots/test_snapshot_timestamps.py b/tests/functional/snapshots/test_snapshot_timestamps.py new file mode 100644 index 00000000000..a0faa6d06a1 --- /dev/null +++ b/tests/functional/snapshots/test_snapshot_timestamps.py @@ -0,0 +1,72 @@ +import pytest + +from dbt.tests.util import run_dbt, run_dbt_and_capture + +create_source_sql = """ +create table {database}.{schema}.source_users ( + id INTEGER, + first_name VARCHAR(50), + last_name VARCHAR(50), + email VARCHAR(50), + gender VARCHAR(50), + ip_address VARCHAR(20), + updated_time TIMESTAMP WITH TIME ZONE +); +insert into {database}.{schema}.source_users (id, first_name, last_name, email, gender, ip_address, updated_time) values +(1, 'Judith', 'Kennedy', '(not provided)', 'Female', '54.60.24.128', '2015-12-24 12:19:28'), +(2, 'Arthur', 'Kelly', '(not provided)', 'Male', '62.56.24.215', '2015-10-28 16:22:15'), +(3, 'Rachel', 'Moreno', 'rmoreno2@msu.edu', 'Female', '31.222.249.23', '2016-04-05 02:05:30'); +""" + +model_users_sql = """ +select * from {{ source('test_source', 'source_users') }} +""" + +snapshot_sql = """ +{% snapshot users_snapshot %} + +select * from {{ ref('users') }} + +{% endsnapshot %} +""" + +source_schema_yml = """ +sources: + - name: test_source + loader: custom + schema: "{{ target.schema }}" + tables: + - name: source_users + loaded_at_field: updated_time +""" + +snapshot_schema_yml = """ +snapshots: + - name: users_snapshot + config: + target_schema: "{{ target.schema }}" + strategy: timestamp + unique_key: id + updated_at: updated_time +""" + + +class TestSnapshotConfig: + @pytest.fixture(scope="class") + def models(self): + return { + "users.sql": model_users_sql, + "source_schema.yml": source_schema_yml, + "snapshot_schema.yml": snapshot_schema_yml, + } + + @pytest.fixture(scope="class") + def snapshots(self): + return {"snapshot.sql": snapshot_sql} + + def test_timestamp_snapshot(self, project): + project.run_sql(create_source_sql) + run_dbt(["run"]) + results, log_output = run_dbt_and_capture(["snapshot"]) + assert len(results) == 1 + assert "Please update snapshot config" in log_output diff --git a/tests/functional/sources/test_name_chars.py b/tests/functional/sources/test_name_chars.py new file mode 100644 index 00000000000..9e58687cf53 --- /dev/null +++ b/tests/functional/sources/test_name_chars.py @@ -0,0 +1,31 @@ +from dbt.tests.util import get_manifest, run_dbt, write_file +from tests.fixtures.jaffle_shop import JaffleShopProject + +# Note: in an actual file (as opposed to a string that we write into a files) +# there would only be a single backslash. +sources_yml = """ +sources: + - name: something_else + database: raw + schema: jaffle_shop + tables: + - name: "\\"/test/orders\\"" + - name: customers +""" + + +class TestNameChars(JaffleShopProject): + def test_quotes_in_table_names(self, project): + # Write out a sources definition that includes a table name with quotes and a forward slash + # Note: forward slashes are not legal in filenames in Linux (or Windows), + # so we won't see forward slashes in model names, because they come from file names. + write_file(sources_yml, project.project_root, "models", "sources.yml") + manifest = run_dbt(["parse"]) + assert len(manifest.sources) == 2 + assert 'source.jaffle_shop.something_else."/test/orders"' in manifest.sources.keys() + # We've written out the manifest.json artifact, we want to ensure + # that it can be read in again (the json is valid). + # Note: the key in the json actually looks like: "source.jaffle_shop.something_else.\"/test/orders\"" + new_manifest = get_manifest(project.project_root) + assert new_manifest + assert 'source.jaffle_shop.something_else."/test/orders"' in new_manifest.sources.keys() diff --git a/tests/functional/sources/test_source_freshness.py b/tests/functional/sources/test_source_freshness.py index 2f42a3aaa56..70c8866869e 100644 --- a/tests/functional/sources/test_source_freshness.py +++ b/tests/functional/sources/test_source_freshness.py @@ -129,12 +129,12 @@ def _assert_freshness_results(self, path, state): ] def _assert_project_hooks_called(self, logs: str): - assert "Running 1 on-run-start hook" in logs - assert "Running 1 on-run-end hook" in logs + assert "test.on-run-start.0" in logs + assert "test.on-run-start.0" in logs def _assert_project_hooks_not_called(self, logs: str): - assert "Running 1 on-run-start hook" not in logs - assert "Running 1 on-run-end hook" not in logs + assert "test.on-run-end.0" not in logs + assert "test.on-run-end.0" not in logs class TestSourceFreshness(SuccessfulSourceFreshnessTest): diff --git a/tests/functional/test_empty.py b/tests/functional/test_empty.py index d284fdcc3e5..7fa026a82eb 100644 --- a/tests/functional/test_empty.py +++ b/tests/functional/test_empty.py @@ -27,6 +27,14 @@ from {{ source('seed_sources', 'raw_source') }} """ +model_no_ephemeral_ref_sql = """ +select * +from {{ ref('model_input') }} +union all +select * +from {{ source('seed_sources', 'raw_source') }} +""" + schema_sources_yml = """ sources: @@ -36,6 +44,29 @@ - name: raw_source """ +unit_tests_yml = """ +unit_tests: + - name: test_my_model + model: model_no_ephemeral_ref + given: + - input: ref('model_input') + format: csv + rows: | + id + 1 + - input: source('seed_sources', 'raw_source') + format: csv + rows: | + id + 2 + expect: + format: csv + rows: | + id + 1 + 2 +""" + class TestEmptyFlag: @pytest.fixture(scope="class") @@ -50,7 +81,9 @@ def models(self): "model_input.sql": model_input_sql, "ephemeral_model_input.sql": ephemeral_model_input_sql, "model.sql": model_sql, + "model_no_ephemeral_ref.sql": model_no_ephemeral_ref_sql, "sources.yml": schema_sources_yml, + "unit_tests.yml": unit_tests_yml, } def assert_row_count(self, project, relation_name: str, expected_row_count: int): diff --git a/tests/functional/time_spines/fixtures.py b/tests/functional/time_spines/fixtures.py new file mode 100644 index 00000000000..0f67488ff11 --- /dev/null +++ b/tests/functional/time_spines/fixtures.py @@ -0,0 +1,105 @@ +models_people_sql = """ +select 1 as id, 'Drew' as first_name, 'Banin' as last_name, 'yellow' as favorite_color, true as loves_dbt, 5 as tenure, current_timestamp as created_at +union all +select 2 as id, 'Jeremy' as first_name, 'Cohen' as last_name, 'indigo' as favorite_color, true as loves_dbt, 4 as tenure, current_timestamp as created_at +union all +select 3 as id, 'Callum' as first_name, 'McCann' as last_name, 'emerald' as favorite_color, true as loves_dbt, 0 as tenure, current_timestamp as created_at +""" + +semantic_model_people_yml = """ +version: 2 + +semantic_models: + - name: semantic_people + model: ref('people') + dimensions: + - name: favorite_color + type: categorical + - name: created_at + type: TIME + type_params: + time_granularity: day + measures: + - name: years_tenure + agg: SUM + expr: tenure + - name: people + agg: count + expr: id + entities: + - name: id + type: primary + defaults: + agg_time_dimension: created_at +""" + +metricflow_time_spine_sql = """ +SELECT to_date('02/20/2023, 'mm/dd/yyyy') as date_day +""" + +metricflow_time_spine_second_sql = """ +SELECT to_datetime('02/20/2023, 'mm/dd/yyyy hh:mm:ss') as ts_second +""" + +valid_time_spines_yml = """ +version: 2 + +models: + - name: metricflow_time_spine_second + time_spine: + standard_granularity_column: ts_second + columns: + - name: ts_second + granularity: second + - name: metricflow_time_spine + time_spine: + standard_granularity_column: date_day + custom_granularities: + - name: retail_month + - name: martian_year + column_name: martian__year_xyz + columns: + - name: date_day + granularity: day + - name: retail_month + - name: martian__year_xyz + +""" + +missing_time_spine_yml = """ +models: + - name: metricflow_time_spine + columns: + - name: ts_second + granularity: second +""" + +time_spine_missing_granularity_yml = """ +models: + - name: metricflow_time_spine_second + time_spine: + standard_granularity_column: ts_second + columns: + - name: ts_second +""" + +time_spine_missing_standard_column_yml = """ +models: + - name: metricflow_time_spine_second + time_spine: + standard_granularity_column: ts_second + columns: + - name: date_day +""" + +time_spine_missing_custom_column_yml = """ +models: + - name: metricflow_time_spine_second + time_spine: + standard_granularity_column: date_day + custom_granularities: + - name: retail_month + columns: + - name: date_day + granularity: day +""" diff --git a/tests/functional/time_spines/test_time_spines.py b/tests/functional/time_spines/test_time_spines.py new file mode 100644 index 00000000000..03063d347be --- /dev/null +++ b/tests/functional/time_spines/test_time_spines.py @@ -0,0 +1,233 @@ +from typing import Set + +import pytest + +from dbt.cli.main import dbtRunner +from dbt.contracts.graph.manifest import Manifest +from dbt.contracts.graph.semantic_manifest import SemanticManifest +from dbt.exceptions import ParsingError +from dbt.tests.util import get_manifest +from dbt_semantic_interfaces.type_enums import TimeGranularity +from tests.functional.time_spines.fixtures import ( + metricflow_time_spine_second_sql, + metricflow_time_spine_sql, + models_people_sql, + semantic_model_people_yml, + time_spine_missing_custom_column_yml, + time_spine_missing_granularity_yml, + time_spine_missing_standard_column_yml, + valid_time_spines_yml, +) + + +class TestValidTimeSpines: + """Tests that YAML using current time spine configs parses as expected.""" + + @pytest.fixture(scope="class") + def models(self): + return { + "metricflow_time_spine.sql": metricflow_time_spine_sql, + "metricflow_time_spine_second.sql": metricflow_time_spine_second_sql, + "time_spines.yml": valid_time_spines_yml, + "semantic_model_people.yml": semantic_model_people_yml, + "people.sql": models_people_sql, + } + + def test_time_spines(self, project): + runner = dbtRunner() + result = runner.invoke(["parse"]) + assert result.success + assert isinstance(result.result, Manifest) + + manifest = get_manifest(project.project_root) + assert manifest + + # Test that models and columns are set as expected + time_spine_models = { + id.split(".")[-1]: node for id, node in manifest.nodes.items() if node.time_spine + } + day_model_name = "metricflow_time_spine" + second_model_name = "metricflow_time_spine_second" + day_column_name = "date_day" + second_column_name = "ts_second" + model_names_to_col_names = { + day_model_name: day_column_name, + second_model_name: second_column_name, + } + model_names_to_granularities = { + day_model_name: TimeGranularity.DAY, + second_model_name: TimeGranularity.SECOND, + } + assert len(time_spine_models) == 2 + expected_time_spine_aliases = {second_model_name, day_model_name} + assert set(time_spine_models.keys()) == expected_time_spine_aliases + for model in time_spine_models.values(): + assert ( + model.time_spine.standard_granularity_column + == model_names_to_col_names[model.name] + ) + if model.name == day_model_name: + assert len(model.time_spine.custom_granularities) == 2 + assert { + custom_granularity.name + for custom_granularity in model.time_spine.custom_granularities + } == {"retail_month", "martian_year"} + for custom_granularity in model.time_spine.custom_granularities: + if custom_granularity.name == "martian_year": + assert custom_granularity.column_name == "martian__year_xyz" + else: + assert len(model.time_spine.custom_granularities) == 0 + assert len(model.columns) > 0 + assert ( + list(model.columns.values())[0].granularity + == model_names_to_granularities[model.name] + ) + + # Test that project configs are set as expected in semantic manifest + semantic_manifest = SemanticManifest(manifest) + assert semantic_manifest.validate() + project_config = semantic_manifest._get_pydantic_semantic_manifest().project_configuration + # Legacy config + assert len(project_config.time_spine_table_configurations) == 1 + legacy_time_spine_config = project_config.time_spine_table_configurations[0] + assert legacy_time_spine_config.column_name == day_column_name + assert legacy_time_spine_config.location.replace('"', "").split(".")[-1] == day_model_name + assert legacy_time_spine_config.grain == TimeGranularity.DAY + # Current configs + assert len(project_config.time_spines) == 2 + sl_time_spine_aliases: Set[str] = set() + for sl_time_spine in project_config.time_spines: + alias = sl_time_spine.node_relation.alias + sl_time_spine_aliases.add(alias) + assert sl_time_spine.primary_column.name == model_names_to_col_names[alias] + assert ( + sl_time_spine.primary_column.time_granularity + == model_names_to_granularities[alias] + ) + assert sl_time_spine_aliases == expected_time_spine_aliases + + +class TestValidLegacyTimeSpine: + """Tests that YAML using only legacy time spine config parses as expected.""" + + @pytest.fixture(scope="class") + def models(self): + return { + "metricflow_time_spine.sql": metricflow_time_spine_sql, + "semantic_model_people.yml": semantic_model_people_yml, + "people.sql": models_people_sql, + } + + def test_time_spines(self, project): + runner = dbtRunner() + result = runner.invoke(["parse"]) + assert result.success + assert isinstance(result.result, Manifest) + + manifest = get_manifest(project.project_root) + assert manifest + + # Test that project configs are set as expected in semantic manifest + semantic_manifest = SemanticManifest(manifest) + assert semantic_manifest.validate() + project_config = semantic_manifest._get_pydantic_semantic_manifest().project_configuration + # Legacy config + assert len(project_config.time_spine_table_configurations) == 1 + legacy_time_spine_config = project_config.time_spine_table_configurations[0] + assert legacy_time_spine_config.column_name == "date_day" + assert ( + legacy_time_spine_config.location.replace('"', "").split(".")[-1] + == "metricflow_time_spine" + ) + assert legacy_time_spine_config.grain == TimeGranularity.DAY + # Current configs + assert len(project_config.time_spines) == 0 + + +class TestMissingTimeSpine: + """Tests that YAML with semantic models but no time spines errors.""" + + @pytest.fixture(scope="class") + def models(self): + return { + "semantic_model_people.yml": semantic_model_people_yml, + "people.sql": models_people_sql, + } + + def test_time_spines(self, project): + runner = dbtRunner() + result = runner.invoke(["parse"]) + assert isinstance(result.exception, ParsingError) + assert ( + "The semantic layer requires a time spine model with granularity DAY or smaller" + in result.exception.msg + ) + + +class TestTimeSpineStandardColumnMissing: + """Tests that YAML with time spine standard granularity column not in model errors.""" + + @pytest.fixture(scope="class") + def models(self): + return { + "semantic_model_people.yml": semantic_model_people_yml, + "people.sql": models_people_sql, + "metricflow_time_spine.sql": metricflow_time_spine_sql, + "metricflow_time_spine_second.sql": metricflow_time_spine_second_sql, + "time_spines.yml": time_spine_missing_standard_column_yml, + } + + def test_time_spines(self, project): + runner = dbtRunner() + result = runner.invoke(["parse"]) + assert isinstance(result.exception, ParsingError) + assert ( + "Time spine standard granularity column must be defined on the model." + in result.exception.msg + ) + + +class TestTimeSpineCustomColumnMissing: + """Tests that YAML with time spine custom granularity column not in model errors.""" + + @pytest.fixture(scope="class") + def models(self): + return { + "semantic_model_people.yml": semantic_model_people_yml, + "people.sql": models_people_sql, + "metricflow_time_spine.sql": metricflow_time_spine_sql, + "metricflow_time_spine_second.sql": metricflow_time_spine_second_sql, + "time_spines.yml": time_spine_missing_custom_column_yml, + } + + def test_time_spines(self, project): + runner = dbtRunner() + result = runner.invoke(["parse"]) + assert isinstance(result.exception, ParsingError) + assert ( + "Time spine custom granularity columns do not exist in the model." + in result.exception.msg + ) + + +class TestTimeSpineGranularityMissing: + """Tests that YAML with time spine column without granularity errors.""" + + @pytest.fixture(scope="class") + def models(self): + return { + "semantic_model_people.yml": semantic_model_people_yml, + "people.sql": models_people_sql, + "metricflow_time_spine.sql": metricflow_time_spine_sql, + "metricflow_time_spine_second.sql": metricflow_time_spine_second_sql, + "time_spines.yml": time_spine_missing_granularity_yml, + } + + def test_time_spines(self, project): + runner = dbtRunner() + result = runner.invoke(["parse"]) + assert isinstance(result.exception, ParsingError) + assert ( + "Time spine standard granularity column must have a granularity defined." + in result.exception.msg + ) diff --git a/tests/functional/unit_testing/fixtures.py b/tests/functional/unit_testing/fixtures.py index e73351f89d8..1da67f7e762 100644 --- a/tests/functional/unit_testing/fixtures.py +++ b/tests/functional/unit_testing/fixtures.py @@ -50,6 +50,18 @@ FROM {{ ref('my_model_a') }} """ +test_my_model_a_yml = """ +models: + - name: my_model_a + columns: + - name: a + tests: + - not_null + - name: id + tests: + - not_null +""" + test_my_model_yml = """ unit_tests: - name: test_my_model @@ -292,6 +304,28 @@ {% endif %} """ +my_incremental_model_with_alias_sql = """ +{{ + config( + materialized='incremental', + alias='alias_name' + ) +}} + +select * from {{ ref('events') }} +{% if is_incremental() %} +where event_time > (select max(event_time) from {{ this }}) +{% endif %} +""" + +my_incremental_model_versioned_yml = """ +models: + - name: my_incremental_model + latest_version: 1 + versions: + - v: 1 +""" + test_my_model_incremental_yml_basic = """ unit_tests: - name: incremental_false diff --git a/tests/functional/unit_testing/test_unit_testing.py b/tests/functional/unit_testing/test_unit_testing.py index 7332ddccb64..160f528787d 100644 --- a/tests/functional/unit_testing/test_unit_testing.py +++ b/tests/functional/unit_testing/test_unit_testing.py @@ -8,6 +8,8 @@ external_package, external_package__accounts_seed_csv, my_incremental_model_sql, + my_incremental_model_versioned_yml, + my_incremental_model_with_alias_sql, my_model_a_sql, my_model_b_sql, my_model_sql, @@ -76,11 +78,19 @@ def test_basic(self, project): ) assert len(results) == 1 - # Exclude unit tests with environment variable + # Exclude unit tests with environment variable for build command os.environ["DBT_EXCLUDE_RESOURCE_TYPES"] = "unit_test" results = run_dbt(["build", "--select", "my_model"], expect_pass=True) assert len(results) == 1 + # Exclude unit tests with environment variable for test command + results = run_dbt(["test", "--select", "my_model"], expect_pass=True) + assert len(results) == 0 + + # Exclude unit tests with environment variable for list command + results = run_dbt(["list", "--select", "my_model"], expect_pass=True) + assert len(results) == 1 + del os.environ["DBT_EXCLUDE_RESOURCE_TYPES"] # Test select by test name @@ -263,6 +273,42 @@ def test_no_this_input(self, project): """ +class TestUnitTestIncrementalModelWithAlias: + @pytest.fixture(scope="class") + def models(self): + return { + "my_incremental_model.sql": my_incremental_model_with_alias_sql, + "events.sql": event_sql, + "schema.yml": test_my_model_incremental_yml_basic, + } + + def test_basic(self, project): + results = run_dbt(["run"]) + assert len(results) == 2 + + # Select by model name + results = run_dbt(["test", "--select", "my_incremental_model"], expect_pass=True) + assert len(results) == 2 + + +class TestUnitTestIncrementalModelWithVersion: + @pytest.fixture(scope="class") + def models(self): + return { + "my_incremental_model.sql": my_incremental_model_sql, + "events.sql": event_sql, + "schema.yml": my_incremental_model_versioned_yml + test_my_model_incremental_yml_basic, + } + + def test_basic(self, project): + results = run_dbt(["run"]) + assert len(results) == 2 + + # Select by model name + results = run_dbt(["test", "--select", "my_incremental_model"], expect_pass=True) + assert len(results) == 2 + + class TestUnitTestExplicitSeed: @pytest.fixture(scope="class") def seeds(self): diff --git a/tests/functional/unit_testing/test_ut_resource_types.py b/tests/functional/unit_testing/test_ut_resource_types.py new file mode 100644 index 00000000000..09f64bdd061 --- /dev/null +++ b/tests/functional/unit_testing/test_ut_resource_types.py @@ -0,0 +1,91 @@ +import pytest +from fixtures import ( # noqa: F401 + my_model_a_sql, + my_model_b_sql, + my_model_sql, + test_my_model_a_yml, + test_my_model_pass_yml, +) + +from dbt.tests.util import run_dbt + +EXPECTED_MODELS = [ + "test.my_model", + "test.my_model_a", + "test.my_model_b", +] + +EXPECTED_DATA_TESTS = [ + "test.not_null_my_model_a_a", + "test.not_null_my_model_a_id", +] + +EXPECTED_UNIT_TESTS = [ + "unit_test:test.test_my_model", +] + + +class TestUnitTestResourceTypes: + @pytest.fixture(scope="class") + def models(self): + return { + "my_model.sql": my_model_sql, + "my_model_a.sql": my_model_a_sql, + "my_model_b.sql": my_model_b_sql, + "test_my_model.yml": test_my_model_pass_yml, + "test_my_model_a.yml": test_my_model_a_yml, + } + + def test_unit_test_list(self, project): + results = run_dbt(["run"]) + + # unit tests + results = run_dbt(["list", "--resource-type", "unit_test"]) + assert sorted(results) == EXPECTED_UNIT_TESTS + + results = run_dbt(["list", "--exclude-resource-types", "model", "test"]) + assert sorted(results) == EXPECTED_UNIT_TESTS + + results = run_dbt(["test", "--resource-type", "unit_test"]) + assert len(results) == len(EXPECTED_UNIT_TESTS) + + results = run_dbt(["test", "--exclude-resource-types", "model", "test"]) + assert len(results) == len(EXPECTED_UNIT_TESTS) + + # data tests + results = run_dbt(["list", "--resource-type", "test"]) + assert sorted(results) == EXPECTED_DATA_TESTS + + results = run_dbt(["list", "--exclude-resource-types", "unit_test", "model"]) + assert sorted(results) == EXPECTED_DATA_TESTS + + results = run_dbt(["test", "--resource-type", "test"]) + assert len(results) == len(EXPECTED_DATA_TESTS) + + results = run_dbt(["test", "--exclude-resource-types", "unit_test", "model"]) + assert len(results) == len(EXPECTED_DATA_TESTS) + + results = run_dbt(["build", "--resource-type", "test"]) + assert len(results) == len(EXPECTED_DATA_TESTS) + + results = run_dbt(["build", "--exclude-resource-types", "unit_test", "model"]) + assert len(results) == len(EXPECTED_DATA_TESTS) + + # models + results = run_dbt(["list", "--resource-type", "model"]) + assert sorted(results) == EXPECTED_MODELS + + results = run_dbt(["list", "--exclude-resource-type", "unit_test", "test"]) + assert sorted(results) == EXPECTED_MODELS + + results = run_dbt(["test", "--resource-type", "model"]) + assert len(results) == 0 + + results = run_dbt(["test", "--exclude-resource-types", "unit_test", "test"]) + assert len(results) == 0 + + results = run_dbt(["build", "--resource-type", "model"]) + assert len(results) == len(EXPECTED_MODELS) + + results = run_dbt(["build", "--exclude-resource-type", "unit_test", "test"]) + assert len(results) == len(EXPECTED_MODELS) diff --git a/tests/functional/utils.py b/tests/functional/utils.py index 8af471e7c04..93c79cf5345 100644 --- a/tests/functional/utils.py +++ b/tests/functional/utils.py @@ -1,11 +1,8 @@ import os from contextlib import contextmanager -from dataclasses import dataclass, field from datetime import datetime from pathlib import Path -from typing import List, Optional - -from dbt_common.events.base_types import BaseEvent, EventMsg +from typing import Optional @contextmanager @@ -20,16 +17,3 @@ def up_one(return_path: Optional[Path] = None): def is_aware(dt: datetime) -> bool: return dt.tzinfo is not None and dt.tzinfo.utcoffset(dt) is not None - - -@dataclass -class EventCatcher: - event_to_catch: BaseEvent - caught_events: List[EventMsg] = field(default_factory=list) - - def catch(self, event: EventMsg): - if event.info.name == self.event_to_catch.__name__: - self.caught_events.append(event) - - def flush(self) -> None: - self.caught_events = [] diff --git a/tests/unit/clients/test_jinja_static.py b/tests/unit/clients/test_jinja_static.py index d575cfb76e8..42264f24331 100644 --- a/tests/unit/clients/test_jinja_static.py +++ b/tests/unit/clients/test_jinja_static.py @@ -1,44 +1,120 @@ -import unittest +import pytest -from dbt.clients.jinja_static import statically_extract_macro_calls +from dbt.artifacts.resources import RefArgs +from dbt.clients.jinja_static import ( + statically_extract_macro_calls, + statically_parse_ref_or_source, + statically_parse_unrendered_config, +) from dbt.context.base import generate_base_context +from dbt.exceptions import ParsingError -class MacroCalls(unittest.TestCase): - def setUp(self): - self.macro_strings = [ +@pytest.mark.parametrize( + "macro_string,expected_possible_macro_calls", + [ + ( "{% macro parent_macro() %} {% do return(nested_macro()) %} {% endmacro %}", - "{% macro lr_macro() %} {{ return(load_result('relations').table) }} {% endmacro %}", - "{% macro get_snapshot_unique_id() -%} {{ return(adapter.dispatch('get_snapshot_unique_id')()) }} {%- endmacro %}", - "{% macro get_columns_in_query(select_sql) -%} {{ return(adapter.dispatch('get_columns_in_query')(select_sql)) }} {% endmacro %}", - """{% macro test_mutually_exclusive_ranges(model) %} - with base as ( - select {{ get_snapshot_unique_id() }} as dbt_unique_id, - * - from {{ model }} ) - {% endmacro %}""", - "{% macro test_my_test(model) %} select {{ current_timestamp_backcompat() }} {% endmacro %}", - "{% macro some_test(model) -%} {{ return(adapter.dispatch('test_some_kind4', 'foo_utils4')) }} {%- endmacro %}", - "{% macro some_test(model) -%} {{ return(adapter.dispatch('test_some_kind5', macro_namespace = 'foo_utils5')) }} {%- endmacro %}", - ] - - self.possible_macro_calls = [ ["nested_macro"], + ), + ( + "{% macro lr_macro() %} {{ return(load_result('relations').table) }} {% endmacro %}", ["load_result"], + ), + ( + "{% macro get_snapshot_unique_id() -%} {{ return(adapter.dispatch('get_snapshot_unique_id')()) }} {%- endmacro %}", ["get_snapshot_unique_id"], + ), + ( + "{% macro get_columns_in_query(select_sql) -%} {{ return(adapter.dispatch('get_columns_in_query')(select_sql)) }} {% endmacro %}", ["get_columns_in_query"], + ), + ( + """{% macro test_mutually_exclusive_ranges(model) %} + with base as ( + select {{ get_snapshot_unique_id() }} as dbt_unique_id, + * + from {{ model }} ) + {% endmacro %}""", ["get_snapshot_unique_id"], + ), + ( + "{% macro test_my_test(model) %} select {{ current_timestamp_backcompat() }} {% endmacro %}", ["current_timestamp_backcompat"], + ), + ( + "{% macro some_test(model) -%} {{ return(adapter.dispatch('test_some_kind4', 'foo_utils4')) }} {%- endmacro %}", ["test_some_kind4", "foo_utils4.test_some_kind4"], + ), + ( + "{% macro some_test(model) -%} {{ return(adapter.dispatch('test_some_kind5', macro_namespace = 'foo_utils5')) }} {%- endmacro %}", ["test_some_kind5", "foo_utils5.test_some_kind5"], - ] + ), + ], +) +def test_extract_macro_calls(macro_string, expected_possible_macro_calls): + cli_vars = {"local_utils_dispatch_list": ["foo_utils4"]} + ctx = generate_base_context(cli_vars) + + possible_macro_calls = statically_extract_macro_calls(macro_string, ctx) + assert possible_macro_calls == expected_possible_macro_calls + + +class TestStaticallyParseRefOrSource: + def test_invalid_expression(self): + with pytest.raises(ParsingError): + statically_parse_ref_or_source("invalid") + + @pytest.mark.parametrize( + "expression,expected_ref_or_source", + [ + ("ref('model')", RefArgs(name="model")), + ("ref('package','model')", RefArgs(name="model", package="package")), + ("ref('model',v=3)", RefArgs(name="model", version=3)), + ("ref('package','model',v=3)", RefArgs(name="model", package="package", version=3)), + ("source('schema', 'table')", ["schema", "table"]), + ], + ) + def test_valid_ref_expression(self, expression, expected_ref_or_source): + ref_or_source = statically_parse_ref_or_source(expression) + assert ref_or_source == expected_ref_or_source - def test_macro_calls(self): - cli_vars = {"local_utils_dispatch_list": ["foo_utils4"]} - ctx = generate_base_context(cli_vars) - index = 0 - for macro_string in self.macro_strings: - possible_macro_calls = statically_extract_macro_calls(macro_string, ctx) - self.assertEqual(self.possible_macro_calls[index], possible_macro_calls) - index += 1 +class TestStaticallyParseUnrenderedConfig: + @pytest.mark.parametrize( + "expression,expected_unrendered_config", + [ + ( + "{{ config(materialized='view') }}", + {"materialized": "Keyword(key='materialized', value=Const(value='view'))"}, + ), + ( + "{{ config(materialized='view', enabled=True) }}", + { + "materialized": "Keyword(key='materialized', value=Const(value='view'))", + "enabled": "Keyword(key='enabled', value=Const(value=True))", + }, + ), + ( + "{{ config(materialized=env_var('test')) }}", + { + "materialized": "Keyword(key='materialized', value=Call(node=Name(name='env_var', ctx='load'), args=[Const(value='test')], kwargs=[], dyn_args=None, dyn_kwargs=None))" + }, + ), + ( + "{{ config(materialized=env_var('test', default='default')) }}", + { + "materialized": "Keyword(key='materialized', value=Call(node=Name(name='env_var', ctx='load'), args=[Const(value='test')], kwargs=[Keyword(key='default', value=Const(value='default'))], dyn_args=None, dyn_kwargs=None))" + }, + ), + ( + "{{ config(materialized=env_var('test', default=env_var('default'))) }}", + { + "materialized": "Keyword(key='materialized', value=Call(node=Name(name='env_var', ctx='load'), args=[Const(value='test')], kwargs=[Keyword(key='default', value=Call(node=Name(name='env_var', ctx='load'), args=[Const(value='default')], kwargs=[], dyn_args=None, dyn_kwargs=None))], dyn_args=None, dyn_kwargs=None))" + }, + ), + ], + ) + def test_statically_parse_unrendered_config(self, expression, expected_unrendered_config): + unrendered_config = statically_parse_unrendered_config(expression) + assert unrendered_config == expected_unrendered_config diff --git a/tests/unit/config/test_project.py b/tests/unit/config/test_project.py index ae0ae3928dc..ddd519cc6ee 100644 --- a/tests/unit/config/test_project.py +++ b/tests/unit/config/test_project.py @@ -31,7 +31,7 @@ class TestProjectMethods: def test_all_source_paths(self, project: Project): assert ( project.all_source_paths.sort() - == ["models", "seeds", "snapshots", "analyses", "macros"].sort() + == ["models", "seeds", "snapshots", "analyses", "macros", "tests"].sort() ) def test_generic_test_paths(self, project: Project): @@ -43,7 +43,7 @@ def test_fixture_paths(self, project: Project): def test__str__(self, project: Project): assert ( str(project) - == "{'name': 'test_project', 'version': 1.0, 'project-root': 'doesnt/actually/exist', 'profile': 'test_profile', 'model-paths': ['models'], 'macro-paths': ['macros'], 'seed-paths': ['seeds'], 'test-paths': ['tests'], 'analysis-paths': ['analyses'], 'docs-paths': ['docs'], 'asset-paths': ['assets'], 'target-path': 'target', 'snapshot-paths': ['snapshots'], 'clean-targets': ['target'], 'log-path': 'path/to/project/logs', 'quoting': {}, 'models': {}, 'on-run-start': [], 'on-run-end': [], 'dispatch': [{'macro_namespace': 'dbt_utils', 'search_order': ['test_project', 'dbt_utils']}], 'seeds': {}, 'snapshots': {}, 'sources': {}, 'data_tests': {}, 'unit_tests': {}, 'metrics': {}, 'semantic-models': {}, 'saved-queries': {}, 'exposures': {}, 'vars': {}, 'require-dbt-version': ['=0.0.0'], 'restrict-access': False, 'dbt-cloud': {}, 'query-comment': {'comment': \"\\n{%- set comment_dict = {} -%}\\n{%- do comment_dict.update(\\n app='dbt',\\n dbt_version=dbt_version,\\n profile_name=target.get('profile_name'),\\n target_name=target.get('target_name'),\\n) -%}\\n{%- if node is not none -%}\\n {%- do comment_dict.update(\\n node_id=node.unique_id,\\n ) -%}\\n{% else %}\\n {# in the node context, the connection name is the node_id #}\\n {%- do comment_dict.update(connection_name=connection_name) -%}\\n{%- endif -%}\\n{{ return(tojson(comment_dict)) }}\\n\", 'append': False, 'job-label': False}, 'packages': []}" + == "{'name': 'test_project', 'version': 1.0, 'project-root': 'doesnt/actually/exist', 'profile': 'test_profile', 'model-paths': ['models'], 'macro-paths': ['macros'], 'seed-paths': ['seeds'], 'test-paths': ['tests'], 'analysis-paths': ['analyses'], 'docs-paths': ['docs'], 'asset-paths': ['assets'], 'target-path': 'target', 'snapshot-paths': ['snapshots'], 'clean-targets': ['target'], 'log-path': 'path/to/project/logs', 'quoting': {}, 'models': {}, 'on-run-start': [], 'on-run-end': [], 'dispatch': [{'macro_namespace': 'dbt_utils', 'search_order': ['test_project', 'dbt_utils']}], 'seeds': {}, 'snapshots': {}, 'sources': {}, 'data_tests': {}, 'unit_tests': {}, 'metrics': {}, 'semantic-models': {}, 'saved-queries': {}, 'exposures': {}, 'vars': {}, 'require-dbt-version': ['=0.0.0'], 'restrict-access': False, 'dbt-cloud': {}, 'flags': {}, 'query-comment': {'comment': \"\\n{%- set comment_dict = {} -%}\\n{%- do comment_dict.update(\\n app='dbt',\\n dbt_version=dbt_version,\\n profile_name=target.get('profile_name'),\\n target_name=target.get('target_name'),\\n) -%}\\n{%- if node is not none -%}\\n {%- do comment_dict.update(\\n node_id=node.unique_id,\\n ) -%}\\n{% else %}\\n {# in the node context, the connection name is the node_id #}\\n {%- do comment_dict.update(connection_name=connection_name) -%}\\n{%- endif -%}\\n{{ return(tojson(comment_dict)) }}\\n\", 'append': False, 'job-label': False}, 'packages': []}" ) def test_get_selector(self, project: Project): @@ -99,7 +99,8 @@ def test_defaults(self): self.assertEqual(project.test_paths, ["tests"]) self.assertEqual(project.analysis_paths, ["analyses"]) self.assertEqual( - set(project.docs_paths), set(["models", "seeds", "snapshots", "analyses", "macros"]) + set(project.docs_paths), + {"models", "seeds", "snapshots", "analyses", "macros", "tests"}, ) self.assertEqual(project.asset_paths, []) self.assertEqual(project.target_path, "target") @@ -128,7 +129,7 @@ def test_implicit_overrides(self): ) self.assertEqual( set(project.docs_paths), - set(["other-models", "seeds", "snapshots", "analyses", "macros"]), + {"other-models", "seeds", "snapshots", "analyses", "macros", "tests"}, ) def test_all_overrides(self): diff --git a/tests/unit/config/test_runtime.py b/tests/unit/config/test_runtime.py index 816ec8f98c3..d03d33dab94 100644 --- a/tests/unit/config/test_runtime.py +++ b/tests/unit/config/test_runtime.py @@ -129,7 +129,7 @@ def test_from_args(self): self.assertEqual(config.test_paths, ["tests"]) self.assertEqual(config.analysis_paths, ["analyses"]) self.assertEqual( - set(config.docs_paths), set(["models", "seeds", "snapshots", "analyses", "macros"]) + set(config.docs_paths), {"models", "seeds", "snapshots", "analyses", "macros", "tests"} ) self.assertEqual(config.asset_paths, []) self.assertEqual(config.target_path, "target") diff --git a/tests/unit/context/test_providers.py b/tests/unit/context/test_providers.py index 224675143e4..54a5e067ad1 100644 --- a/tests/unit/context/test_providers.py +++ b/tests/unit/context/test_providers.py @@ -1,11 +1,15 @@ +import os from unittest import mock import pytest +from pytest_mock import MockerFixture from dbt.adapters.base import BaseRelation -from dbt.artifacts.resources import Quoting +from dbt.artifacts.resources import NodeConfig, Quoting +from dbt.artifacts.resources.types import BatchSize from dbt.context.providers import ( BaseResolver, + EventTimeFilter, RuntimeRefResolver, RuntimeSourceResolver, ) @@ -34,6 +38,49 @@ def test_resolve_limit(self, resolver, empty, expected_resolve_limit): assert resolver.resolve_limit == expected_resolve_limit + @pytest.mark.parametrize( + "dbt_experimental_microbatch,materialized,incremental_strategy,expect_filter", + [ + (True, "incremental", "microbatch", True), + (False, "incremental", "microbatch", False), + (True, "table", "microbatch", False), + (True, "incremental", "merge", False), + ], + ) + def test_resolve_event_time_filter( + self, + mocker: MockerFixture, + resolver: ResolverSubclass, + dbt_experimental_microbatch: bool, + materialized: str, + incremental_strategy: str, + expect_filter: bool, + ) -> None: + if dbt_experimental_microbatch: + mocker.patch.dict(os.environ, {"DBT_EXPERIMENTAL_MICROBATCH": "True"}) + + # Target mocking + target = mock.Mock() + target.config = mock.MagicMock(NodeConfig) + target.config.event_time = "created_at" + + # Resolver mocking + resolver.config.args.EVENT_TIME_END = None + resolver.config.args.EVENT_TIME_START = None + resolver.model.config = mock.MagicMock(NodeConfig) + resolver.model.config.materialized = materialized + resolver.model.config.incremental_strategy = incremental_strategy + resolver.model.config.batch_size = BatchSize.day + resolver.model.config.lookback = 1 + + # Try to get an EventTimeFilter + event_time_filter = resolver.resolve_event_time_filter(target=target) + + if expect_filter: + assert isinstance(event_time_filter, EventTimeFilter) + else: + assert event_time_filter is None + class TestRuntimeRefResolver: @pytest.fixture diff --git a/tests/unit/contracts/graph/test_manifest.py b/tests/unit/contracts/graph/test_manifest.py index 35e96308da7..146a51d5856 100644 --- a/tests/unit/contracts/graph/test_manifest.py +++ b/tests/unit/contracts/graph/test_manifest.py @@ -26,7 +26,7 @@ WhereFilterIntersection, ) from dbt.contracts.files import FileHash -from dbt.contracts.graph.manifest import Manifest, ManifestMetadata +from dbt.contracts.graph.manifest import DisabledLookup, Manifest, ManifestMetadata from dbt.contracts.graph.nodes import ( DependsOn, Exposure, @@ -37,7 +37,7 @@ SeedNode, SourceDefinition, ) -from dbt.exceptions import AmbiguousResourceNameRefError +from dbt.exceptions import AmbiguousResourceNameRefError, ParsingError from dbt.flags import set_from_args from dbt.node_types import NodeType from dbt_common.events.functions import reset_metadata_vars @@ -84,6 +84,7 @@ "docs", "checksum", "unrendered_config", + "unrendered_config_call_dict", "created_at", "config_call_dict", "relation_name", @@ -94,6 +95,8 @@ "constraints", "deprecation_date", "defer_relation", + "time_spine", + "batch_info", } ) @@ -1962,3 +1965,176 @@ def test_resolve_doc(docs, package, expected): expected_package, expected_name = expected assert result.name == expected_name assert result.package_name == expected_package + + +class TestManifestFindNodeFromRefOrSource: + @pytest.fixture + def mock_node(self): + return MockNode("my_package", "my_model") + + @pytest.fixture + def mock_disabled_node(self): + return MockNode("my_package", "disabled_node", config={"enabled": False}) + + @pytest.fixture + def mock_source(self): + return MockSource("root", "my_source", "source_table") + + @pytest.fixture + def mock_disabled_source(self): + return MockSource("root", "my_source", "disabled_source_table", config={"enabled": False}) + + @pytest.fixture + def mock_manifest(self, mock_node, mock_source, mock_disabled_node, mock_disabled_source): + return make_manifest( + nodes=[mock_node, mock_disabled_node], sources=[mock_source, mock_disabled_source] + ) + + @pytest.mark.parametrize( + "expression,expected_node", + [ + ("ref('my_package', 'my_model')", "mock_node"), + ("ref('my_package', 'doesnt_exist')", None), + ("ref('my_package', 'disabled_node')", "mock_disabled_node"), + ("source('my_source', 'source_table')", "mock_source"), + ("source('my_source', 'doesnt_exist')", None), + ("source('my_source', 'disabled_source_table')", "mock_disabled_source"), + ], + ) + def test_find_node_from_ref_or_source(self, expression, expected_node, mock_manifest, request): + node = mock_manifest.find_node_from_ref_or_source(expression) + + if expected_node is None: + assert node is None + else: + assert node == request.getfixturevalue(expected_node) + + @pytest.mark.parametrize("invalid_expression", ["invalid", "ref(')"]) + def test_find_node_from_ref_or_source_invalid_expression( + self, invalid_expression, mock_manifest + ): + with pytest.raises(ParsingError): + mock_manifest.find_node_from_ref_or_source(invalid_expression) + + +class TestDisabledLookup: + @pytest.fixture(scope="class") + def manifest(self): + return Manifest( + nodes={}, + sources={}, + macros={}, + docs={}, + disabled={}, + files={}, + exposures={}, + selectors={}, + ) + + @pytest.fixture(scope="class") + def mock_model(self): + return MockNode("package", "name", NodeType.Model) + + @pytest.fixture(scope="class") + def mock_model_with_version(self): + return MockNode("package", "name", NodeType.Model, version=3) + + @pytest.fixture(scope="class") + def mock_seed(self): + return MockNode("package", "name", NodeType.Seed) + + def test_find(self, manifest, mock_model): + manifest.disabled = {"model.package.name": [mock_model]} + lookup = DisabledLookup(manifest) + + assert lookup.find("name", "package") == [mock_model] + + def test_find_wrong_name(self, manifest, mock_model): + manifest.disabled = {"model.package.name": [mock_model]} + lookup = DisabledLookup(manifest) + + assert lookup.find("missing_name", "package") is None + + def test_find_wrong_package(self, manifest, mock_model): + manifest.disabled = {"model.package.name": [mock_model]} + lookup = DisabledLookup(manifest) + + assert lookup.find("name", "missing_package") is None + + def test_find_wrong_version(self, manifest, mock_model): + manifest.disabled = {"model.package.name": [mock_model]} + lookup = DisabledLookup(manifest) + + assert lookup.find("name", "package", version=3) is None + + def test_find_wrong_resource_types(self, manifest, mock_model): + manifest.disabled = {"model.package.name": [mock_model]} + lookup = DisabledLookup(manifest) + + assert lookup.find("name", "package", resource_types=[NodeType.Analysis]) is None + + def test_find_no_package(self, manifest, mock_model): + manifest.disabled = {"model.package.name": [mock_model]} + lookup = DisabledLookup(manifest) + + assert lookup.find("name", None) == [mock_model] + + def test_find_versioned_node(self, manifest, mock_model_with_version): + manifest.disabled = {"model.package.name": [mock_model_with_version]} + lookup = DisabledLookup(manifest) + + assert lookup.find("name", "package", version=3) == [mock_model_with_version] + + def test_find_versioned_node_no_package(self, manifest, mock_model_with_version): + manifest.disabled = {"model.package.name": [mock_model_with_version]} + lookup = DisabledLookup(manifest) + + assert lookup.find("name", None, version=3) == [mock_model_with_version] + + def test_find_versioned_node_no_version(self, manifest, mock_model_with_version): + manifest.disabled = {"model.package.name": [mock_model_with_version]} + lookup = DisabledLookup(manifest) + + assert lookup.find("name", "package") is None + + def test_find_versioned_node_wrong_version(self, manifest, mock_model_with_version): + manifest.disabled = {"model.package.name": [mock_model_with_version]} + lookup = DisabledLookup(manifest) + + assert lookup.find("name", "package", version=2) is None + + def test_find_versioned_node_wrong_name(self, manifest, mock_model_with_version): + manifest.disabled = {"model.package.name": [mock_model_with_version]} + lookup = DisabledLookup(manifest) + + assert lookup.find("wrong_name", "package", version=3) is None + + def test_find_versioned_node_wrong_package(self, manifest, mock_model_with_version): + manifest.disabled = {"model.package.name": [mock_model_with_version]} + lookup = DisabledLookup(manifest) + + assert lookup.find("name", "wrong_package", version=3) is None + + def test_find_multiple_nodes(self, manifest, mock_model, mock_seed): + manifest.disabled = {"model.package.name": [mock_model, mock_seed]} + lookup = DisabledLookup(manifest) + + assert lookup.find("name", "package") == [mock_model, mock_seed] + + def test_find_multiple_nodes_with_resource_types(self, manifest, mock_model, mock_seed): + manifest.disabled = {"model.package.name": [mock_model, mock_seed]} + lookup = DisabledLookup(manifest) + + assert lookup.find("name", "package", resource_types=[NodeType.Model]) == [mock_model] + + def test_find_multiple_nodes_with_wrong_resource_types(self, manifest, mock_model, mock_seed): + manifest.disabled = {"model.package.name": [mock_model, mock_seed]} + lookup = DisabledLookup(manifest) + + assert lookup.find("name", "package", resource_types=[NodeType.Analysis]) is None + + def test_find_multiple_nodes_with_resource_types_empty(self, manifest, mock_model, mock_seed): + manifest.disabled = {"model.package.name": [mock_model, mock_seed]} + lookup = DisabledLookup(manifest) + + assert lookup.find("name", "package", resource_types=[]) is None diff --git a/tests/unit/contracts/graph/test_nodes.py b/tests/unit/contracts/graph/test_nodes.py index a498b99dcbc..3b509d0d20d 100644 --- a/tests/unit/contracts/graph/test_nodes.py +++ b/tests/unit/contracts/graph/test_nodes.py @@ -1,5 +1,6 @@ import pickle import re +from argparse import Namespace from dataclasses import replace import pytest @@ -24,6 +25,11 @@ ) +@pytest.fixture +def args_for_flags() -> Namespace: + return Namespace(state_modified_compare_vars=False) + + def norm_whitespace(string): _RE_COMBINE_WHITESPACE = re.compile(r"\s+") string = _RE_COMBINE_WHITESPACE.sub(" ", string).strip() @@ -139,6 +145,7 @@ def basic_uncompiled_dict(): "checksum": "e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", }, "unrendered_config": {}, + "unrendered_config_call_dict": {}, "config_call_dict": {}, } @@ -183,6 +190,7 @@ def basic_compiled_dict(): "contract": {"enforced": False, "alias_types": True}, "docs": {"show": True}, "access": "protected", + "lookback": 1, }, "docs": {"show": True}, "columns": {}, @@ -197,6 +205,7 @@ def basic_compiled_dict(): "checksum": "e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", }, "unrendered_config": {}, + "unrendered_config_call_dict": {}, "config_call_dict": {}, "access": "protected", "constraints": [], @@ -458,6 +467,7 @@ def basic_uncompiled_schema_test_dict(): "checksum": "e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", }, "unrendered_config": {}, + "unrendered_config_call_dict": {}, "config_call_dict": {}, } @@ -515,6 +525,7 @@ def basic_compiled_schema_test_dict(): "unrendered_config": { "severity": "warn", }, + "unrendered_config_call_dict": {}, "config_call_dict": {}, } diff --git a/tests/unit/contracts/graph/test_nodes_parsed.py b/tests/unit/contracts/graph/test_nodes_parsed.py index ebbe2443771..dc5a326f4d9 100644 --- a/tests/unit/contracts/graph/test_nodes_parsed.py +++ b/tests/unit/contracts/graph/test_nodes_parsed.py @@ -65,8 +65,12 @@ @pytest.fixture -def flags_for_args() -> Namespace: - return Namespace(SEND_ANONYMOUS_USAGE_STATS=False) +def args_for_flags() -> Namespace: + return Namespace( + send_anonymous_usage_stats=False, + state_modified_compare_more_unrendered_values=False, + state_modified_compare_vars=False, + ) @pytest.fixture @@ -100,6 +104,7 @@ def populated_node_config_dict(): "docs": {"show": True}, "contract": {"enforced": False, "alias_types": True}, "access": "protected", + "lookback": 1, } @@ -187,6 +192,7 @@ def base_parsed_model_dict(): "contract": {"enforced": False, "alias_types": True}, "packages": [], "access": "protected", + "lookback": 1, }, "docs": {"show": True}, "contract": {"enforced": False, "alias_types": True}, @@ -197,6 +203,7 @@ def base_parsed_model_dict(): "checksum": "e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", }, "unrendered_config": {}, + "unrendered_config_call_dict": {}, "config_call_dict": {}, "access": AccessType.Protected.value, "constraints": [], @@ -297,6 +304,7 @@ def complex_parsed_model_dict(): "contract": {"enforced": False, "alias_types": True}, "packages": [], "access": "protected", + "lookback": 1, }, "docs": {"show": True}, "contract": {"enforced": False, "alias_types": True}, @@ -318,6 +326,7 @@ def complex_parsed_model_dict(): "materialized": "ephemeral", "post_hook": ['insert into blah(a, b) select "1", 1'], }, + "unrendered_config_call_dict": {}, "config_call_dict": {}, "access": AccessType.Protected.value, "constraints": [], @@ -520,12 +529,14 @@ def basic_parsed_seed_dict(): "docs": {"show": True}, "contract": {"enforced": False, "alias_types": True}, "packages": [], + "lookback": 1, }, "docs": {"show": True}, "columns": {}, "meta": {}, "checksum": {"name": "path", "checksum": "seeds/seed.csv"}, "unrendered_config": {}, + "unrendered_config_call_dict": {}, "config_call_dict": {}, } @@ -611,6 +622,7 @@ def complex_parsed_seed_dict(): "docs": {"show": True}, "contract": {"enforced": False, "alias_types": True}, "packages": [], + "lookback": 1, }, "docs": {"show": True}, "columns": { @@ -630,6 +642,7 @@ def complex_parsed_seed_dict(): "unrendered_config": { "persist_docs": {"relation": True, "columns": True}, }, + "unrendered_config_call_dict": {}, "config_call_dict": {}, } @@ -818,6 +831,7 @@ def base_parsed_hook_dict(): "docs": {"show": True}, "contract": {"enforced": False, "alias_types": True}, "packages": [], + "lookback": 1, }, "docs": {"show": True}, "contract": {"enforced": False, "alias_types": True}, @@ -828,6 +842,7 @@ def base_parsed_hook_dict(): "checksum": "e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", }, "unrendered_config": {}, + "unrendered_config_call_dict": {}, "config_call_dict": {}, } @@ -899,6 +914,7 @@ def complex_parsed_hook_dict(): "docs": {"show": True}, "contract": {"enforced": False, "alias_types": True}, "packages": [], + "lookback": 1, }, "docs": {"show": True}, "contract": {"enforced": False, "alias_types": True}, @@ -920,6 +936,7 @@ def complex_parsed_hook_dict(): "column_types": {"a": "text"}, "materialized": "table", }, + "unrendered_config_call_dict": {}, "config_call_dict": {}, } @@ -1013,6 +1030,7 @@ def minimal_parsed_schema_test_dict(): "name": "sha256", "checksum": "e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", }, + "unrendered_config_call_dict": {}, "config_call_dict": {}, } @@ -1063,6 +1081,7 @@ def basic_parsed_schema_test_dict(): "checksum": "e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", }, "unrendered_config": {}, + "unrendered_config_call_dict": {}, "config_call_dict": {}, } @@ -1151,6 +1170,7 @@ def complex_parsed_schema_test_dict(): "checksum": "e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", }, "unrendered_config": {"materialized": "table", "severity": "WARN"}, + "unrendered_config_call_dict": {}, "config_call_dict": {}, } @@ -1242,6 +1262,7 @@ def basic_timestamp_snapshot_config_dict(): "quoting": {}, "tags": [], "unique_key": "id", + "snapshot_meta_column_names": {}, "strategy": "timestamp", "updated_at": "last_update", "target_database": "some_snapshot_db", @@ -1253,6 +1274,7 @@ def basic_timestamp_snapshot_config_dict(): "packages": [], "docs": {"show": True}, "contract": {"enforced": False, "alias_types": True}, + "lookback": 1, } @@ -1277,6 +1299,7 @@ def complex_timestamp_snapshot_config_dict(): "post-hook": [{"sql": 'insert into blah(a, b) select "1", 1', "transaction": True}], "pre-hook": [], "quoting": {}, + "snapshot_meta_column_names": {}, "tags": [], "target_database": "some_snapshot_db", "target_schema": "some_snapshot_schema", @@ -1291,6 +1314,7 @@ def complex_timestamp_snapshot_config_dict(): "packages": [], "docs": {"show": True}, "contract": {"enforced": False, "alias_types": True}, + "lookback": 1, } @@ -1344,6 +1368,7 @@ def basic_check_snapshot_config_dict(): "post-hook": [], "pre-hook": [], "quoting": {}, + "snapshot_meta_column_names": {}, "tags": [], "target_database": "some_snapshot_db", "target_schema": "some_snapshot_schema", @@ -1357,6 +1382,7 @@ def basic_check_snapshot_config_dict(): "packages": [], "docs": {"show": True}, "contract": {"enforced": False, "alias_types": True}, + "lookback": 1, } @@ -1381,6 +1407,7 @@ def complex_set_snapshot_config_dict(): "post-hook": [{"sql": 'insert into blah(a, b) select "1", 1', "transaction": True}], "pre-hook": [], "quoting": {}, + "snapshot_meta_column_names": {}, "tags": [], "target_database": "some_snapshot_db", "target_schema": "some_snapshot_schema", @@ -1395,6 +1422,7 @@ def complex_set_snapshot_config_dict(): "packages": [], "docs": {"show": True}, "contract": {"enforced": False, "alias_types": True}, + "lookback": 1, } @@ -1506,6 +1534,7 @@ def basic_timestamp_snapshot_dict(): "post-hook": [], "pre-hook": [], "quoting": {}, + "snapshot_meta_column_names": {}, "tags": [], "target_database": "some_snapshot_db", "target_schema": "some_snapshot_schema", @@ -1519,6 +1548,7 @@ def basic_timestamp_snapshot_dict(): "docs": {"show": True}, "contract": {"enforced": False, "alias_types": True}, "packages": [], + "lookback": 1, }, "docs": {"show": True}, "contract": {"enforced": False, "alias_types": True}, @@ -1535,6 +1565,7 @@ def basic_timestamp_snapshot_dict(): "target_database": "some_snapshot_db", "target_schema": "some_snapshot_schema", }, + "unrendered_config_call_dict": {}, "config_call_dict": {}, } @@ -1608,6 +1639,7 @@ def basic_check_snapshot_dict(): "post-hook": [], "pre-hook": [], "quoting": {}, + "snapshot_meta_column_names": {}, "tags": [], "target_database": "some_snapshot_db", "target_schema": "some_snapshot_schema", @@ -1621,6 +1653,7 @@ def basic_check_snapshot_dict(): "docs": {"show": True}, "contract": {"enforced": False, "alias_types": True}, "packages": [], + "lookback": 1, }, "docs": {"show": True}, "contract": {"enforced": False, "alias_types": True}, @@ -1637,6 +1670,7 @@ def basic_check_snapshot_dict(): "strategy": "check", "check_cols": "all", }, + "unrendered_config_call_dict": {}, "config_call_dict": {}, } diff --git a/tests/unit/contracts/graph/test_semantic_manifest.py b/tests/unit/contracts/graph/test_semantic_manifest.py index cf3121dc9b0..b0ae0a2a600 100644 --- a/tests/unit/contracts/graph/test_semantic_manifest.py +++ b/tests/unit/contracts/graph/test_semantic_manifest.py @@ -1,5 +1,9 @@ +from unittest.mock import patch + import pytest +from core.dbt.contracts.graph.manifest import Manifest +from core.dbt.contracts.graph.nodes import ModelNode from dbt.contracts.graph.semantic_manifest import SemanticManifest @@ -24,6 +28,21 @@ def metrics( class TestSemanticManifest: + def test_validate(self, manifest): - sm_manifest = SemanticManifest(manifest) - assert sm_manifest.validate() + with patch("dbt.contracts.graph.semantic_manifest.get_flags") as patched_get_flags: + patched_get_flags.return_value.require_yaml_configuration_for_mf_time_spines = True + sm_manifest = SemanticManifest(manifest) + assert sm_manifest.validate() + + def test_require_yaml_configuration_for_mf_time_spines( + self, manifest: Manifest, metricflow_time_spine_model: ModelNode + ): + with patch("dbt.contracts.graph.semantic_manifest.get_flags") as patched_get_flags, patch( + "dbt.contracts.graph.semantic_manifest.deprecations" + ) as patched_deprecations: + patched_get_flags.return_value.require_yaml_configuration_for_mf_time_spines = False + manifest.nodes[metricflow_time_spine_model.unique_id] = metricflow_time_spine_model + sm_manifest = SemanticManifest(manifest) + assert sm_manifest.validate() + assert patched_deprecations.warn.call_count == 1 diff --git a/tests/unit/events/test_logging.py b/tests/unit/events/test_logging.py index 00284ecab78..b2077b64793 100644 --- a/tests/unit/events/test_logging.py +++ b/tests/unit/events/test_logging.py @@ -23,7 +23,7 @@ def test_clears_preexisting_event_manager_state(self) -> None: setup_event_logger(get_flags()) assert len(manager.loggers) == 0 - assert len(manager.callbacks) == 0 + assert len(manager.callbacks) == 1 # snowplow tracker for behavior flags def test_specify_max_bytes( self, diff --git a/tests/unit/graph/test_nodes.py b/tests/unit/graph/test_nodes.py index ff14874eb65..a0e0a8d7e56 100644 --- a/tests/unit/graph/test_nodes.py +++ b/tests/unit/graph/test_nodes.py @@ -15,7 +15,7 @@ ) from dbt.artifacts.resources.v1.semantic_model import NodeRelation from dbt.contracts.graph.model_config import TestConfig -from dbt.contracts.graph.nodes import ColumnInfo, ModelNode, SemanticModel +from dbt.contracts.graph.nodes import ColumnInfo, ModelNode, ParsedNode, SemanticModel from dbt.node_types import NodeType from dbt_common.contracts.constraints import ( ColumnLevelConstraint, @@ -68,6 +68,48 @@ def test_is_past_deprecation_date( assert default_model_node.is_past_deprecation_date is expected_is_past_deprecation_date + @pytest.mark.parametrize( + "model_constraints,columns,expected_all_constraints", + [ + ([], {}, []), + ( + [ModelLevelConstraint(type=ConstraintType.foreign_key)], + {}, + [ModelLevelConstraint(type=ConstraintType.foreign_key)], + ), + ( + [], + { + "id": ColumnInfo( + name="id", + constraints=[ColumnLevelConstraint(type=ConstraintType.foreign_key)], + ) + }, + [ColumnLevelConstraint(type=ConstraintType.foreign_key)], + ), + ( + [ModelLevelConstraint(type=ConstraintType.foreign_key)], + { + "id": ColumnInfo( + name="id", + constraints=[ColumnLevelConstraint(type=ConstraintType.foreign_key)], + ) + }, + [ + ModelLevelConstraint(type=ConstraintType.foreign_key), + ColumnLevelConstraint(type=ConstraintType.foreign_key), + ], + ), + ], + ) + def test_all_constraints( + self, default_model_node, model_constraints, columns, expected_all_constraints + ): + default_model_node.constraints = model_constraints + default_model_node.columns = columns + + assert default_model_node.all_constraints == expected_all_constraints + class TestSemanticModel: @pytest.fixture(scope="function") @@ -349,3 +391,35 @@ def test_disabled_unique_combo_multiple(): def assertSameContents(list1, list2): assert sorted(list1) == sorted(list2) + + +class TestParsedNode: + @pytest.fixture(scope="class") + def parsed_node(self) -> ParsedNode: + return ParsedNode( + resource_type=NodeType.Model, + unique_id="model.test_package.test_name", + name="test_name", + package_name="test_package", + schema="test_schema", + alias="test_alias", + fqn=["models", "test_name"], + original_file_path="test_original_file_path", + checksum=FileHash.from_contents("checksum"), + path="test_path.sql", + database=None, + ) + + def test_get_target_write_path(self, parsed_node): + write_path = parsed_node.get_target_write_path("target_path", "subdirectory") + assert ( + write_path + == "target_path/subdirectory/test_package/test_original_file_path/test_path.sql" + ) + + def test_get_target_write_path_split(self, parsed_node): + write_path = parsed_node.get_target_write_path("target_path", "subdirectory", "split") + assert ( + write_path + == "target_path/subdirectory/test_package/test_original_file_path/test_path/test_path_split.sql" + ) diff --git a/tests/unit/graph/test_selector.py b/tests/unit/graph/test_selector.py index 48b3c78436c..3da1ccd4226 100644 --- a/tests/unit/graph/test_selector.py +++ b/tests/unit/graph/test_selector.py @@ -297,3 +297,29 @@ def test_dependency_list(self, runtime_config: RuntimeConfig): queue.get(block=False) queue.mark_done(got.unique_id) assert queue.empty() + + def test_select_downstream_of_empty_model(self, runtime_config: RuntimeConfig): + # empty model + model_one = make_model(pkg="other", name="model_one", code="") + # non-empty model + model_two = make_model( + pkg="pkg", + name="model_two", + code="""select * from {{ref('model_one')}}""", + refs=[model_one], + ) + models = [model_one, model_two] + manifest = make_manifest(nodes=models) + + # Get the graph + compiler = dbt.compilation.Compiler(runtime_config) + graph = compiler.compile(manifest) + + # Ensure that model_two is selected as downstream of model_one + selector = NodeSelector(graph, manifest) + spec = graph_selector.SelectionCriteria.from_single_spec("model_one+") + assert selector.get_selected(spec) == {"model.pkg.model_two"} + + # Ensure that --indirect-selection empty returns the same result + spec.indirect_selection = graph_selector.IndirectSelection.Empty + assert selector.get_selected(spec) == {"model.pkg.model_two"} diff --git a/tests/unit/graph/test_selector_methods.py b/tests/unit/graph/test_selector_methods.py index 04aebe052d1..d500c631a1b 100644 --- a/tests/unit/graph/test_selector_methods.py +++ b/tests/unit/graph/test_selector_methods.py @@ -1,4 +1,5 @@ import copy +from argparse import Namespace from dataclasses import replace from pathlib import Path from unittest import mock @@ -643,6 +644,13 @@ def previous_state(manifest): return create_previous_state(manifest) +@pytest.fixture +def args_for_flags(): + return Namespace( + state_modified_compare_more_unrendered_values=False, state_modified_compare_vars=False + ) + + def add_node(manifest, node): manifest.nodes[node.unique_id] = node diff --git a/tests/unit/materializations/incremental/test_microbatch.py b/tests/unit/materializations/incremental/test_microbatch.py new file mode 100644 index 00000000000..f114d8649c3 --- /dev/null +++ b/tests/unit/materializations/incremental/test_microbatch.py @@ -0,0 +1,670 @@ +from datetime import datetime +from unittest import mock + +import pytest +import pytz +from freezegun import freeze_time + +from dbt.artifacts.resources import NodeConfig +from dbt.artifacts.resources.types import BatchSize +from dbt.materializations.incremental.microbatch import MicrobatchBuilder + +MODEL_CONFIG_BEGIN = datetime(2024, 1, 1, 0, 0, 0, 0, pytz.UTC) + + +class TestMicrobatchBuilder: + @pytest.fixture(scope="class") + def microbatch_model(self): + model = mock.Mock() + model.config = mock.MagicMock(NodeConfig) + model.config.materialized = "incremental" + model.config.incremental_strategy = "microbatch" + model.config.begin = MODEL_CONFIG_BEGIN + model.config.batch_size = BatchSize.day + + return model + + @freeze_time("2024-09-05 08:56:00") + @pytest.mark.parametrize( + "is_incremental,event_time_end,expected_end_time", + [ + ( + False, + None, + datetime(2024, 9, 6, 0, 0, 0, 0, pytz.UTC), + ), + ( + True, + None, + datetime(2024, 9, 6, 0, 0, 0, 0, pytz.UTC), + ), + ( + False, + datetime(2024, 10, 1, 0, 0, 0, 0, pytz.UTC), + datetime(2024, 10, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + True, + datetime(2024, 10, 1, 0, 0, 0, 0, pytz.UTC), + datetime(2024, 10, 1, 0, 0, 0, 0, pytz.UTC), + ), + ], + ) + def test_build_end_time( + self, microbatch_model, is_incremental, event_time_end, expected_end_time + ): + microbatch_builder = MicrobatchBuilder( + model=microbatch_model, + is_incremental=is_incremental, + event_time_start=None, + event_time_end=event_time_end, + ) + + assert microbatch_builder.build_end_time() == expected_end_time + + @pytest.mark.parametrize( + "is_incremental,event_time_start,checkpoint,batch_size,lookback,expected_start_time", + [ + ( + False, + None, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.day, + 0, + # is_incremental: False => model.config.begin + MODEL_CONFIG_BEGIN, + ), + # BatchSize.year + ( + False, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.year, + 0, + datetime(2024, 1, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + False, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.year, + # Offset not applied when event_time_start provided + 1, + datetime(2024, 1, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + False, + None, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.year, + 0, + # is_incremental=False + no start_time -> model.config.begin + MODEL_CONFIG_BEGIN, + ), + ( + True, + None, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.year, + 0, + datetime(2024, 1, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + True, + None, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.year, + 1, + datetime(2023, 1, 1, 0, 0, 0, 0, pytz.UTC), + ), + # BatchSize.month + ( + False, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.month, + 0, + datetime(2024, 9, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + False, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.month, + # Offset not applied when event_time_start provided + 1, + datetime(2024, 9, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + False, + None, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.month, + 0, + # is_incremental=False + no start_time -> model.config.begin + MODEL_CONFIG_BEGIN, + ), + ( + True, + None, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.month, + 0, + datetime(2024, 9, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + True, + None, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.month, + 1, + datetime(2024, 8, 1, 0, 0, 0, 0, pytz.UTC), + ), + # BatchSize.day + ( + False, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.day, + 0, + datetime(2024, 9, 5, 0, 0, 0, 0, pytz.UTC), + ), + ( + False, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.day, + # Offset not applied when event_time_start provided + 1, + datetime(2024, 9, 5, 0, 0, 0, 0, pytz.UTC), + ), + ( + False, + None, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.day, + 0, + # is_incremental=False + no start_time -> model.config.begin + MODEL_CONFIG_BEGIN, + ), + ( + True, + None, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.day, + 0, + datetime(2024, 9, 5, 0, 0, 0, 0, pytz.UTC), + ), + ( + True, + None, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.day, + 1, + datetime(2024, 9, 4, 0, 0, 0, 0, pytz.UTC), + ), + # BatchSize.hour + ( + False, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.hour, + 0, + datetime(2024, 9, 5, 8, 0, 0, 0, pytz.UTC), + ), + ( + False, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.hour, + # Offset not applied when event_time_start provided + 1, + datetime(2024, 9, 5, 8, 0, 0, 0, pytz.UTC), + ), + ( + False, + None, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.hour, + 0, + # is_incremental=False + no start_time -> model.config.begin + MODEL_CONFIG_BEGIN, + ), + ( + True, + None, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.hour, + 0, + datetime(2024, 9, 5, 8, 0, 0, 0, pytz.UTC), + ), + ( + True, + None, + datetime(2024, 9, 5, 8, 56, 0, 0, pytz.UTC), + BatchSize.hour, + 1, + datetime(2024, 9, 5, 7, 0, 0, 0, pytz.UTC), + ), + ( + True, + None, + datetime(2024, 9, 5, 0, 0, 0, 0, pytz.UTC), + BatchSize.hour, + 0, + datetime(2024, 9, 4, 23, 0, 0, 0, pytz.UTC), + ), + ( + True, + None, + datetime(2024, 9, 5, 0, 0, 0, 0, pytz.UTC), + BatchSize.hour, + 1, + datetime(2024, 9, 4, 22, 0, 0, 0, pytz.UTC), + ), + ( + True, + None, + datetime(2024, 9, 5, 0, 0, 0, 0, pytz.UTC), + BatchSize.day, + 0, + datetime(2024, 9, 4, 0, 0, 0, 0, pytz.UTC), + ), + ( + True, + None, + datetime(2024, 9, 5, 0, 0, 0, 0, pytz.UTC), + BatchSize.day, + 1, + datetime(2024, 9, 3, 0, 0, 0, 0, pytz.UTC), + ), + ( + True, + None, + datetime(2024, 9, 1, 0, 0, 0, 0, pytz.UTC), + BatchSize.month, + 0, + datetime(2024, 8, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + True, + None, + datetime(2024, 9, 1, 0, 0, 0, 0, pytz.UTC), + BatchSize.month, + 1, + datetime(2024, 7, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + True, + None, + datetime(2024, 1, 1, 0, 0, 0, 0, pytz.UTC), + BatchSize.year, + 0, + datetime(2023, 1, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + True, + None, + datetime(2024, 1, 1, 0, 0, 0, 0, pytz.UTC), + BatchSize.year, + 1, + datetime(2022, 1, 1, 0, 0, 0, 0, pytz.UTC), + ), + ], + ) + def test_build_start_time( + self, + microbatch_model, + is_incremental, + event_time_start, + checkpoint, + batch_size, + lookback, + expected_start_time, + ): + microbatch_model.config.batch_size = batch_size + microbatch_model.config.lookback = lookback + microbatch_builder = MicrobatchBuilder( + model=microbatch_model, + is_incremental=is_incremental, + event_time_start=event_time_start, + event_time_end=None, + ) + + assert microbatch_builder.build_start_time(checkpoint) == expected_start_time + + @pytest.mark.parametrize( + "start,end,batch_size,expected_batches", + [ + # BatchSize.year + ( + datetime(2024, 1, 1, 0, 0, 0, 0, pytz.UTC), + datetime(2026, 1, 7, 3, 56, 0, 0, pytz.UTC), + BatchSize.year, + [ + ( + datetime(2024, 1, 1, 0, 0, 0, 0, pytz.UTC), + datetime(2025, 1, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2025, 1, 1, 0, 0, 0, 0, pytz.UTC), + datetime(2026, 1, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2026, 1, 1, 0, 0, 0, 0, pytz.UTC), + datetime(2026, 1, 7, 3, 56, 0, 0, pytz.UTC), + ), + ], + ), + # BatchSize.month + ( + datetime(2024, 9, 1, 0, 0, 0, 0, pytz.UTC), + datetime(2024, 11, 7, 3, 56, 0, 0, pytz.UTC), + BatchSize.month, + [ + ( + datetime(2024, 9, 1, 0, 0, 0, 0, pytz.UTC), + datetime(2024, 10, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 10, 1, 0, 0, 0, 0, pytz.UTC), + datetime(2024, 11, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 11, 1, 0, 0, 0, 0, pytz.UTC), + datetime(2024, 11, 7, 3, 56, 0, 0, pytz.UTC), + ), + ], + ), + # BatchSize.day + ( + datetime(2024, 9, 5, 0, 0, 0, 0, pytz.UTC), + datetime(2024, 9, 7, 3, 56, 0, 0, pytz.UTC), + BatchSize.day, + [ + ( + datetime(2024, 9, 5, 0, 0, 0, 0, pytz.UTC), + datetime(2024, 9, 6, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 6, 0, 0, 0, 0, pytz.UTC), + datetime(2024, 9, 7, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 7, 0, 0, 0, 0, pytz.UTC), + datetime(2024, 9, 7, 3, 56, 0, 0, pytz.UTC), + ), + ], + ), + # BatchSize.hour + ( + datetime(2024, 9, 5, 1, 0, 0, 0, pytz.UTC), + datetime(2024, 9, 5, 3, 56, 0, 0, pytz.UTC), + BatchSize.hour, + [ + ( + datetime(2024, 9, 5, 1, 0, 0, 0, pytz.UTC), + datetime(2024, 9, 5, 2, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 5, 2, 0, 0, 0, pytz.UTC), + datetime(2024, 9, 5, 3, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 5, 3, 0, 0, 0, pytz.UTC), + datetime(2024, 9, 5, 3, 56, 0, 0, pytz.UTC), + ), + ], + ), + # Test when event_time_end matches the truncated batch size + ( + datetime(2024, 1, 1, 0, 0, 0, 0, pytz.UTC), + datetime(2026, 1, 1, 0, 0, 0, 0, pytz.UTC), + BatchSize.year, + [ + ( + datetime(2024, 1, 1, 0, 0, 0, 0, pytz.UTC), + datetime(2025, 1, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2025, 1, 1, 0, 0, 0, 0, pytz.UTC), + datetime(2026, 1, 1, 0, 0, 0, 0, pytz.UTC), + ), + ], + ), + ( + datetime(2024, 9, 1, 0, 0, 0, 0, pytz.UTC), + datetime(2024, 11, 1, 0, 0, 0, 0, pytz.UTC), + BatchSize.month, + [ + ( + datetime(2024, 9, 1, 0, 0, 0, 0, pytz.UTC), + datetime(2024, 10, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 10, 1, 0, 0, 0, 0, pytz.UTC), + datetime(2024, 11, 1, 0, 0, 0, 0, pytz.UTC), + ), + ], + ), + ( + datetime(2024, 9, 5, 0, 0, 0, 0, pytz.UTC), + datetime(2024, 9, 7, 0, 0, 0, 0, pytz.UTC), + BatchSize.day, + [ + ( + datetime(2024, 9, 5, 0, 0, 0, 0, pytz.UTC), + datetime(2024, 9, 6, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 6, 0, 0, 0, 0, pytz.UTC), + datetime(2024, 9, 7, 0, 0, 0, 0, pytz.UTC), + ), + ], + ), + ( + datetime(2024, 9, 5, 1, 0, 0, 0, pytz.UTC), + datetime(2024, 9, 5, 3, 0, 0, 0, pytz.UTC), + BatchSize.hour, + [ + ( + datetime(2024, 9, 5, 1, 0, 0, 0, pytz.UTC), + datetime(2024, 9, 5, 2, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 5, 2, 0, 0, 0, pytz.UTC), + datetime(2024, 9, 5, 3, 0, 0, 0, pytz.UTC), + ), + ], + ), + ], + ) + def test_build_batches(self, microbatch_model, start, end, batch_size, expected_batches): + microbatch_model.config.batch_size = batch_size + microbatch_builder = MicrobatchBuilder( + model=microbatch_model, is_incremental=True, event_time_start=None, event_time_end=None + ) + + actual_batches = microbatch_builder.build_batches(start, end) + assert len(actual_batches) == len(expected_batches) + assert actual_batches == expected_batches + + def test_build_batch_context_incremental_batch(self, microbatch_model): + microbatch_builder = MicrobatchBuilder( + model=microbatch_model, is_incremental=True, event_time_start=None, event_time_end=None + ) + context = microbatch_builder.build_batch_context(incremental_batch=True) + + assert context["model"] == microbatch_model.to_dict() + assert context["sql"] == microbatch_model.compiled_code + assert context["compiled_code"] == microbatch_model.compiled_code + + assert context["is_incremental"]() is True + assert context["should_full_refresh"]() is False + + def test_build_batch_context_incremental_batch_false(self, microbatch_model): + microbatch_builder = MicrobatchBuilder( + model=microbatch_model, is_incremental=True, event_time_start=None, event_time_end=None + ) + context = microbatch_builder.build_batch_context(incremental_batch=False) + + assert context["model"] == microbatch_model.to_dict() + assert context["sql"] == microbatch_model.compiled_code + assert context["compiled_code"] == microbatch_model.compiled_code + + # Only build is_incremental callables when not first batch + assert "is_incremental" not in context + assert "should_full_refresh" not in context + + @pytest.mark.parametrize( + "timestamp,batch_size,offset,expected_timestamp", + [ + ( + datetime(2024, 9, 5, 3, 56, 1, 1, pytz.UTC), + BatchSize.year, + 1, + datetime(2025, 1, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 5, 3, 56, 1, 1, pytz.UTC), + BatchSize.year, + -1, + datetime(2023, 1, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 5, 3, 56, 1, 1, pytz.UTC), + BatchSize.month, + 1, + datetime(2024, 10, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 5, 3, 56, 1, 1, pytz.UTC), + BatchSize.month, + -1, + datetime(2024, 8, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 5, 3, 56, 1, 1, pytz.UTC), + BatchSize.day, + 1, + datetime(2024, 9, 6, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 5, 3, 56, 1, 1, pytz.UTC), + BatchSize.day, + -1, + datetime(2024, 9, 4, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 5, 3, 56, 1, 1, pytz.UTC), + BatchSize.hour, + 1, + datetime(2024, 9, 5, 4, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 5, 3, 56, 1, 1, pytz.UTC), + BatchSize.hour, + -1, + datetime(2024, 9, 5, 2, 0, 0, 0, pytz.UTC), + ), + ], + ) + def test_offset_timestamp(self, timestamp, batch_size, offset, expected_timestamp): + assert ( + MicrobatchBuilder.offset_timestamp(timestamp, batch_size, offset) == expected_timestamp + ) + + @pytest.mark.parametrize( + "timestamp,batch_size,expected_timestamp", + [ + ( + datetime(2024, 9, 5, 3, 56, 1, 1, pytz.UTC), + BatchSize.year, + datetime(2024, 1, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 5, 3, 56, 1, 1, pytz.UTC), + BatchSize.month, + datetime(2024, 9, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 5, 3, 56, 1, 1, pytz.UTC), + BatchSize.day, + datetime(2024, 9, 5, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 5, 3, 56, 1, 1, pytz.UTC), + BatchSize.hour, + datetime(2024, 9, 5, 3, 0, 0, 0, pytz.UTC), + ), + ], + ) + def test_truncate_timestamp(self, timestamp, batch_size, expected_timestamp): + assert MicrobatchBuilder.truncate_timestamp(timestamp, batch_size) == expected_timestamp + + @pytest.mark.parametrize( + "batch_size,batch_start,expected_formatted_batch_start", + [ + (None, None, None), + (BatchSize.year, datetime(2020, 1, 1, 1), "2020-01-01"), + (BatchSize.month, datetime(2020, 1, 1, 1), "2020-01-01"), + (BatchSize.day, datetime(2020, 1, 1, 1), "2020-01-01"), + (BatchSize.hour, datetime(2020, 1, 1, 1), "2020-01-01 01:00:00"), + ], + ) + def test_format_batch_start(self, batch_size, batch_start, expected_formatted_batch_start): + assert ( + MicrobatchBuilder.format_batch_start(batch_start, batch_size) + == expected_formatted_batch_start + ) + + @pytest.mark.parametrize( + "timestamp,batch_size,expected_datetime", + [ + ( + datetime(2024, 9, 17, 16, 6, 0, 0, pytz.UTC), + BatchSize.hour, + datetime(2024, 9, 17, 17, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 17, 16, 0, 0, 0, pytz.UTC), + BatchSize.hour, + datetime(2024, 9, 17, 16, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 17, 16, 6, 0, 0, pytz.UTC), + BatchSize.day, + datetime(2024, 9, 18, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 17, 0, 0, 0, 0, pytz.UTC), + BatchSize.day, + datetime(2024, 9, 17, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 17, 16, 6, 0, 0, pytz.UTC), + BatchSize.month, + datetime(2024, 10, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 1, 0, 0, 0, 0, pytz.UTC), + BatchSize.month, + datetime(2024, 9, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 9, 17, 16, 6, 0, 0, pytz.UTC), + BatchSize.year, + datetime(2025, 1, 1, 0, 0, 0, 0, pytz.UTC), + ), + ( + datetime(2024, 1, 1, 0, 0, 0, 0, pytz.UTC), + BatchSize.year, + datetime(2024, 1, 1, 0, 0, 0, 0, pytz.UTC), + ), + ], + ) + def test_ceiling_timestamp( + self, timestamp: datetime, batch_size: BatchSize, expected_datetime: datetime + ) -> None: + ceilinged = MicrobatchBuilder.ceiling_timestamp(timestamp, batch_size) + assert ceilinged == expected_datetime diff --git a/tests/unit/parser/test_parser.py b/tests/unit/parser/test_parser.py index 20a2c9e8c83..d5f809252b3 100644 --- a/tests/unit/parser/test_parser.py +++ b/tests/unit/parser/test_parser.py @@ -6,8 +6,6 @@ import yaml -import dbt.flags -import dbt.parser from dbt import tracking from dbt.artifacts.resources import ModelConfig, RefArgs from dbt.context.context_config import ContextConfig @@ -60,7 +58,9 @@ normalize, ) -set_from_args(Namespace(WARN_ERROR=False), None) +set_from_args( + Namespace(warn_error=False, state_modified_compare_more_unrendered_values=False), None +) def get_abs_os_path(unix_path): @@ -94,7 +94,10 @@ def _generate_macros(self): yield pm def setUp(self): - dbt.flags.WARN_ERROR = True + set_from_args( + Namespace(warn_error=True, state_modified_compare_more_unrendered_values=False), + None, + ) # HACK: this is needed since tracking events can # be sent when using the model parser tracking.do_not_track() @@ -276,6 +279,7 @@ def assertEqualNodes(node_one, node_two): - not_null: severity: WARN - accepted_values: + description: Only primary colors are allowed in here values: ['red', 'blue', 'green'] - foreign_package.test_case: arg: 100 @@ -628,6 +632,7 @@ def test__parse_basic_model_tests(self): self.assertEqual(tests[0].tags, []) self.assertEqual(tests[0].refs, [RefArgs(name="my_model")]) self.assertEqual(tests[0].column_name, "color") + self.assertEqual(tests[0].description, "Only primary colors are allowed in here") self.assertEqual(tests[0].package_name, "snowplow") self.assertTrue(tests[0].name.startswith("accepted_values_")) self.assertEqual(tests[0].fqn, ["snowplow", tests[0].name]) @@ -651,7 +656,7 @@ def test__parse_basic_model_tests(self): self.assertEqual(tests[1].tags, []) self.assertEqual(tests[1].refs, [RefArgs(name="my_model")]) self.assertEqual(tests[1].column_name, "color") - self.assertEqual(tests[1].column_name, "color") + self.assertEqual(tests[1].description, "") self.assertEqual(tests[1].fqn, ["snowplow", tests[1].name]) self.assertTrue(tests[1].name.startswith("foreign_package_test_case_")) self.assertEqual(tests[1].package_name, "snowplow") @@ -1471,6 +1476,7 @@ def test_single_block(self): "unique_key": "id", "updated_at": "last_update", }, + unrendered_config_call_dict={}, ) assertEqualNodes(expected, node) file_id = "snowplow://" + normalize("snapshots/nested/snap_1.sql") @@ -1541,6 +1547,8 @@ def test_multi_block(self): "unique_key": "id", "updated_at": "last_update", }, + # Empty until state_modified_compare_more_unrendered_values=True + unrendered_config_call_dict={}, ) expect_bar = SnapshotNode( alias="bar", @@ -1578,6 +1586,8 @@ def test_multi_block(self): "unique_key": "id", "updated_at": "last_update", }, + # Empty until state_modified_compare_more_unrendered_values=True + unrendered_config_call_dict={}, ) assertEqualNodes(nodes[0], expect_bar) assertEqualNodes(nodes[1], expect_foo) diff --git a/tests/unit/task/test_run.py b/tests/unit/task/test_run.py index 19a88f6aa8f..8f239ccfc3a 100644 --- a/tests/unit/task/test_run.py +++ b/tests/unit/task/test_run.py @@ -1,17 +1,34 @@ +import threading from argparse import Namespace +from dataclasses import dataclass +from datetime import datetime, timedelta +from importlib import import_module +from typing import Optional, Type, Union +from unittest import mock from unittest.mock import MagicMock, patch import pytest +from psycopg2 import DatabaseError +from pytest_mock import MockerFixture +from dbt.adapters.contracts.connection import AdapterResponse from dbt.adapters.postgres import PostgresAdapter +from dbt.artifacts.resources.base import FileHash +from dbt.artifacts.resources.types import BatchSize, NodeType, RunHookType +from dbt.artifacts.resources.v1.components import DependsOn +from dbt.artifacts.resources.v1.config import NodeConfig +from dbt.artifacts.resources.v1.model import ModelConfig +from dbt.artifacts.schemas.batch_results import BatchResults from dbt.artifacts.schemas.results import RunStatus from dbt.artifacts.schemas.run import RunResult from dbt.config.runtime import RuntimeConfig from dbt.contracts.graph.manifest import Manifest -from dbt.contracts.graph.nodes import ModelNode +from dbt.contracts.graph.nodes import HookNode, ModelNode from dbt.events.types import LogModelResult +from dbt.exceptions import DbtRuntimeError from dbt.flags import get_flags, set_from_args -from dbt.task.run import ModelRunner, RunTask +from dbt.materializations.incremental.microbatch import MicrobatchBuilder +from dbt.task.run import ModelRunner, RunTask, _get_adapter_info from dbt.tests.util import safe_set_invocation_context from dbt_common.events.base_types import EventLevel from dbt_common.events.event_manager_client import add_callback_to_manager @@ -61,6 +78,22 @@ def test_run_task_preserve_edges(): mock_node_selector.get_graph_queue.assert_called_with(mock_spec, True) +def test_tracking_fails_safely_for_missing_adapter(): + assert {} == _get_adapter_info(None, {}) + + +def test_adapter_info_tracking(): + mock_run_result = MagicMock() + mock_run_result.node = MagicMock() + mock_run_result.node.config = {} + assert _get_adapter_info(PostgresAdapter, mock_run_result) == { + "model_adapter_details": {}, + "adapter_name": PostgresAdapter.__name__.split("Adapter")[0].lower(), + "adapter_version": import_module("dbt.adapters.postgres.__version__").version, + "base_adapter_version": import_module("dbt.adapters.__about__").version, + } + + class TestModelRunner: @pytest.fixture def log_model_result_catcher(self) -> EventCatcher: @@ -93,6 +126,7 @@ def run_result(self, table_model: ModelNode) -> RunResult: adapter_response={}, message="It did it", failures=None, + batch_results=None, node=table_model, ) @@ -126,3 +160,217 @@ def test_execute( ) -> None: model_runner.execute(model=table_model, manifest=manifest) # TODO: Assert that the model was executed + + def test__build_run_microbatch_model_result( + self, table_model: ModelNode, model_runner: ModelRunner + ) -> None: + batch = (datetime.now() - timedelta(days=1), datetime.now()) + only_successes = [ + RunResult( + node=table_model, + status=RunStatus.Success, + timing=[], + thread_id=threading.current_thread().name, + execution_time=0, + message="SUCCESS", + adapter_response={}, + failures=0, + batch_results=BatchResults(successful=[batch]), + ) + ] + only_failures = [ + RunResult( + node=table_model, + status=RunStatus.Error, + timing=[], + thread_id=threading.current_thread().name, + execution_time=0, + message="ERROR", + adapter_response={}, + failures=1, + batch_results=BatchResults(failed=[batch]), + ) + ] + mixed_results = only_failures + only_successes + + expect_success = model_runner._build_run_microbatch_model_result( + table_model, only_successes + ) + expect_error = model_runner._build_run_microbatch_model_result(table_model, only_failures) + expect_partial_success = model_runner._build_run_microbatch_model_result( + table_model, mixed_results + ) + + assert expect_success.status == RunStatus.Success + assert expect_error.status == RunStatus.Error + assert expect_partial_success.status == RunStatus.PartialSuccess + + @pytest.mark.parametrize( + "has_relation,relation_type,materialized,full_refresh_config,full_refresh_flag,expectation", + [ + (False, "table", "incremental", None, False, False), + (True, "other", "incremental", None, False, False), + (True, "table", "other", None, False, False), + # model config takes precendence + (True, "table", "incremental", True, False, False), + # model config takes precendence + (True, "table", "incremental", True, True, False), + # model config takes precendence + (True, "table", "incremental", False, False, True), + # model config takes precendence + (True, "table", "incremental", False, True, True), + # model config is none, so opposite flag value + (True, "table", "incremental", None, True, False), + # model config is none, so opposite flag value + (True, "table", "incremental", None, False, True), + ], + ) + def test__is_incremental( + self, + mocker: MockerFixture, + model_runner: ModelRunner, + has_relation: bool, + relation_type: str, + materialized: str, + full_refresh_config: Optional[bool], + full_refresh_flag: bool, + expectation: bool, + ) -> None: + + # Setup adapter relation getting + @dataclass + class RelationInfo: + database: str = "database" + schema: str = "schema" + name: str = "name" + + @dataclass + class Relation: + type: str + + model_runner.adapter = mocker.Mock() + model_runner.adapter.Relation.create_from.return_value = RelationInfo() + + if has_relation: + model_runner.adapter.get_relation.return_value = Relation(type=relation_type) + else: + model_runner.adapter.get_relation.return_value = None + + # Set ModelRunner configs + model_runner.config.args = Namespace(FULL_REFRESH=full_refresh_flag) + + # Create model with configs + model = model_runner.node + model.config = ModelConfig(materialized=materialized, full_refresh=full_refresh_config) + + # Assert result of _is_incremental + assert model_runner._is_incremental(model) == expectation + + def test_keyboard_breaks__execute_microbatch_materialization( + self, + table_model: ModelNode, + manifest: Manifest, + model_runner: ModelRunner, + ) -> None: + def mock_build_batch_context(*args, **kwargs): + raise KeyboardInterrupt("Test exception") + + def mock_is_incremental(*args, **kwargs): + return True + + table_model.config.materialized = "incremental" + table_model.config.incremental_strategy = "microbatch" + table_model.config.batch_size = BatchSize.day + + with patch.object( + MicrobatchBuilder, "build_batch_context", mock_build_batch_context + ), patch.object(ModelRunner, "_is_incremental", mock_is_incremental): + try: + model_runner._execute_microbatch_materialization( + table_model, manifest, {}, MagicMock() + ) + assert False, "KeybaordInterrupt failed to escape" + except KeyboardInterrupt: + assert True + + +class TestRunTask: + @pytest.fixture + def hook_node(self) -> HookNode: + return HookNode( + package_name="test", + path="/root/x/path.sql", + original_file_path="/root/path.sql", + language="sql", + raw_code="select * from wherever", + name="foo", + resource_type=NodeType.Operation, + unique_id="model.test.foo", + fqn=["test", "models", "foo"], + refs=[], + sources=[], + metrics=[], + depends_on=DependsOn(), + description="", + database="test_db", + schema="test_schema", + alias="bar", + tags=[], + config=NodeConfig(), + index=None, + checksum=FileHash.from_contents(""), + unrendered_config={}, + ) + + @pytest.mark.parametrize( + "error_to_raise,expected_result", + [ + (None, RunStatus.Success), + (DbtRuntimeError, RunStatus.Error), + (DatabaseError, RunStatus.Error), + (KeyboardInterrupt, KeyboardInterrupt), + ], + ) + def test_safe_run_hooks( + self, + mocker: MockerFixture, + runtime_config: RuntimeConfig, + manifest: Manifest, + hook_node: HookNode, + error_to_raise: Optional[Type[Exception]], + expected_result: Union[RunStatus, Type[Exception]], + ): + mocker.patch("dbt.task.run.RunTask.get_hooks_by_type").return_value = [hook_node] + mocker.patch("dbt.task.run.RunTask.get_hook_sql").return_value = hook_node.raw_code + + flags = mock.Mock() + flags.state = None + flags.defer_state = None + + run_task = RunTask( + args=flags, + config=runtime_config, + manifest=manifest, + ) + + adapter = mock.Mock() + adapter_execute = mock.Mock() + adapter_execute.return_value = (AdapterResponse(_message="Success"), None) + + if error_to_raise: + adapter_execute.side_effect = error_to_raise("Oh no!") + + adapter.execute = adapter_execute + + try: + result = run_task.safe_run_hooks( + adapter=adapter, + hook_type=RunHookType.End, + extra_context={}, + ) + assert isinstance(expected_result, RunStatus) + assert result == expected_result + except BaseException as e: + assert not isinstance(expected_result, RunStatus) + assert issubclass(expected_result, BaseException) + assert type(e) == expected_result diff --git a/tests/unit/test_behavior_flags.py b/tests/unit/test_behavior_flags.py new file mode 100644 index 00000000000..d899a83f283 --- /dev/null +++ b/tests/unit/test_behavior_flags.py @@ -0,0 +1,66 @@ +import pytest + +from dbt.tracking import ( + disable_tracking, + initialize_from_flags, + track_behavior_change_warn, +) +from dbt_common.behavior_flags import Behavior +from dbt_common.events.event_manager_client import ( + add_callback_to_manager, + cleanup_event_logger, +) + + +@pytest.fixture +def snowplow_tracker(mocker): + # initialize `active_user` without writing the cookie to disk + initialize_from_flags(True, "") + mocker.patch("dbt.tracking.User.set_cookie").return_value = {"id": 42} + + # add the relevant callback to the event manager + add_callback_to_manager(track_behavior_change_warn) + + # don't make a call, catch the request + # to avoid confusion, this is snowplow_tracker's track, not our wrapper that's also named track + snowplow_tracker = mocker.patch("dbt.tracking.tracker.track") + + yield snowplow_tracker + + # teardown + cleanup_event_logger() + disable_tracking() + + +def test_false_evaluation_triggers_snowplow_tracking(snowplow_tracker): + behavior = Behavior( + [{"name": "my_flag", "default": False, "description": "This is a false flag."}], {} + ) + if behavior.my_flag: + # trigger a False evaluation + assert False, "This flag should evaluate to false and skip this line" + assert snowplow_tracker.called + + +def test_true_evaluation_does_not_trigger_snowplow_tracking(snowplow_tracker): + behavior = Behavior( + [{"name": "my_flag", "default": True, "description": "This is a true flag."}], {} + ) + if behavior.my_flag: + pass + else: + # trigger a True evaluation + assert False, "This flag should evaluate to false and skip this line" + assert not snowplow_tracker.called + + +def test_false_evaluation_does_not_trigger_snowplow_tracking_when_disabled(snowplow_tracker): + disable_tracking() + + behavior = Behavior( + [{"name": "my_flag", "default": False, "description": "This is a false flag."}], {} + ) + if behavior.my_flag: + # trigger a False evaluation + assert False, "This flag should evaluate to false and skip this line" + assert not snowplow_tracker.called diff --git a/tests/unit/test_deprecations.py b/tests/unit/test_deprecations.py new file mode 100644 index 00000000000..5ac7cb003fa --- /dev/null +++ b/tests/unit/test_deprecations.py @@ -0,0 +1,38 @@ +import pytest + +import dbt.deprecations as deprecations + + +@pytest.fixture(scope="function") +def active_deprecations(): + deprecations.reset_deprecations() + assert not deprecations.active_deprecations + + yield deprecations.active_deprecations + + deprecations.reset_deprecations() + + +@pytest.fixture(scope="function") +def buffered_deprecations(): + deprecations.buffered_deprecations.clear() + assert not deprecations.buffered_deprecations + + yield deprecations.buffered_deprecations + + deprecations.buffered_deprecations.clear() + + +def test_buffer_deprecation(active_deprecations, buffered_deprecations): + deprecations.buffer("project-flags-moved") + + assert active_deprecations == set() + assert len(buffered_deprecations) == 1 + + +def test_fire_buffered_deprecations(active_deprecations, buffered_deprecations): + deprecations.buffer("project-flags-moved") + deprecations.fire_buffered_deprecations() + + assert active_deprecations == set(["project-flags-moved"]) + assert len(buffered_deprecations) == 0 diff --git a/tests/unit/test_events.py b/tests/unit/test_events.py index ed4ebfeac61..61f7068a139 100644 --- a/tests/unit/test_events.py +++ b/tests/unit/test_events.py @@ -157,6 +157,7 @@ def test_event_codes(self): package_name="my_package", materialization_name="view" ), core_types.SourceFreshnessProjectHooksNotRun(), + core_types.MFTimespineWithoutYamlConfigurationDeprecation(), # E - DB Adapter ====================== adapter_types.AdapterEventDebug(), adapter_types.AdapterEventInfo(), @@ -286,6 +287,7 @@ def test_event_codes(self): core_types.WarnStateTargetEqual(state_path=""), core_types.FreshnessConfigProblem(msg=""), core_types.SemanticValidationFailure(msg=""), + core_types.MicrobatchModelNoEventTimeInputs(model_name=""), # M - Deps generation ====================== core_types.GitSparseCheckoutSubdirectory(subdir=""), core_types.GitProgressCheckoutRevision(revision=""), @@ -319,6 +321,8 @@ def test_event_codes(self): core_types.DepsScrubbedPackageName(package_name=""), core_types.DepsUnpinned(revision="", git=""), core_types.NoNodesForSelectionCriteria(spec_raw=""), + # P - Artifacts ====================== + core_types.ArtifactWritten(artifact_type="manifest", artifact_path="path/to/artifact.json"), # Q - Node execution ====================== core_types.RunningOperationCaughtError(exc=""), core_types.CompileComplete(), @@ -412,6 +416,9 @@ def test_event_codes(self): core_types.CompiledNode( node_name="", compiled="", is_inline=True, unique_id="model.test.my_model" ), + core_types.SnapshotTimestampWarning( + snapshot_time_data_type="DATETIME", updated_at_data_type="DATETIMEZ" + ), # W - Node testing ====================== core_types.CatchableExceptionOnRun(exc=""), core_types.InternalErrorOnRun(build_path="", exc=""), @@ -514,7 +521,7 @@ def test_all_serializable(self): def test_date_serialization(): - ti = TimingInfo("test") + ti = TimingInfo("compile") ti.begin() ti.end() ti_dict = ti.to_dict() @@ -534,10 +541,7 @@ def test_bad_serialization(): with pytest.raises(Exception) as excinfo: types.Note(param_event_doesnt_have="This should break") - assert ( - str(excinfo.value) - == "[Note]: Unable to parse dict {'param_event_doesnt_have': 'This should break'}" - ) + assert 'has no field named "param_event_doesnt_have" at "Note"' in str(excinfo.value) def test_single_run_error(): @@ -555,7 +559,8 @@ def test_single_run_error(): node=None, adapter_response=dict(), message="oh no!", - failures=[], + failures=1, + batch_results=None, ) print_run_result_error(error_result) diff --git a/tests/unit/utils/project.py b/tests/unit/utils/project.py index 2e374b82fac..c8a22143daf 100644 --- a/tests/unit/utils/project.py +++ b/tests/unit/utils/project.py @@ -70,6 +70,7 @@ def project(selector_config: SelectorConfig) -> Project: project_env_vars={}, restrict_access=False, dbt_cloud={}, + flags={}, ) diff --git a/tests/utils.py b/tests/utils.py index 3f71cac38ef..e09e377a62b 100644 --- a/tests/utils.py +++ b/tests/utils.py @@ -1,5 +1,5 @@ from dataclasses import dataclass, field -from typing import List +from typing import Callable, List from dbt_common.events.base_types import BaseEvent, EventMsg @@ -8,9 +8,10 @@ class EventCatcher: event_to_catch: BaseEvent caught_events: List[EventMsg] = field(default_factory=list) + predicate: Callable[[EventMsg], bool] = lambda event: True def catch(self, event: EventMsg): - if event.info.name == self.event_to_catch.__name__: + if event.info.name == self.event_to_catch.__name__ and self.predicate(event): self.caught_events.append(event) def flush(self) -> None: