> ## Documentation Index
> Fetch the complete documentation index at: https://docs.cube.dev/llms.txt
> Use this file to discover all available pages before exploring further.

# Semantic Layer Sync

> Automatically synchronize your Cube data model into BI tools like Superset, Metabase, Preset, and Tableau.

Semantic Layer Sync

Semantic Layer Sync synchronizes the [data model][ref-data-model] of a semantic
layer from Cube to BI tools. It's the easiest way to connect a BI tool to Cube.

<Info>
  Support for Preset and Superset is available on [all plans](https://cube.dev/pricing).
  Support for Tableau is available on [Enterprise plan](https://cube.dev/pricing).
</Info>

Semantic Layer Sync programmatically connects a BI tool to Cube via the [SQL
API][ref-sql-api] and creates or updates BI-specific entities that correspond to
entities within the data model in Cube, e.g., cubes, views, measures,
dimensions.

In general, here's how Cube entities match BI-specific ones:

| Cube                        | BI tool                     |
| --------------------------- | --------------------------- |
| Deployment, branch          | Database                    |
| Cube, view                  | Dataset, table, data source |
| Measure, dimension, segment | Column, dimension, metric   |

## Supported tools

Semantic Layer Sync supports the following BI tools. Check relevant pages for details
on configuration and features for specific tools:

<CardGroup cols={2}>
  <Card title="Apache Superset" img="https://static.cube.dev/icons/superset.svg" href="/docs/integrations/semantic-layer-sync/superset" />

  <Card title="Preset" img="https://static.cube.dev/icons/preset.svg" href="/docs/integrations/semantic-layer-sync/preset" />

  <Card title="Tableau" img="https://static.cube.dev/icons/tableau.svg" href="/docs/integrations/semantic-layer-sync/tableau" />
</CardGroup>

## Creating syncs

You can create a new sync by navigating to **IDE → Integrations**
and clicking **+ Create Sync**.

Follow the steps in the wizard to create a sync with any of supported BI tools.

### Configuration

Under the hood, Semantic Layer Sync is configured using the `semantic_layer_sync`
option in the [configuration file][ref-config-file].

This function accepts a security context and returns an array of configured
syncs. It can also be asynchronous, allowing for dynamic definition of syncs,
e.g., loading the configuration from a remote API endpoint.

Each sync is configured with a mandatory `name` and a `type` as well as the
`config` object with BI-specific credentials, e.g., a workspace URL and an API
key. `config` also includes the `database` name that will be created and updated
in the corresponding BI tool.

A sync can be disabled by setting `active` to `false`; such syncs will not run
automatically. If `active` is undefined, a sync is considered enabled.

Example configuration with a single disabled sync:

<CodeGroup>
  ```python title="Python" theme={"dark"}
  from cube import config

  @config('semantic_layer_sync')
  def semantic_layer_sync(ctx: dict) -> list[dict]:
    return [
      {
        'type': 'superset',
        'name': 'Superset Sync',
        'active': False,
        'config': {
          'user': 'mail@example.com',
          'password': '4dceae-606a03-93ae6dc7',
          'url': 'superset.example.com',
          'database': 'Cube Cloud: staging-deployment'
        }
      }
    ]
  ```

  ```javascript title="JavaScript" theme={"dark"}
  module.exports = {
    semanticLayerSync: ({ securityContext }) => {
      return [
        {
          type: "superset",
          name: "Superset Sync",
          active: false,
          config: {
            user: "mail@example.com",
            password: "4dceae-606a03-93ae6dc7",
            url: "superset.example.com",
            database: "Cube Cloud: staging-deployment"
          }
        }
      ]
    }
  }
  ```
</CodeGroup>

### Multitenancy

If multiple security contexts are defined via the
[`scheduledRefreshContexts`][ref-config-contexts] configuration option,
`semanticLayerSync` can provide custom configuration for each of them.

<Warning>
  By default, multitenancy support in `semanticLayerSync` is disabled. Please
  contact support to enable multitenancy support in `semanticLayerSync` for your
  Cube Cloud account.
</Warning>

You can synchronize the data model of each tenant to a different BI tool or a
different database in a single BI tool, or implement another scenario that makes
sense in your use case.

Example configuration for the case when each department wants have their data
model synchronized to a dedicated database in a BI tool:

<CodeGroup>
  ```python title="Python" theme={"dark"}
  from cube import config

  @config('semantic_layer_sync')
  def semantic_layer_sync(ctx: dict) -> list[dict]:
    department = ctx['securityContext']['department']

    return [
      {
        'type': 'superset',
        'name': f"Superset Sync for {department}",
        'config': {
          'user': 'mail@example.com',
          'password': '4dceae-606a03-93ae6dc7',
          'url': 'superset.example.com',
          'database': f"Cube Cloud: {department}",
        }
      }
    ]
  ```

  ```javascript title="JavaScript" theme={"dark"}
  module.exports = {
    semanticLayerSync: ({ securityContext }) => {
      const department = securityContext.department

      return [
        {
          type: "superset",
          name: `Superset Sync for ${department}`,
          config: {
            user: "mail@example.com",
            password: "4dceae-606a03-93ae6dc7",
            url: "superset.example.com",
            database: `Cube Cloud: ${department}`
          }
        }
      ]
    }
  }
  ```
</CodeGroup>

Example confguration for the case when only `admin` data model should be
synchronized with a couple of BI tools:

<CodeGroup>
  ```python title="Python" theme={"dark"}
  from cube import config

  @config('semantic_layer_sync')
  def semantic_layer_sync(ctx: dict) -> list[dict]:
    if ctx['securityContext']['role'] == 'admin':
      return [
        {
          'type': 'superset',
          'name': 'Superset Sync',
          'config': {
            'user': 'mail@example.com',
            'password': '4dceae-606a03-93ae6dc7',
            'url': 'superset.example.com',
            'database': 'Cube Cloud: sls-test (admin)'
          }
        },
        {
          'type': 'preset',
          'name': 'Preset Sync',
          'config': {
            'api_token': '07988f63-c200-499e-97c9-ba137d8918aa',
            'api_secret': 'c19fbab4fd4945899795d32898f2e1165bef8e5ee653',
            'workspace_url': '12345678.us1a.app.preset.io',
            'database': 'Cube Cloud: sls-test (admin)'
          }
        }
      ]
    } else {
      # Only sync the 'admin' data model
      return []
    }
  ```

  ```javascript title="JavaScript" theme={"dark"}
  module.exports = {
    semanticLayerSync: ({ securityContext }) => {
      if (securityContext.role === "admin") {
        return [
          {
            type: "superset",
            name: "Superset Sync",
            config: {
              user: "mail@example.com",
              password: "4dceae-606a03-93ae6dc7",
              url: "superset.example.com",
              database: "Cube Cloud: sls-test (admin)"
            }
          },
          {
            type: "preset",
            name: "Preset Sync",
            config: {
              api_token: "07988f63-c200-499e-97c9-ba137d8918aa",
              api_secret: "c19fbab4fd4945899795d32898f2e1165bef8e5ee653",
              workspace_url: "12345678.us1a.app.preset.io",
              database: "Cube Cloud: sls-test (admin)"
            }
          }
              ]
      } else {
        // Only sync the 'admin' data model
              return []
      }
    }
  }
  ```
</CodeGroup>

## Running syncs

Cube automatically runs configured syncs every time a [new build is deployed](#on-build).
Additionally, you can configure syncs to run [on schedule](#on-schedule).
Finally, you can [manually](#manually) trigger any sync to run.

During the sync, Cube will either create from scratch or update a database in
the corresponding BI tool and the data model associated with it. Syncing a
branch in [development mode][ref-dev-mode] will create a separate database,
leaving the one associated with the production branch intact.

### On build

When the data model is updated, all configured syncs will automatically run.

<Warning>
  If data model is updated dynamically and the
  [`schema_version`][ref-config-schemaversion] configuration option is used to
  track data model changes, syncs will not automatically run. This behavior is
  disabled by default. Please contact support to enable running syncs when the
  data model is updated dynamically for your Cube Cloud account.
</Warning>

### On schedule

You can use the `scheduleInterval` option to specify the interval for running
a particular sync on schedule. This is useful for cases when you expect that
a BI tool might be temporarily unavailable via its API; the next scheduled sync
will likely succeed anyway.

<CodeGroup>
  ```python title="Python" theme={"dark"}
  from cube import config

  @config('semantic_layer_sync')
  def semantic_layer_sync(ctx: dict) -> list[dict]:
    return [
      {
        'type': 'superset',
        'name': 'Superset Sync',
        'active': False,
        'config': {
          'user': 'mail@example.com',
          'password': '4dceae-606a03-93ae6dc7',
          'url': 'superset.example.com',
          'database': 'Cube Cloud: staging-deployment',
          'scheduleInterval': 'every 10 minutes'
        }
      }
    ]
  ```

  ```javascript title="JavaScript" theme={"dark"}
  module.exports = {
    semanticLayerSync: ({ securityContext }) => {
      return [
        {
          type: "superset",
          name: "Superset Sync",
          active: false,
          config: {
            user: "mail@example.com",
            password: "4dceae-606a03-93ae6dc7",
            url: "superset.example.com",
            database: "Cube Cloud: staging-deployment",
            scheduleInterval: "every 10 minutes"
          }
        }
      ]
    }
  }
  ```
</CodeGroup>

The `scheduleInterval` option accepts values in the `every [number] [period]`
format, where `[number]` can be any positive integer, and `[period]` can be
any value from the following list: `day`, `hour`, `minute`, or `days`, `hours`,
`minutes`.

<Info>
  Shared deployments and [auto-suspended][ref-auto-sus] Dedicated deployments
  can't run syncs on schedule.
</Info>

### Manually

You can also run a sync manually by navigating to **IDE → Integrations**
and clicking **Start Sync** next to a relevant sync. This is convenient for working in the
[development mode][ref-dev-mode].

## Viewing history

You can view the history of runs for a particular sync by navigating to
**IDE → Integrations** and clicking **Show History** next to a
relevant sync.

## Cleaning up

You can remove the synced assets (e.g., data sources and datasets) from BI tools
by clicking on **...** of a respective tool and choosing **Remove synced
assets**. It would remove the assets synced from the branch you're currently
on, i.e., your development mode branch, shared branch, or main branch.

[ref-data-model]: /docs/data-modeling/overview

[ref-sql-api]: /reference/core-data-apis/sql-api

[ref-config-file]: /admin/connect-to-data#configuration-options

[ref-config-contexts]: /reference/configuration/config#scheduledrefreshcontexts

[ref-config-schemaversion]: /reference/configuration/config#schema_version

[ref-dev-mode]: /docs/data-modeling/dev-mode

[ref-auto-sus]: /admin/deployment/auto-suspension
