dataface 0.1.2__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- d3_format/__init__.py +14 -0
- d3_format/errors.py +19 -0
- d3_format/format.py +551 -0
- d3_format/spec.py +159 -0
- dataface/DATAFACE_SYNTAX.md +1135 -0
- dataface/__init__.py +93 -0
- dataface/_docs_site.py +20 -0
- dataface/_install_hint.py +26 -0
- dataface/agent_api/__init__.py +79 -0
- dataface/agent_api/_init_templates/__init__.py +0 -0
- dataface/agent_api/_init_templates/agents_dft_snippet.md +26 -0
- dataface/agent_api/_init_templates/dataface.yml +15 -0
- dataface/agent_api/_init_templates/faces-dataface.yml +144 -0
- dataface/agent_api/_init_templates/index.md +24 -0
- dataface/agent_api/_paths.py +118 -0
- dataface/agent_api/_project_agents_md.py +43 -0
- dataface/agent_api/_session_store.py +486 -0
- dataface/agent_api/_state.py +28 -0
- dataface/agent_api/chat.py +221 -0
- dataface/agent_api/dashboards.py +257 -0
- dataface/agent_api/describe.py +366 -0
- dataface/agent_api/describe_query.py +120 -0
- dataface/agent_api/docs/__init__.py +25 -0
- dataface/agent_api/docs/_loader.py +292 -0
- dataface/agent_api/docs/yaml-reference.md +2757 -0
- dataface/agent_api/file_refs.py +118 -0
- dataface/agent_api/init.py +126 -0
- dataface/agent_api/inspect.py +128 -0
- dataface/agent_api/mcp_install.py +170 -0
- dataface/agent_api/query.py +274 -0
- dataface/agent_api/schema.py +658 -0
- dataface/agent_api/schema_search.py +284 -0
- dataface/agent_api/search.py +270 -0
- dataface/agent_api/skill_install.py +141 -0
- dataface/agent_api/skill_render.py +90 -0
- dataface/agent_api/skills.py +293 -0
- dataface/agent_api/surface_aliases.yaml +128 -0
- dataface/agent_api/validate.py +175 -0
- dataface/agent_api/validate_query.py +84 -0
- dataface/ai/__init__.py +39 -0
- dataface/ai/agent.py +139 -0
- dataface/ai/context.py +45 -0
- dataface/ai/events.py +62 -0
- dataface/ai/external_mcp.py +610 -0
- dataface/ai/generate_sql.py +96 -0
- dataface/ai/llm.py +403 -0
- dataface/ai/mcp/__init__.py +51 -0
- dataface/ai/mcp/server.py +289 -0
- dataface/ai/memories.py +85 -0
- dataface/ai/prompts.py +177 -0
- dataface/ai/schema_context.py +138 -0
- dataface/ai/skills/before-after-comparison/SKILL.md +102 -0
- dataface/ai/skills/before-after-comparison/examples/before-after-comparison.yml +24 -0
- dataface/ai/skills/dashboard-build/SKILL.md +212 -0
- dataface/ai/skills/dashboard-build/examples/_smoke.yml +15 -0
- dataface/ai/skills/dashboard-design/SKILL.md +182 -0
- dataface/ai/skills/dashboard-review/SKILL.md +113 -0
- dataface/ai/skills/dashboard-structural-review/SKILL.md +173 -0
- dataface/ai/skills/dashboard-visual-review/SKILL.md +139 -0
- dataface/ai/skills/dataface-mcp-setup/SKILL.md +177 -0
- dataface/ai/skills/dataface-troubleshooting/SKILL.md +225 -0
- dataface/ai/skills/drill-down-link/SKILL.md +112 -0
- dataface/ai/skills/drill-down-link/examples/drill-down-link.yml +27 -0
- dataface/ai/skills/faceted-small-multiples/SKILL.md +116 -0
- dataface/ai/skills/faceted-small-multiples/examples/faceted-small-multiples.yml +33 -0
- dataface/ai/skills/filter-bar-with-variables/SKILL.md +105 -0
- dataface/ai/skills/filter-bar-with-variables/examples/filter-bar-with-variables.yml +49 -0
- dataface/ai/skills/kpi-row/SKILL.md +101 -0
- dataface/ai/skills/kpi-row/examples/kpi-row.yml +55 -0
- dataface/ai/skills/report-design/SKILL.md +184 -0
- dataface/ai/skills/single-metric-bignum/SKILL.md +90 -0
- dataface/ai/skills/single-metric-bignum/examples/single-metric-bignum.yml +27 -0
- dataface/ai/skills/table-heavy-ops-dashboard/SKILL.md +114 -0
- dataface/ai/skills/table-heavy-ops-dashboard/examples/table-heavy-ops-dashboard.yml +48 -0
- dataface/ai/skills/time-series-trend/SKILL.md +93 -0
- dataface/ai/skills/time-series-trend/examples/time-series-trend.yml +26 -0
- dataface/ai/skills/top-n-with-detail/SKILL.md +98 -0
- dataface/ai/skills/top-n-with-detail/examples/top-n-with-detail.yml +45 -0
- dataface/ai/skills/two-by-two-grid-overview/SKILL.md +78 -0
- dataface/ai/skills/two-by-two-grid-overview/examples/two-by-two-grid-overview.yml +64 -0
- dataface/ai/tool_schemas.py +132 -0
- dataface/ai/tools/__init__.py +312 -0
- dataface/ai/yaml_utils.py +57 -0
- dataface/cli/__init__.py +3 -0
- dataface/cli/_console.py +48 -0
- dataface/cli/_error_format.py +83 -0
- dataface/cli/_extras.py +190 -0
- dataface/cli/_json_output.py +8 -0
- dataface/cli/_parsing.py +17 -0
- dataface/cli/_version_info.py +56 -0
- dataface/cli/commands/__init__.py +3 -0
- dataface/cli/commands/_agent_input.py +205 -0
- dataface/cli/commands/_agent_server.py +115 -0
- dataface/cli/commands/chat.py +645 -0
- dataface/cli/commands/describe.py +107 -0
- dataface/cli/commands/docs.py +131 -0
- dataface/cli/commands/extension.py +179 -0
- dataface/cli/commands/init.py +240 -0
- dataface/cli/commands/inspect.py +94 -0
- dataface/cli/commands/mcp_init.py +167 -0
- dataface/cli/commands/query.py +386 -0
- dataface/cli/commands/render.py +291 -0
- dataface/cli/commands/schema.py +411 -0
- dataface/cli/commands/search.py +49 -0
- dataface/cli/commands/serve.py +114 -0
- dataface/cli/commands/skills.py +133 -0
- dataface/cli/commands/skills_init.py +161 -0
- dataface/cli/commands/validate.py +63 -0
- dataface/cli/main.py +1501 -0
- dataface/core/__init__.py +75 -0
- dataface/core/compile/__init__.py +244 -0
- dataface/core/compile/_jinja_helpers.py +78 -0
- dataface/core/compile/channel.py +222 -0
- dataface/core/compile/chart_focus.py +101 -0
- dataface/core/compile/chart_resolved.py +169 -0
- dataface/core/compile/chart_type_detection.py +489 -0
- dataface/core/compile/chart_update.py +261 -0
- dataface/core/compile/colors.py +64 -0
- dataface/core/compile/compiler.py +904 -0
- dataface/core/compile/config.py +823 -0
- dataface/core/compile/custom_chart_types.py +208 -0
- dataface/core/compile/data_table_attachment.py +1287 -0
- dataface/core/compile/detect.py +110 -0
- dataface/core/compile/errors.py +302 -0
- dataface/core/compile/filter_injection.py +319 -0
- dataface/core/compile/introspection.py +527 -0
- dataface/core/compile/jinja.py +511 -0
- dataface/core/compile/labels_env.py +52 -0
- dataface/core/compile/markdown.py +154 -0
- dataface/core/compile/meta.py +388 -0
- dataface/core/compile/models/__init__.py +0 -0
- dataface/core/compile/models/chart/__init__.py +0 -0
- dataface/core/compile/models/chart/authored.py +2137 -0
- dataface/core/compile/models/chart/compiled.py +398 -0
- dataface/core/compile/models/config.py +347 -0
- dataface/core/compile/models/face/__init__.py +0 -0
- dataface/core/compile/models/face/authored.py +659 -0
- dataface/core/compile/models/face/compiled.py +522 -0
- dataface/core/compile/models/factories.py +201 -0
- dataface/core/compile/models/markers.py +40 -0
- dataface/core/compile/models/palette.py +36 -0
- dataface/core/compile/models/primitives.py +415 -0
- dataface/core/compile/models/query/__init__.py +0 -0
- dataface/core/compile/models/query/authored.py +246 -0
- dataface/core/compile/models/query/compiled.py +710 -0
- dataface/core/compile/models/refs.py +137 -0
- dataface/core/compile/models/source.py +611 -0
- dataface/core/compile/models/style/__init__.py +0 -0
- dataface/core/compile/models/style/authored.py +481 -0
- dataface/core/compile/models/style/compiled.py +3399 -0
- dataface/core/compile/models/style/merged.py +1682 -0
- dataface/core/compile/models/theme.py +362 -0
- dataface/core/compile/models/variable/__init__.py +0 -0
- dataface/core/compile/models/variable/authored.py +254 -0
- dataface/core/compile/models/vega_lite/__init__.py +0 -0
- dataface/core/compile/models/vega_lite/config.py +510 -0
- dataface/core/compile/models/vega_lite/contracts.py +171 -0
- dataface/core/compile/normalize_charts.py +494 -0
- dataface/core/compile/normalize_layout.py +1000 -0
- dataface/core/compile/normalize_queries.py +297 -0
- dataface/core/compile/normalize_variables.py +489 -0
- dataface/core/compile/normalizer.py +543 -0
- dataface/core/compile/palette.py +1100 -0
- dataface/core/compile/parameterized.py +658 -0
- dataface/core/compile/parser.py +228 -0
- dataface/core/compile/schema.py +20 -0
- dataface/core/compile/schema_renderers/__init__.py +0 -0
- dataface/core/compile/schema_renderers/json_schema.py +163 -0
- dataface/core/compile/schema_renderers/prompt.py +152 -0
- dataface/core/compile/schema_renderers/vscode_schema.py +301 -0
- dataface/core/compile/sizing.py +2126 -0
- dataface/core/compile/sources.py +518 -0
- dataface/core/compile/sql_authoring_lint.py +56 -0
- dataface/core/compile/style_cascade.py +471 -0
- dataface/core/compile/typography.py +299 -0
- dataface/core/compile/validator.py +301 -0
- dataface/core/compile/variables.py +53 -0
- dataface/core/compile/vega_config.py +98 -0
- dataface/core/compile/vega_lite/__init__.py +6 -0
- dataface/core/compile/vega_lite/validation.py +95 -0
- dataface/core/compile/yaml_error_formatter.py +838 -0
- dataface/core/connections.py +38 -0
- dataface/core/dashboard.py +358 -0
- dataface/core/defaults/default_config.yml +101 -0
- dataface/core/defaults/palettes/categorical/category-10-dark.yml +32 -0
- dataface/core/defaults/palettes/categorical/category-10-light.yml +43 -0
- dataface/core/defaults/palettes/categorical/category-10.yml +31 -0
- dataface/core/defaults/palettes/categorical/category-6-tonal-blue.yml +22 -0
- dataface/core/defaults/palettes/categorical/category-6-tonal-brown.yml +29 -0
- dataface/core/defaults/palettes/categorical/category-6-tonal-green.yml +20 -0
- dataface/core/defaults/palettes/categorical/category-6-tonal-orange.yml +21 -0
- dataface/core/defaults/palettes/categorical/category-6-tonal-purple.yml +20 -0
- dataface/core/defaults/palettes/categorical/editorial-10-dark.yml +32 -0
- dataface/core/defaults/palettes/categorical/editorial-10.yml +40 -0
- dataface/core/defaults/palettes/categorical/hero-6.yml +17 -0
- dataface/core/defaults/palettes/categorical/single-blue.yml +11 -0
- dataface/core/defaults/palettes/categorical/tableau.yml +20 -0
- dataface/core/defaults/palettes/data/xkcd_colors.json +3803 -0
- dataface/core/defaults/palettes/diverging/blue-red.yml +25 -0
- dataface/core/defaults/palettes/diverging/coolwarm.yml +24 -0
- dataface/core/defaults/palettes/diverging/crimson-green.yml +23 -0
- dataface/core/defaults/palettes/diverging/orange-teal.yml +23 -0
- dataface/core/defaults/palettes/diverging/sunset.yml +24 -0
- dataface/core/defaults/palettes/scaffold/dft-creams.yml +38 -0
- dataface/core/defaults/palettes/scaffold/dft-grays.yml +53 -0
- dataface/core/defaults/palettes/sequential/amber.yml +22 -0
- dataface/core/defaults/palettes/sequential/blue.yml +22 -0
- dataface/core/defaults/palettes/sequential/brown.yml +22 -0
- dataface/core/defaults/palettes/sequential/gray.yml +22 -0
- dataface/core/defaults/palettes/sequential/green.yml +22 -0
- dataface/core/defaults/palettes/sequential/purple.yml +22 -0
- dataface/core/defaults/palettes/sequential/rust.yml +22 -0
- dataface/core/defaults/palettes/sequential/teal.yml +22 -0
- dataface/core/defaults/palettes/tone/negative.yml +32 -0
- dataface/core/defaults/palettes/tone/positive.yml +22 -0
- dataface/core/defaults/palettes/tone/warning.yml +22 -0
- dataface/core/defaults/themes/_base.yaml +786 -0
- dataface/core/defaults/themes/bi.yaml +16 -0
- dataface/core/defaults/themes/carbong100.yaml +41 -0
- dataface/core/defaults/themes/cream.yaml +122 -0
- dataface/core/defaults/themes/dark.yaml +40 -0
- dataface/core/defaults/themes/diagnostics-title-angle-extreme.yaml +9 -0
- dataface/core/defaults/themes/diagnostics-title-baseline-extreme.yaml +9 -0
- dataface/core/defaults/themes/diagnostics-title-baseline.yaml +24 -0
- dataface/core/defaults/themes/diagnostics-title-center.yaml +8 -0
- dataface/core/defaults/themes/diagnostics-title-color-extreme.yaml +24 -0
- dataface/core/defaults/themes/diagnostics-title-font-extreme.yaml +25 -0
- dataface/core/defaults/themes/diagnostics-title-left.yaml +8 -0
- dataface/core/defaults/themes/diagnostics-title-offset-extreme.yaml +9 -0
- dataface/core/defaults/themes/diagnostics-title-size-extreme.yaml +24 -0
- dataface/core/defaults/themes/diagnostics-title-weight-extreme.yaml +24 -0
- dataface/core/defaults/themes/editorial.yaml +147 -0
- dataface/core/defaults/themes/light.yaml +30 -0
- dataface/core/defaults/themes/looker.yaml +17 -0
- dataface/core/defaults/themes/stark.yaml +134 -0
- dataface/core/errors/__init__.py +67 -0
- dataface/core/errors/codes_compile.py +56 -0
- dataface/core/errors/codes_execute.py +177 -0
- dataface/core/errors/codes_render.py +106 -0
- dataface/core/errors/codes_unknown.py +15 -0
- dataface/core/errors/hints.py +74 -0
- dataface/core/errors/registry.py +42 -0
- dataface/core/errors/structured.py +92 -0
- dataface/core/execute/__init__.py +91 -0
- dataface/core/execute/adapters/__init__.py +49 -0
- dataface/core/execute/adapters/adapter_registry.py +400 -0
- dataface/core/execute/adapters/base.py +245 -0
- dataface/core/execute/adapters/csv_adapter.py +239 -0
- dataface/core/execute/adapters/dbt_adapter.py +283 -0
- dataface/core/execute/adapters/dbt_adapter_factory.py +212 -0
- dataface/core/execute/adapters/dbt_macro_loader.py +95 -0
- dataface/core/execute/adapters/dbt_utils.py +150 -0
- dataface/core/execute/adapters/http_adapter.py +224 -0
- dataface/core/execute/adapters/metricflow_adapter.py +94 -0
- dataface/core/execute/adapters/schema_resolver_adapter.py +144 -0
- dataface/core/execute/adapters/sql_adapter.py +710 -0
- dataface/core/execute/adapters/values_adapter.py +58 -0
- dataface/core/execute/batch.py +744 -0
- dataface/core/execute/cache_backend.py +135 -0
- dataface/core/execute/cache_keys.py +66 -0
- dataface/core/execute/dbt_jinja.py +21 -0
- dataface/core/execute/dialects/__init__.py +121 -0
- dataface/core/execute/dialects/athena.py +75 -0
- dataface/core/execute/dialects/base.py +302 -0
- dataface/core/execute/dialects/bigquery.py +38 -0
- dataface/core/execute/dialects/databricks.py +68 -0
- dataface/core/execute/dialects/duckdb.py +35 -0
- dataface/core/execute/dialects/mysql.py +68 -0
- dataface/core/execute/dialects/postgres.py +39 -0
- dataface/core/execute/dialects/redshift.py +12 -0
- dataface/core/execute/dialects/snowflake.py +51 -0
- dataface/core/execute/dialects/sqlserver.py +92 -0
- dataface/core/execute/duckdb_cache.py +712 -0
- dataface/core/execute/duckdb_config.py +26 -0
- dataface/core/execute/errors.py +213 -0
- dataface/core/execute/executor.py +1249 -0
- dataface/core/execute/parallel.py +162 -0
- dataface/core/execute/setup_sql.py +58 -0
- dataface/core/execute/source_registry.py +72 -0
- dataface/core/execute/source_resolver.py +255 -0
- dataface/core/execute/sql_guard.py +387 -0
- dataface/core/execute/sql_literals.py +199 -0
- dataface/core/fonts.py +52 -0
- dataface/core/inspect/__init__.py +32 -0
- dataface/core/inspect/cache_factory.py +98 -0
- dataface/core/inspect/db_types.py +162 -0
- dataface/core/inspect/dbt_schema.py +96 -0
- dataface/core/inspect/defaults.yml +37 -0
- dataface/core/inspect/fanout_risk.py +109 -0
- dataface/core/inspect/manifest_utils.py +77 -0
- dataface/core/inspect/partials/categorical.yml +40 -0
- dataface/core/inspect/partials/date.yml +40 -0
- dataface/core/inspect/partials/numeric.yml +55 -0
- dataface/core/inspect/partition_types.py +38 -0
- dataface/core/inspect/query_validator.py +975 -0
- dataface/core/inspect/renderer.py +354 -0
- dataface/core/inspect/resolver.py +808 -0
- dataface/core/inspect/search.py +461 -0
- dataface/core/inspect/sources/__init__.py +32 -0
- dataface/core/inspect/sources/dbt.py +738 -0
- dataface/core/inspect/sources/duckdb_utils.py +66 -0
- dataface/core/inspect/templates/__init__.py +1 -0
- dataface/core/inspect/templates/categorical_column.yml +196 -0
- dataface/core/inspect/templates/charts.yml +109 -0
- dataface/core/inspect/templates/date_column.yml +248 -0
- dataface/core/inspect/templates/model.yml +138 -0
- dataface/core/inspect/templates/numeric_column.yml +261 -0
- dataface/core/inspect/templates/quality.yml +80 -0
- dataface/core/inspect/templates/string_column.yml +263 -0
- dataface/core/project_roots.py +165 -0
- dataface/core/render/__init__.py +87 -0
- dataface/core/render/board_links.py +176 -0
- dataface/core/render/chart/__init__.py +27 -0
- dataface/core/render/chart/arc_attached_table.py +251 -0
- dataface/core/render/chart/artifacts.py +16 -0
- dataface/core/render/chart/callout.py +225 -0
- dataface/core/render/chart/decisions.py +358 -0
- dataface/core/render/chart/geo.py +700 -0
- dataface/core/render/chart/kpi.py +916 -0
- dataface/core/render/chart/labels.py +76 -0
- dataface/core/render/chart/pipeline.py +818 -0
- dataface/core/render/chart/presentation.py +36 -0
- dataface/core/render/chart/profile.py +3438 -0
- dataface/core/render/chart/render_single.py +347 -0
- dataface/core/render/chart/renderers.py +193 -0
- dataface/core/render/chart/rendering.py +565 -0
- dataface/core/render/chart/serialization.py +90 -0
- dataface/core/render/chart/spark.py +496 -0
- dataface/core/render/chart/spark_bar.py +370 -0
- dataface/core/render/chart/spec_builders.py +154 -0
- dataface/core/render/chart/standard_renderer.py +2645 -0
- dataface/core/render/chart/table.py +2957 -0
- dataface/core/render/chart/table_support.py +1452 -0
- dataface/core/render/chart/tick_values.py +66 -0
- dataface/core/render/chart/time_unit_detect.py +809 -0
- dataface/core/render/chart/title_overflow.py +157 -0
- dataface/core/render/chart/type_inference.py +122 -0
- dataface/core/render/chart/validation.py +99 -0
- dataface/core/render/chart/vega_lite.py +125 -0
- dataface/core/render/chart/vega_lite_types.py +268 -0
- dataface/core/render/chart/vl_field_maps.py +346 -0
- dataface/core/render/chart_interactivity.py +24 -0
- dataface/core/render/control_registry.py +287 -0
- dataface/core/render/converters/__init__.py +24 -0
- dataface/core/render/converters/chart.py +276 -0
- dataface/core/render/converters/html.py +98 -0
- dataface/core/render/converters/pdf.py +40 -0
- dataface/core/render/converters/png.py +41 -0
- dataface/core/render/errors.py +144 -0
- dataface/core/render/face_api.py +160 -0
- dataface/core/render/faces.py +1194 -0
- dataface/core/render/font_measurement.py +48 -0
- dataface/core/render/font_support.py +197 -0
- dataface/core/render/fonts/DFTSansTabular-Regular.ttf +0 -0
- dataface/core/render/fonts/DFTSansTabular-Regular.woff2 +0 -0
- dataface/core/render/fonts/DFTSerifOldstyleProportional-Regular.ttf +0 -0
- dataface/core/render/fonts/DFTSerifOldstyleTabular-Regular.ttf +0 -0
- dataface/core/render/fonts/InterVariable.ttf +0 -0
- dataface/core/render/fonts/InterVariable.woff2 +0 -0
- dataface/core/render/fonts/NOTO_COLOR_EMOJI_LICENSE.txt +93 -0
- dataface/core/render/fonts/NOTO_EMOJI_LICENSE.txt +93 -0
- dataface/core/render/fonts/NotoColorEmoji-Regular.ttf +0 -0
- dataface/core/render/fonts/NotoColorEmoji-Regular.woff2 +0 -0
- dataface/core/render/fonts/NotoEmoji-Regular.ttf +0 -0
- dataface/core/render/fonts/NotoEmoji-Regular.woff2 +0 -0
- dataface/core/render/fonts/SOURCE_CODE_PRO_LICENSE.txt +93 -0
- dataface/core/render/fonts/SOURCE_SERIF_4_LICENSE.txt +98 -0
- dataface/core/render/fonts/SourceCodePro-Regular.ttf +0 -0
- dataface/core/render/fonts/SourceSerif4-Regular.ttf +0 -0
- dataface/core/render/fonts/_emoji_font_face.css +43 -0
- dataface/core/render/fonts/source-serif-4-variable-latin.woff2 +0 -0
- dataface/core/render/format_utils.py +329 -0
- dataface/core/render/geo_defaults.yml +28 -0
- dataface/core/render/json_format.py +146 -0
- dataface/core/render/layout_sizing.py +865 -0
- dataface/core/render/layouts.py +541 -0
- dataface/core/render/markdown_defaults.yml +16 -0
- dataface/core/render/missing_vars_prompt.py +79 -0
- dataface/core/render/placeholder.py +389 -0
- dataface/core/render/render_result.py +14 -0
- dataface/core/render/renderer.py +467 -0
- dataface/core/render/script_embedding.py +16 -0
- dataface/core/render/svg_utils.py +212 -0
- dataface/core/render/template_loader.py +69 -0
- dataface/core/render/templates/controls/_styles.css +606 -0
- dataface/core/render/templates/controls/checkbox.html +16 -0
- dataface/core/render/templates/controls/date.html +16 -0
- dataface/core/render/templates/controls/number.html +19 -0
- dataface/core/render/templates/controls/readonly.html +9 -0
- dataface/core/render/templates/controls/select.html +21 -0
- dataface/core/render/templates/controls/slider.html +22 -0
- dataface/core/render/templates/controls/text.html +16 -0
- dataface/core/render/templates/scripts/chart_interactivity.js +191 -0
- dataface/core/render/templates/scripts/variables.js +976 -0
- dataface/core/render/templates/svg/grid_pattern.svg +3 -0
- dataface/core/render/templates/svg/styles.css +51 -0
- dataface/core/render/terminal.py +311 -0
- dataface/core/render/terminal_charts.py +563 -0
- dataface/core/render/terminal_defaults.yml +2 -0
- dataface/core/render/terminal_layouts.py +299 -0
- dataface/core/render/terminal_text.py +31 -0
- dataface/core/render/text/__init__.py +1 -0
- dataface/core/render/text/case.py +113 -0
- dataface/core/render/text_format.py +129 -0
- dataface/core/render/utils.py +106 -0
- dataface/core/render/variable_controls.py +946 -0
- dataface/core/render/variable_input_refinement.py +140 -0
- dataface/core/render/warnings/__init__.py +15 -0
- dataface/core/render/warnings/bar_color_1_to_1_with_x.py +80 -0
- dataface/core/render/warnings/base.py +44 -0
- dataface/core/render/warnings/fanout_risk.py +15 -0
- dataface/core/render/warnings/from_query_diagnostic.py +56 -0
- dataface/core/render/warnings/missing_join_predicate.py +13 -0
- dataface/core/render/warnings/query_parse_error.py +14 -0
- dataface/core/render/warnings/query_returned_zero_rows.py +42 -0
- dataface/core/render/warnings/reaggregation.py +14 -0
- dataface/core/render/warnings/registry.py +45 -0
- dataface/core/render/warnings/suppression.py +46 -0
- dataface/core/render/warnings/temporal_single_point.py +63 -0
- dataface/core/render/warnings/unreferenced_chart.py +15 -0
- dataface/core/render/warnings/y_encoding_mostly_null.py +76 -0
- dataface/core/render/yaml_format.py +167 -0
- dataface/core/resolve_face.py +195 -0
- dataface/core/schema/__init__.py +0 -0
- dataface/core/schema/guidance.py +151 -0
- dataface/core/scoped_paths.py +59 -0
- dataface/core/serve/__init__.py +14 -0
- dataface/core/serve/bootstrap.py +39 -0
- dataface/core/serve/embedded.py +57 -0
- dataface/core/serve/port.py +129 -0
- dataface/core/serve/server.py +938 -0
- dataface/core/serve/templates/__init__.py +0 -0
- dataface/core/serve/templates/directory.yml +6 -0
- dataface/core/serve/templates/error.html.j2 +217 -0
- dataface/core/utils.py +121 -0
- dataface/core/validate.py +64 -0
- dataface/integrations/__init__.py +0 -0
- dataface/integrations/highlighting.py +351 -0
- dataface/integrations/markdown.py +537 -0
- dataface/py.typed +0 -0
- dataface-0.1.2.dist-info/METADATA +375 -0
- dataface-0.1.2.dist-info/RECORD +455 -0
- dataface-0.1.2.dist-info/WHEEL +4 -0
- dataface-0.1.2.dist-info/entry_points.txt +2 -0
- dataface-0.1.2.dist-info/licenses/LICENSE +202 -0
- mdsvg/__init__.py +168 -0
- mdsvg/fonts.py +656 -0
- mdsvg/images.py +299 -0
- mdsvg/parser.py +629 -0
- mdsvg/playground.py +284 -0
- mdsvg/py.typed +2 -0
- mdsvg/renderer.py +1623 -0
- mdsvg/style.py +355 -0
- mdsvg/types.py +200 -0
- mdsvg/utils.py +86 -0
|
@@ -0,0 +1,351 @@
|
|
|
1
|
+
"""Dataface YAML Pygments lexer with embedded SQL block highlighting.
|
|
2
|
+
|
|
3
|
+
The lexer handles Dataface face YAML and delegates the body of SQL block
|
|
4
|
+
scalars (``sql: |``, ``sql: >``, ``query: |``, ``query: >``) to
|
|
5
|
+
``pygments.lexers.SqlLexer`` so SQL keywords are visually distinct from
|
|
6
|
+
the surrounding YAML structure.
|
|
7
|
+
|
|
8
|
+
Public API
|
|
9
|
+
----------
|
|
10
|
+
- ``DatafaceYamlLexer`` — Pygments lexer class. Reference it from
|
|
11
|
+
MkDocs / pymdownx config via
|
|
12
|
+
``!!python/name:dataface.integrations.highlighting.DatafaceYamlLexer``.
|
|
13
|
+
- ``highlight_face_yaml(source)`` — Highlight Dataface YAML and return an
|
|
14
|
+
HTML string.
|
|
15
|
+
"""
|
|
16
|
+
|
|
17
|
+
from __future__ import annotations
|
|
18
|
+
|
|
19
|
+
import re
|
|
20
|
+
from collections.abc import Generator
|
|
21
|
+
|
|
22
|
+
from pygments import highlight as pygments_highlight
|
|
23
|
+
from pygments.formatters import HtmlFormatter
|
|
24
|
+
from pygments.lexer import RegexLexer, bygroups
|
|
25
|
+
from pygments.lexers import SqlLexer
|
|
26
|
+
from pygments.token import (
|
|
27
|
+
Comment,
|
|
28
|
+
Keyword,
|
|
29
|
+
Name,
|
|
30
|
+
Number,
|
|
31
|
+
Punctuation,
|
|
32
|
+
String,
|
|
33
|
+
Text,
|
|
34
|
+
)
|
|
35
|
+
|
|
36
|
+
# ---------------------------------------------------------------------------
|
|
37
|
+
# Constants
|
|
38
|
+
# ---------------------------------------------------------------------------
|
|
39
|
+
|
|
40
|
+
# Keys that open a SQL block: "sql: |", "query: |", "sql: >", "query: >"
|
|
41
|
+
# Indent captured in group 1; key in group 2.
|
|
42
|
+
_SQL_BLOCK_OPEN_RE = re.compile(
|
|
43
|
+
r"^( *)(?:sql|query)\s*:\s*[|>]",
|
|
44
|
+
re.MULTILINE,
|
|
45
|
+
)
|
|
46
|
+
|
|
47
|
+
# Detect a plain YAML key: <indent><word chars>:
|
|
48
|
+
_YAML_KEY_RE = re.compile(r"^( *)([a-zA-Z_][a-zA-Z0-9_\-]*)\s*:")
|
|
49
|
+
|
|
50
|
+
# Detect a list-item YAML key: <indent>- <word chars>:
|
|
51
|
+
# Effective indent is the column of the dash (group 1 length), not the key itself.
|
|
52
|
+
_YAML_LIST_KEY_RE = re.compile(r"^( *)-\s+([a-zA-Z_][a-zA-Z0-9_\-]*)\s*:")
|
|
53
|
+
|
|
54
|
+
|
|
55
|
+
def _sql_block_ranges(source: str) -> list[tuple[int, int]]:
|
|
56
|
+
"""Return (start, end) byte offsets of SQL block scalar bodies in *source*.
|
|
57
|
+
|
|
58
|
+
The body starts on the line after the ``sql: |`` / ``query: |`` line and
|
|
59
|
+
ends just before the next YAML key at the same or lower indent level (or
|
|
60
|
+
at EOF). Offsets are character positions in *source*.
|
|
61
|
+
"""
|
|
62
|
+
ranges: list[tuple[int, int]] = []
|
|
63
|
+
lines = source.splitlines(keepends=True)
|
|
64
|
+
line_starts: list[int] = []
|
|
65
|
+
pos = 0
|
|
66
|
+
for line in lines:
|
|
67
|
+
line_starts.append(pos)
|
|
68
|
+
pos += len(line)
|
|
69
|
+
|
|
70
|
+
i = 0
|
|
71
|
+
while i < len(lines):
|
|
72
|
+
line = lines[i]
|
|
73
|
+
m = _SQL_BLOCK_OPEN_RE.match(line)
|
|
74
|
+
if m:
|
|
75
|
+
opener_indent = len(m.group(1))
|
|
76
|
+
body_start_line = i + 1
|
|
77
|
+
# Find end: first line at indent <= opener_indent that is a YAML key.
|
|
78
|
+
# Also treat " - key:" list-item lines as terminators; their effective
|
|
79
|
+
# indent is the dash column (group 1 length), not the key column.
|
|
80
|
+
j = body_start_line
|
|
81
|
+
while j < len(lines):
|
|
82
|
+
candidate = lines[j]
|
|
83
|
+
km = _YAML_KEY_RE.match(candidate)
|
|
84
|
+
if km and len(km.group(1)) <= opener_indent:
|
|
85
|
+
break
|
|
86
|
+
lm = _YAML_LIST_KEY_RE.match(candidate)
|
|
87
|
+
if lm and len(lm.group(1)) <= opener_indent:
|
|
88
|
+
break
|
|
89
|
+
j += 1
|
|
90
|
+
if body_start_line < j:
|
|
91
|
+
start = line_starts[body_start_line]
|
|
92
|
+
end = line_starts[j] if j < len(line_starts) else pos
|
|
93
|
+
ranges.append((start, end))
|
|
94
|
+
i = j
|
|
95
|
+
else:
|
|
96
|
+
i += 1
|
|
97
|
+
return ranges
|
|
98
|
+
|
|
99
|
+
|
|
100
|
+
class DatafaceYamlLexer(RegexLexer):
|
|
101
|
+
"""Pygments lexer for Dataface face YAML with embedded SQL in block scalars.
|
|
102
|
+
|
|
103
|
+
Top-level and nested keys, chart types, input types, and other Dataface
|
|
104
|
+
vocabulary are coloured distinctly. Lines inside ``sql: |``, ``sql: >``,
|
|
105
|
+
``query: |``, and ``query: >`` block scalars are delegated to
|
|
106
|
+
``SqlLexer`` so SQL keywords (SELECT, FROM, WHERE, …) receive keyword
|
|
107
|
+
styling.
|
|
108
|
+
"""
|
|
109
|
+
|
|
110
|
+
name = "Dataface YAML"
|
|
111
|
+
aliases = ["dataface", "dataface-yaml", "dfyaml"]
|
|
112
|
+
filenames = ["*.yml", "*.yaml"]
|
|
113
|
+
mimetypes = ["text/x-dataface-yaml"]
|
|
114
|
+
|
|
115
|
+
TOP_LEVEL_KEYWORDS = [
|
|
116
|
+
"title",
|
|
117
|
+
"description",
|
|
118
|
+
"tags",
|
|
119
|
+
"content",
|
|
120
|
+
"variables",
|
|
121
|
+
"queries",
|
|
122
|
+
"charts",
|
|
123
|
+
"rows",
|
|
124
|
+
"cols",
|
|
125
|
+
"grid",
|
|
126
|
+
"tabs",
|
|
127
|
+
"theme",
|
|
128
|
+
"id",
|
|
129
|
+
"style",
|
|
130
|
+
"width",
|
|
131
|
+
"height",
|
|
132
|
+
]
|
|
133
|
+
CHART_PROPERTIES = [
|
|
134
|
+
"type",
|
|
135
|
+
"query",
|
|
136
|
+
"x",
|
|
137
|
+
"y",
|
|
138
|
+
"color",
|
|
139
|
+
"size",
|
|
140
|
+
"shape",
|
|
141
|
+
"value",
|
|
142
|
+
"glyph",
|
|
143
|
+
"tone",
|
|
144
|
+
"support",
|
|
145
|
+
"limit",
|
|
146
|
+
"interactions",
|
|
147
|
+
"filters",
|
|
148
|
+
]
|
|
149
|
+
CHART_TYPES = [
|
|
150
|
+
"line",
|
|
151
|
+
"bar",
|
|
152
|
+
"area",
|
|
153
|
+
"scatter",
|
|
154
|
+
"table",
|
|
155
|
+
"kpi",
|
|
156
|
+
"heatmap",
|
|
157
|
+
"pie",
|
|
158
|
+
"donut",
|
|
159
|
+
]
|
|
160
|
+
VARIABLE_PROPERTIES = [
|
|
161
|
+
"input",
|
|
162
|
+
"label",
|
|
163
|
+
"description",
|
|
164
|
+
"default",
|
|
165
|
+
"placeholder",
|
|
166
|
+
"required",
|
|
167
|
+
"allow_null",
|
|
168
|
+
"column",
|
|
169
|
+
"dimension",
|
|
170
|
+
"measure",
|
|
171
|
+
"model",
|
|
172
|
+
"options",
|
|
173
|
+
"min",
|
|
174
|
+
"max",
|
|
175
|
+
"step",
|
|
176
|
+
"operator",
|
|
177
|
+
]
|
|
178
|
+
VARIABLE_INPUT_TYPES = [
|
|
179
|
+
"select",
|
|
180
|
+
"multiselect",
|
|
181
|
+
"input",
|
|
182
|
+
"textarea",
|
|
183
|
+
"slider",
|
|
184
|
+
"datepicker",
|
|
185
|
+
"daterange",
|
|
186
|
+
"checkbox",
|
|
187
|
+
"radio",
|
|
188
|
+
]
|
|
189
|
+
QUERY_PROPERTIES = [
|
|
190
|
+
"sql",
|
|
191
|
+
"model",
|
|
192
|
+
"columns",
|
|
193
|
+
"metrics",
|
|
194
|
+
"dimensions",
|
|
195
|
+
"filters",
|
|
196
|
+
"time_grain",
|
|
197
|
+
"limit",
|
|
198
|
+
"url",
|
|
199
|
+
"method",
|
|
200
|
+
"headers",
|
|
201
|
+
"params",
|
|
202
|
+
"body",
|
|
203
|
+
]
|
|
204
|
+
LAYOUT_PROPERTIES = [
|
|
205
|
+
"columns",
|
|
206
|
+
"row_height",
|
|
207
|
+
"gap",
|
|
208
|
+
"default_width",
|
|
209
|
+
"default_height",
|
|
210
|
+
"items",
|
|
211
|
+
"position",
|
|
212
|
+
"default",
|
|
213
|
+
]
|
|
214
|
+
INTERACTION_PROPERTIES = [
|
|
215
|
+
"by",
|
|
216
|
+
"order",
|
|
217
|
+
"action",
|
|
218
|
+
"target",
|
|
219
|
+
"value_map",
|
|
220
|
+
"click",
|
|
221
|
+
"hover",
|
|
222
|
+
]
|
|
223
|
+
STYLE_PROPERTIES = [
|
|
224
|
+
"palette",
|
|
225
|
+
"legend",
|
|
226
|
+
"disable",
|
|
227
|
+
"labels",
|
|
228
|
+
"x_axis",
|
|
229
|
+
"y_axis",
|
|
230
|
+
]
|
|
231
|
+
SORT_ORDERS = ["asc", "desc"]
|
|
232
|
+
ACTIONS = ["filter", "open_link", "set_variable", "drill"]
|
|
233
|
+
TIME_GRAINS = ["day", "week", "month", "quarter", "year"]
|
|
234
|
+
HTTP_METHODS = ["GET", "POST", "PUT", "DELETE", "PATCH"]
|
|
235
|
+
LEGEND_VALUES = ["show", "hide"]
|
|
236
|
+
LABEL_VALUES = ["auto", "show", "hide"]
|
|
237
|
+
GAP_VALUES = ["sm", "md", "lg", "xl"]
|
|
238
|
+
POSITION_VALUES = ["top", "left"]
|
|
239
|
+
|
|
240
|
+
ALL_KEYWORDS = (
|
|
241
|
+
TOP_LEVEL_KEYWORDS
|
|
242
|
+
+ CHART_PROPERTIES
|
|
243
|
+
+ CHART_TYPES
|
|
244
|
+
+ VARIABLE_PROPERTIES
|
|
245
|
+
+ VARIABLE_INPUT_TYPES
|
|
246
|
+
+ QUERY_PROPERTIES
|
|
247
|
+
+ LAYOUT_PROPERTIES
|
|
248
|
+
+ INTERACTION_PROPERTIES
|
|
249
|
+
+ STYLE_PROPERTIES
|
|
250
|
+
+ SORT_ORDERS
|
|
251
|
+
+ ACTIONS
|
|
252
|
+
+ TIME_GRAINS
|
|
253
|
+
+ HTTP_METHODS
|
|
254
|
+
+ LEGEND_VALUES
|
|
255
|
+
+ LABEL_VALUES
|
|
256
|
+
+ GAP_VALUES
|
|
257
|
+
+ POSITION_VALUES
|
|
258
|
+
)
|
|
259
|
+
KEYWORD_PATTERN = r"\b(" + "|".join(ALL_KEYWORDS) + r")\b"
|
|
260
|
+
|
|
261
|
+
tokens = {
|
|
262
|
+
"root": [
|
|
263
|
+
(r"#.*$", Comment),
|
|
264
|
+
(r"\{#[^#]*#\}", Comment),
|
|
265
|
+
(r"^---\s*$", Punctuation),
|
|
266
|
+
(r"^\.\.\.\s*$", Punctuation),
|
|
267
|
+
(r'"[^"]*"', String.Double),
|
|
268
|
+
(r"'[^']*'", String.Single),
|
|
269
|
+
(r"\{\{[^}]*\}\}", String.Interpol),
|
|
270
|
+
(r"!![^\s]+", Keyword.Type),
|
|
271
|
+
(r"&[^\s]+", Name.Variable),
|
|
272
|
+
(r"\*[^\s]+", Name.Variable),
|
|
273
|
+
(r"([a-zA-Z_][a-zA-Z0-9_\-]*)(\s*:)", bygroups(Name.Tag, Punctuation)),
|
|
274
|
+
(r"^\s*-\s+", Punctuation),
|
|
275
|
+
(r"[\[\]{}]", Punctuation),
|
|
276
|
+
(r":", Punctuation),
|
|
277
|
+
(r"\|", Punctuation),
|
|
278
|
+
(r">", Punctuation),
|
|
279
|
+
(r"[-+]", Punctuation),
|
|
280
|
+
(r"\d+\.\d+", Number.Float),
|
|
281
|
+
(r"\d+", Number.Integer),
|
|
282
|
+
(r"\b(true|false|null|yes|no|on|off)\b", Keyword.Constant),
|
|
283
|
+
(KEYWORD_PATTERN, Keyword),
|
|
284
|
+
(r"[^\s:]+", Text),
|
|
285
|
+
(r"\s+", Text),
|
|
286
|
+
]
|
|
287
|
+
}
|
|
288
|
+
|
|
289
|
+
def get_tokens_unprocessed(
|
|
290
|
+
self, text: str, stack: tuple[str, ...] = ("root",)
|
|
291
|
+
) -> Generator[tuple[int, object, str], None, None]:
|
|
292
|
+
"""Yield tokens, delegating SQL block bodies to SqlLexer.
|
|
293
|
+
|
|
294
|
+
For each character range identified as a SQL block body, the
|
|
295
|
+
corresponding text is re-tokenised with SqlLexer; the rest goes
|
|
296
|
+
through the normal YAML rules.
|
|
297
|
+
|
|
298
|
+
All yielded offsets are positions in *text* (the original string),
|
|
299
|
+
not positions within a chunk — required by the Pygments contract.
|
|
300
|
+
"""
|
|
301
|
+
sql_ranges = _sql_block_ranges(text)
|
|
302
|
+
if not sql_ranges:
|
|
303
|
+
yield from super().get_tokens_unprocessed(text, stack)
|
|
304
|
+
return
|
|
305
|
+
|
|
306
|
+
sql_lexer = SqlLexer()
|
|
307
|
+
cursor = 0
|
|
308
|
+
|
|
309
|
+
for sql_start, sql_end in sql_ranges:
|
|
310
|
+
# Yield YAML tokens for the segment before this SQL block,
|
|
311
|
+
# adjusting chunk-relative offsets back to text-absolute positions.
|
|
312
|
+
if cursor < sql_start:
|
|
313
|
+
yaml_chunk = text[cursor:sql_start]
|
|
314
|
+
for chunk_offset, ttype, value in super().get_tokens_unprocessed(
|
|
315
|
+
yaml_chunk, stack
|
|
316
|
+
):
|
|
317
|
+
yield cursor + chunk_offset, ttype, value
|
|
318
|
+
|
|
319
|
+
# Yield SQL tokens for the block body, offset-adjusted
|
|
320
|
+
sql_chunk = text[sql_start:sql_end]
|
|
321
|
+
for offset, ttype, value in sql_lexer.get_tokens_unprocessed(sql_chunk):
|
|
322
|
+
yield sql_start + offset, ttype, value
|
|
323
|
+
|
|
324
|
+
cursor = sql_end
|
|
325
|
+
|
|
326
|
+
# Remaining YAML after the last SQL block
|
|
327
|
+
if cursor < len(text):
|
|
328
|
+
yaml_tail = text[cursor:]
|
|
329
|
+
for chunk_offset, ttype, value in super().get_tokens_unprocessed(
|
|
330
|
+
yaml_tail, stack
|
|
331
|
+
):
|
|
332
|
+
yield cursor + chunk_offset, ttype, value
|
|
333
|
+
|
|
334
|
+
|
|
335
|
+
def highlight_face_yaml(source: str) -> str:
|
|
336
|
+
"""Highlight Dataface face YAML and return an HTML string.
|
|
337
|
+
|
|
338
|
+
SQL inside ``sql: |``, ``sql: >``, ``query: |``, and ``query: >``
|
|
339
|
+
block scalars is coloured with SQL keyword rules. The wrapping HTML
|
|
340
|
+
uses ``cssclass="highlight"`` for compatibility with the existing
|
|
341
|
+
``extra.css`` rules.
|
|
342
|
+
|
|
343
|
+
Args:
|
|
344
|
+
source: Raw YAML source text.
|
|
345
|
+
|
|
346
|
+
Returns:
|
|
347
|
+
HTML string with Pygments token spans.
|
|
348
|
+
"""
|
|
349
|
+
lexer = DatafaceYamlLexer()
|
|
350
|
+
formatter = HtmlFormatter(cssclass="highlight", nowrap=True)
|
|
351
|
+
return pygments_highlight(source, lexer, formatter)
|