databricks-labs-ucx


Namedatabricks-labs-ucx JSON
Version 0.53.1 PyPI version JSON
download
home_pageNone
SummaryUCX - Unity Catalog Migration Toolkit
upload_time2024-12-30 16:58:14
maintainerNone
docs_urlNone
authorNone
requires_python>=3.10
licenseNone
keywords databricks unity catalog
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            Databricks Labs UCX
===
![UCX by Databricks Labs](docs/logo-no-background.png)

The companion for upgrading to Unity Catalog (UC).

After [installation](#install-ucx), ensure to [trigger](#ensure-assessment-run-command) the [assessment workflow](#assessment-workflow),
so that you'll be able to [scope the migration](docs/assessment.md) and execute the [group migration workflow](#group-migration-workflow).

The [README notebook](#readme-notebook), which can be found in the installation folder contains further instructions and explanations of the different ucx workflows & dashboards.
Once the migration is scoped, you can start with the [table migration process](#Table-Migration).


More workflows, like notebook code migration are coming in future releases.


UCX also provides a number of command line utilities accessible via `databricks labs ucx`.

For questions, troubleshooting or bug fixes, please see our [troubleshooting guide](docs/troubleshooting.md) or submit [an issue](https://github.com/databrickslabs/ucx/issues).
See [contributing instructions](CONTRIBUTING.md) to help improve this project.

[![build](https://github.com/databrickslabs/ucx/actions/workflows/push.yml/badge.svg)](https://github.com/databrickslabs/ucx/actions/workflows/push.yml) [![codecov](https://codecov.io/github/databrickslabs/ucx/graph/badge.svg?token=p0WKAfW5HQ)](https://codecov.io/github/databrickslabs/ucx)  ![linesofcode](https://aschey.tech/tokei/github/databrickslabs/ucx?category=code)

<!-- TOC -->
* [Databricks Labs UCX](#databricks-labs-ucx)
* [Installation](#installation)
  * [Installation requirements](#installation-requirements)
  * [Authenticate Databricks CLI](#authenticate-databricks-cli)
  * [Install UCX](#install-ucx)
  * [Installation resources](#installation-resources)
  * [Installation folder](#installation-folder)
    * [Readme notebook](#readme-notebook)
    * [Debug notebook](#debug-notebook)
    * [Debug logs](#debug-logs)
    * [Installation configuration](#installation-configuration)
  * [Advanced installation options](#advanced-installation-options)
    * [Force install over existing UCX](#force-install-over-existing-ucx)
    * [Installing UCX on all workspaces within a Databricks account](#installing-ucx-on-all-workspaces-within-a-databricks-account)
    * [Installing UCX with company hosted PYPI mirror](#installing-ucx-with-company-hosted-pypi-mirror)
  * [Upgrading UCX for newer versions](#upgrading-ucx-for-newer-versions)
  * [Uninstall UCX](#uninstall-ucx)
* [Migration process](#migration-process)
  * [Table migration process](#table-migration-process)
    * [Table mapping](#table-mapping)
      * [Step 1 : Create the mapping file](#step-1--create-the-mapping-file)
      * [Step 2: Update the mapping file](#step-2-update-the-mapping-file)
    * [Data access](#data-access)
      * [Step 1 : Map cloud principals to cloud storage locations](#step-1--map-cloud-principals-to-cloud-storage-locations)
      * [Step 2 : Create or modify cloud principals and credentials](#step-2--create-or-modify-cloud-principals-and-credentials)
      * [Step 3: Create the "uber" Principal](#step-3-create-the-uber-principal)
    * [New Unity Catalog resources](#new-unity-catalog-resources)
      * [Step 0: Attach a metastore](#step-0-attach-a-metastore)
      * [Step 1: Create external Locations](#step-1-create-external-locations)
        * [Step 2: Create Catalogs and Schemas](#step-2-create-catalogs-and-schemas)
    * [Migrate Hive metastore data objects](#migrate-hive-metastore-data-objects)
    * [Odds and Ends](#odds-and-ends)
      * [Skip migrating schemas, tables or views](#skip-migrating-schemas-tables-or-views)
      * [Move data objects](#move-data-objects)
      * [Alias data objects](#alias-data-objects)
      * [Revert migrated data objects](#revert-migrated-data-objects)
* [Workflows](#workflows)
  * [Assessment workflow](#assessment-workflow)
  * [Group migration workflow](#group-migration-workflow)
  * [Table migration workflows](#table-migration-workflows)
    * [Migrate tables](#migrate-tables)
    * [Migrate external Hive SerDe tables](#migrate-external-hive-serde-tables)
    * [Migrate external tables CTAS](#migrate-external-tables-ctas)
  * [Post-migration data reconciliation workflow](#post-migration-data-reconciliation-workflow)
    * [[LEGACY] Scan tables in mounts Workflow](#legacy-scan-tables-in-mounts-workflow)
      * [<b>Always run this workflow AFTER the assessment has finished</b>](#balways-run-this-workflow-after-the-assessment-has-finishedb)
    * [[LEGACY] Migrate tables in mounts Workflow](#legacy-migrate-tables-in-mounts-workflow)
  * [[EXPERIMENTAL] Migration Progress Workflow](#experimental-migration-progress-workflow)
* [Dashboards](#dashboards)
* [Linter message codes](#linter-message-codes)
  * [`cannot-autofix-table-reference`](#cannot-autofix-table-reference)
  * [`catalog-api-in-shared-clusters`](#catalog-api-in-shared-clusters)
  * [`changed-result-format-in-uc`](#changed-result-format-in-uc)
  * [`direct-filesystem-access-in-sql-query`](#direct-filesystem-access-in-sql-query)
  * [`direct-filesystem-access`](#direct-filesystem-access)
  * [`dependency-not-found`](#dependency-not-found)
  * [`jvm-access-in-shared-clusters`](#jvm-access-in-shared-clusters)
  * [`legacy-context-in-shared-clusters`](#legacy-context-in-shared-clusters)
  * [`not-supported`](#not-supported)
  * [`notebook-run-cannot-compute-value`](#notebook-run-cannot-compute-value)
  * [`python-udf-in-shared-clusters`](#python-udf-in-shared-clusters)
  * [`rdd-in-shared-clusters`](#rdd-in-shared-clusters)
  * [`spark-logging-in-shared-clusters`](#spark-logging-in-shared-clusters)
  * [`sql-parse-error`](#sql-parse-error)
  * [`sys-path-cannot-compute-value`](#sys-path-cannot-compute-value)
  * [`table-migrated-to-uc`](#table-migrated-to-uc)
  * [`to-json-in-shared-clusters`](#to-json-in-shared-clusters)
  * [`unsupported-magic-line`](#unsupported-magic-line)
* [Utility commands](#utility-commands)
  * [`logs` command](#logs-command)
  * [`ensure-assessment-run` command](#ensure-assessment-run-command)
  * [`update-migration-progress` command](#update-migration-progress-command)
  * [`repair-run` command](#repair-run-command)
  * [`workflows` command](#workflows-command)
  * [`open-remote-config` command](#open-remote-config-command)
  * [`installations` command](#installations-command)
  * [`report-account-compatibility` command](#report-account-compatibility-command)
  * [`export-assessment` command](#export-assessment-command)
* [Metastore related commands](#metastore-related-commands)
  * [`show-all-metastores` command](#show-all-metastores-command)
  * [`assign-metastore` command](#assign-metastore-command)
  * [`create-ucx-catalog` command](#create-ucx-catalog-command)
* [Table migration commands](#table-migration-commands)
  * [`principal-prefix-access` command](#principal-prefix-access-command)
    * [Access for AWS S3 Buckets](#access-for-aws-s3-buckets)
    * [Access for Azure Storage Accounts](#access-for-azure-storage-accounts)
  * [`create-missing-principals` command (AWS Only)](#create-missing-principals-command-aws-only)
  * [`delete-missing-principals` command (AWS Only)](#delete-missing-principals-command-aws-only)
  * [`create-uber-principal` command](#create-uber-principal-command)
  * [`migrate-credentials` command](#migrate-credentials-command)
  * [`validate-external-locations` command](#validate-external-locations-command)
  * [`migrate-locations` command](#migrate-locations-command)
  * [`create-table-mapping` command](#create-table-mapping-command)
  * [`skip` command](#skip-command)
  * [`unskip` command](#unskip-command)
  * [`create-catalogs-schemas` command](#create-catalogs-schemas-command)
  * [`assign-owner-group` command](#assign-owner-group-command)
  * [`migrate-tables` command](#migrate-tables-command)
  * [`revert-migrated-tables` command](#revert-migrated-tables-command)
  * [`move` command](#move-command)
  * [`alias` command](#alias-command)
* [Code migration commands](#code-migration-commands)
  * [`lint-local-code` command](#lint-local-code-command)
  * [`migrate-local-code` command](#migrate-local-code-command)
  * [`migrate-dbsql-dashboards` command](#migrate-dbsql-dashboards-command)
  * [`revert-dbsql-dashboards` command](#revert-dbsql-dashboards-command)
* [Cross-workspace installations](#cross-workspace-installations)
  * [`sync-workspace-info` command](#sync-workspace-info-command)
  * [`manual-workspace-info` command](#manual-workspace-info-command)
  * [`create-account-groups` command](#create-account-groups-command)
  * [`validate-groups-membership` command](#validate-groups-membership-command)
  * [`validate-table-locations` command](#validate-table-locations-command)
  * [`cluster-remap` command](#cluster-remap-command)
  * [`revert-cluster-remap` command](#revert-cluster-remap-command)
  * [`upload` command](#upload-command)
  * [`download` command](#download-command)
  * [`join-collection` command](#join-collection-command)
  * [collection eligible command](#collection-eligible-command)
* [Common Challenges and the Solutions](#common-challenges-and-the-solutions)
    * [Network Connectivity Issues](#network-connectivity-issues)
    * [Insufficient Privileges](#insufficient-privileges)
    * [Version Issues](#version-issues)
    * [Authentication Issues](#authentication-issues)
    * [Multiple Profiles in Databricks CLI](#multiple-profiles-in-databricks-cli)
    * [Workspace has an external Hive Metastore (HMS)](#workspace-has-an-external-hive-metastore-hms)
    * [Verify the Installation](#verify-the-installation)
* [Star History](#star-history)
* [Project Support](#project-support)
<!-- TOC -->

# Installation

UCX installation is covered by this section.

## Installation requirements

UCX has the following installation requirements:
- Databricks CLI v0.213 or later. See [instructions](#authenticate-databricks-cli).
- Python 3.10 or later. See [Windows](https://www.python.org/downloads/windows/) instructions.
- Databricks Premium or Enterprise workspace.
- Network access to your Databricks Workspace used for the [installation process](#install-ucx).
- Network access to the Internet for [pypi.org](https://pypi.org) and [github.com](https://github.com) from machine running the installation.
- Databricks Workspace Administrator privileges for the user, that runs the installation. Running UCX as a Service Principal is not supported.
- Account level Identity Setup. See instructions for [AWS](https://docs.databricks.com/en/administration-guide/users-groups/best-practices.html), [Azure](https://learn.microsoft.com/en-us/azure/databricks/administration-guide/users-groups/best-practices), and [GCP](https://docs.gcp.databricks.com/administration-guide/users-groups/best-practices.html).
- Unity Catalog Metastore Created (per region). See instructions for [AWS](https://docs.databricks.com/en/data-governance/unity-catalog/create-metastore.html), [Azure](https://learn.microsoft.com/en-us/azure/databricks/data-governance/unity-catalog/create-metastore), and [GCP](https://docs.gcp.databricks.com/data-governance/unity-catalog/create-metastore.html).
- If your Databricks Workspace relies on an external Hive Metastore (such as AWS Glue), make sure to read [this guide](docs/external_hms_glue.md).
- A PRO or Serverless SQL Warehouse to render the [report](docs/assessment.md) for the [assessment workflow](#assessment-workflow).

Once you [install UCX](#install-ucx), you can proceed to the [assessment workflow](#assessment-workflow) to ensure
the compatibility of your workspace with Unity Catalog.

[[back to top](#databricks-labs-ucx)]

## Authenticate Databricks CLI

We only support installations and upgrades through [Databricks CLI](https://docs.databricks.com/en/dev-tools/cli/index.html), as UCX requires an installation script run
to make sure all the necessary and correct configurations are in place. Install Databricks CLI on macOS:
![macos_install_databricks](docs/macos_1_databrickslabsmac_installdatabricks.gif)

Install Databricks CLI on Windows:
![windows_install_databricks.png](docs/windows_install_databricks.png)

Once you install Databricks CLI, authenticate your current machine to a Databricks Workspace:

```commandline
databricks auth login --host WORKSPACE_HOST
```

To enable debug logs, simply add `--debug` flag to any command.

[[back to top](#databricks-labs-ucx)]

## Install UCX

Install UCX via Databricks CLI:

```commandline
databricks labs install ucx
```

You'll be prompted to select a [configuration profile](https://docs.databricks.com/en/dev-tools/auth.html#databricks-client-unified-authentication) created by `databricks auth login` command.

Once you install, proceed to the [assessment workflow](#assessment-workflow) to ensure the compatibility of your workspace with UCX.

The `WorkspaceInstaller` class is used to create a new configuration for Unity Catalog migration in a Databricks workspace.
It guides the user through a series of prompts to gather necessary information, such as selecting an inventory database, choosing
a PRO or SERVERLESS SQL warehouse, specifying a log level and number of threads, and setting up an external Hive Metastore if necessary.
Upon the first installation, you're prompted for a workspace local [group migration strategy](docs/group_name_conflict.md).
Based on user input, the class creates a new cluster policy with the specified configuration. The user can review and confirm the configuration,
which is saved to the workspace and can be opened in a web browser.

The [`WorkspaceInstallation`](src/databricks/labs/ucx/install.py) manages the installation and uninstallation of UCX in a workspace. It handles
the configuration and exception management during the installation process. The installation process creates dashboards, databases, and jobs.
It also includes the creation of a database with given configuration and the deployment of workflows with specific settings. The installation
process can handle exceptions and infer errors from job runs and task runs. The workspace installation uploads wheels, creates cluster policies,
and wheel runners to the workspace. It can also handle the creation of job tasks for a given task, such as job dashboard tasks, job notebook tasks,
and job wheel tasks. The class handles the installation of UCX, including configuring the workspace, installing necessary libraries, and verifying
the installation, making it easier for users to migrate their workspaces to UCX.
At the end of the installation, the user will be prompted if the current installation needs to join an existing collection (create new collection if none present).
For large organization with many workspaces, grouping workspaces into collection helps in managing UCX migration at collection level (instead of workspaces level)
User should be an account admin to be able to join a collection.

After this, UCX will be installed locally and a number of assets will be deployed in the selected workspace.
These assets are available under the installation folder, i.e. `/Applications/ucx` is the default installation folder. Please check [here](#advanced-force-install-over-existing-ucx) for more details.

You can also install a specific version by specifying it like `@v0.13.2` - `databricks labs install ucx@v0.13.2`.

![macos_install_ucx](docs/macos_2_databrickslabsmac_installucx.gif)

[[back to top](#databricks-labs-ucx)]

## Installation resources

The following resources are installed by UCX:

| Installed UCX resources                           | Description                                                                                      |
|---------------------------------------------------|--------------------------------------------------------------------------------------------------|
| [Inventory database](./docs/table_persistence.md) | A Hive metastore database/schema in which UCX persist inventory required for the upgrade process |
| [Workflows](#workflows)                           | Workflows to execute UCX                                                                         |
| [Dashboards](#dashboards)                         | Dashboards to visualize UCX outcomes                                                             |
| [Installation folder](#installation-folder)       | A workspace folder containing UCX files in `/Applications/ucx/`.                                 |

## Installation folder

UCX is in installed in the workspace folder `/Applications/ucx/`. This folder contains UCX's code resources, like the
[source code](./src) from this GitHub repository and the [dashboard](#dashboards). Generally, these resources are not
*directly* used by UCX users. Resources that can be of importance to users are detailed in the subsections below.

### Readme notebook

![readme](docs/readme-notebook.png)

Every installation creates a `README` notebook with a detailed description of all deployed workflows and their tasks,
providing quick links to the relevant workflows and dashboards.

[[back to top](#databricks-labs-ucx)]

### Debug notebook

![debug](docs/debug-notebook.png)

Every installation creates a `DEBUG` notebook, that initializes UCX as a library for you to execute interactively.

[[back to top](#databricks-labs-ucx)]

### Debug logs

![debug](docs/debug-logs.png)

The [workflow](#workflows) runs store debug logs in the `logs` folder of the installation folder. The logs are flushed
every minute in a separate file. Debug logs for [the command-line interface](#authenticate-databricks-cli) are shown
by adding the `--debug` flag:

```commandline
databricks --debug labs ucx <command>
```

[[back to top](#databricks-labs-ucx)]

### Installation configuration

In the installation folder, the UCX configuration is kept.

## Advanced installation options

Advanced installation options are detailed below.

### Force install over existing UCX
Using an environment variable `UCX_FORCE_INSTALL` you can force the installation of UCX over an existing installation.
The values for the environment variable are 'global' and 'user'.

Global Install: When UCX is installed at '/Applications/ucx'
User Install: When UCX is installed at '/Users/<user>/.ucx'

If there is an existing global installation of UCX, you can force a user installation of UCX over the existing installation by setting the environment variable `UCX_FORCE_INSTALL` to 'global'.

At this moment there is no global override over a user installation of UCX. As this requires migration and can break existing installations.


| global | user | expected install location | install_folder      | mode                                            |
|--------|------|---------------------------|---------------------|-------------------------------------------------|
| no     | no   | default                   | `/Applications/ucx` | install                                         |
| yes    | no   | default                   | `/Applications/ucx` | upgrade                                         |
| no     | yes  | default                   | `/Users/X/.ucx`     | upgrade (existing installations must not break) |
| yes    | yes  | default                   | `/Users/X/.ucx`     | upgrade                                         |
| yes    | no   | **USER**                  | `/Users/X/.ucx`     | install (show prompt)                           |
| no     | yes  | **GLOBAL**                | ...                 | migrate                                         |


* `UCX_FORCE_INSTALL=user databricks labs install ucx` - will force the installation to be for user only
* `UCX_FORCE_INSTALL=global databricks labs install ucx` - will force the installation to be for root only


[[back to top](#databricks-labs-ucx)]

### Installing UCX on all workspaces within a Databricks account
Setting the environment variable `UCX_FORCE_INSTALL` to 'account' will install UCX on all workspaces within a Databricks account.

* `UCX_FORCE_INSTALL=account databricks labs install ucx`

After the first installation, UCX will prompt the user to confirm whether to install UCX on the remaining workspaces with the same answers. If confirmed, the remaining installations will be completed silently.

This installation mode will automatically select the following options:
* Automatically create and enable HMS lineage init script
* Automatically create a new SQL warehouse for UCX assessment

[[back to top](#databricks-labs-ucx)]

### Installing UCX with company hosted PYPI mirror

Some enterprise block the public PYPI index and host a company controlled PYPI mirror. To install UCX while using a
company hosted PYPI mirror for finding its dependencies, add all UCX dependencies to the company hosted PYPI mirror (see
"dependencies" in [`pyproject.toml`](./pyproject.toml)) and set the environment variable `PIP_INDEX_URL` to the company
hosted PYPI mirror URL while installing UCX:

```commandline
PIP_INDEX_URL="https://url-to-company-hosted-pypi.internal" databricks labs install ucx
```

During installation reply *yes* to the question "Does the given workspace block Internet access"?

[[back to top](#databricks-labs-ucx)]

## Upgrading UCX for newer versions

Verify that UCX is installed

```commandline
databricks labs installed

Name  Description                            Version
ucx   Unity Catalog Migration Toolkit (UCX)  <version>
```

Upgrade UCX via Databricks CLI:

```commandline
databricks labs upgrade ucx
```

The prompts will be similar to [Installation](#install-ucx)

![macos_upgrade_ucx](docs/macos_3_databrickslabsmac_upgradeucx.gif)

[[back to top](#databricks-labs-ucx)]

## Uninstall UCX

Uninstall UCX via Databricks CLI:

```commandline
databricks labs uninstall ucx
```

Databricks CLI will confirm a few options:
- Whether you want to remove all ucx artefacts from the workspace as well. Defaults to no.
- Whether you want to delete the inventory database in `hive_metastore`. Defaults to no.

![macos_uninstall_ucx](docs/macos_4_databrickslabsmac_uninstallucx.gif)

[[back to top](#databricks-labs-ucx)]

# Migration process

On a high level, the steps in migration process are:
1. [assessment](#assessment-workflow)
2. [group migration](#group-migration-workflow)
3. [table migration](#table-migration-process)
4. [data reconciliation](#post-migration-data-reconciliation-workflow)
5. [code migration](#code-migration-commands)

The migration process can be schematic visualized as:

```mermaid
flowchart TD
    subgraph workspace-admin
        assessment --> group-migration
        group-migration --> table-migration
        table-migration --> code-migration
        assessment --> create-table-mapping
        create-table-mapping --> table-migration
        create-table-mapping --> code-migration
        validate-external-locations --> table-migration
        assessment --> validate-table-locations
        validate-table-locations --> table-migration
        table-migration --> revert-migrated-tables
        revert-migrated-tables --> table-migration
    end
    subgraph account-admin
        create-account-groups --> group-migration
        sync-workspace-info --> create-table-mapping
        group-migration --> validate-groups-membership
    end
    subgraph iam-admin
        setup-account-scim --> create-account-groups
        assessment --> create-uber-principal
        create-uber-principal --> table-migration
        assessment --> principal-prefix-access
        principal-prefix-access --> migrate-credentials
        migrate-credentials --> validate-external-locations
        setup-account-scim
    end
```

[[back to top](#databricks-labs-ucx)]

## Table migration process

> You are required to complete the [assessment workflow](#assessment-workflow) before starting the table migration workflow.

This section explains how to migrate Hive metastore data objects to Unity Catalog. The table migration process consists
of more steps than only a [workflow](#table-migration-workflow), these steps are:

1. [Table mapping](#table-mapping) : Create a file that maps Hive metastore data objects to Unity Catalog locations.
2. [Data access](#data-access) : Setup identities to access table data.
3. [New Unity Catalog resources](#new-unity-catalog-resources) : Create **new** Unity Catalog resources that do not
   require touching existing Hive metastore resources.
4. [Migrate Hive metastore data objects](#migrate-hive-metastore-data-objects) : Migrate Hive metastore data objects to UC.

### Table mapping

This section details how to create the table mapping file. This file points existing Hive metastore tables and views to
Unity Catalog locations. When [migrating the tables and views](#migrate-hive-metastore-data-objects), the file is read
to decide where to migrate the tables and views to.

#### Step 1 : Create the mapping file

Create the mapping file in the [UCX installation folder](#installation-resources) by running the
[`create-table-mapping` command](#create-table-mapping-command). By default, the file contains all the Hive metastore
tables and views mapped to a single UC catalog, while maintaining the original schema and table names.

#### Step 2: Update the mapping file

Edit the mapping file from the previous step to:
- Exclude tables and/or views by removing the lines
- Change UC location by editing the destination catalog and/or schema

The mapping file is in CSV format and can be edited using any text editor or Excel. If using Excel, save the file in CSV
format.

Example changes:

**Before editing**

| **workspace_name**   | **catalog_name** | **src_schema** | **dst_schema** | **src_table** | **dst_table** |
|----------------------|--------------|----------------|----------------|---------------|---------------|
| data_engineering_ws  | 123333333   | sales_analysis      | sales_analysis      | ytd_sales        | ytd_sales        |

**After editing**

| **workspace_name**   | **catalog_name** | **src_schema** | **dst_schema** | **src_table** | **dst_table** |
|----------------------|------------------|----------------|----------------|---------------|---------------|
| data_engineering_ws  | data_engineering | sales_analysis      | sales      | ytd_sales        | ytd_analysis        |

### Data access

> Throughout this guide, we refer to IAM roles/instance profiles in AWS & service principals/managed identities in
> as "cloud principals".

This section creates the cloud principals to access data in Unity Catalog and during the table data during migration. To
understand the motivation for this step, read how Databricks accesses cloud locations:
- [AWS - Create External Locations](https://docs.databricks.com/en/connect/unity-catalog/external-locations.html)
- [Azure - Create External Locations](https://learn.microsoft.com/en-us/azure/databricks/connect/unity-catalog/external-locations)

#### Step 1 : Map cloud principals to cloud storage locations

Map the cloud principals to cloud storage by running the
[`principal-prefix-access` command](#principal-prefix-access-command).

#### Step 2 : Create or modify cloud principals and credentials

Manually create the cloud principals to access data from Unity Catalog:

- AWS:
  - [`create-missing-principals` command](#create-missing-principals-command-aws-only) creates new AWS roles for Unity
    Catalog to access data.
  - Or, [Manually create storage credentials](https://docs.databricks.com/en/connect/unity-catalog/storage-credentials.html)
- Azure:
  - [Manually create storage Credentials](https://learn.microsoft.com/en-us/azure/databricks/sql/language-manual/sql-ref-storage-credentials)

Then, run the [`migrate-credentials` command](#migrate-credentials-command) to migrate the cloud principal
credentials to Unity Catalog.

#### Step 3: Create the "uber" Principal

Create the "uber" principal by running the [`create-uber-principal` command](#create-uber-principal-command).
The table migration requires a cloud principal with access to table data stored in cloud storage. These tables are known
as external tables. UCX name this cloud principal the "uber" principal as it has access to **all** in-scope cloud
storage. This principal is only required for the table migration process and not for ongoing UC usage. Once the upgrade
is completed, this principal should be deleted.

### New Unity Catalog resources

This section details the **new** Unity Catalog resources required for migration the data objects. These resources can be
created without touching the existing Hive metastore objecs.

#### Step 0: Attach a metastore

If skipping the [group migration](#group-migration-workflow), then a metastore should be attached to the workspace by
following [these instructions](https://docs.databricks.com/en/data-governance/unity-catalog/create-metastore.html) or running the [`assign-metastore` command](#assign-metastore-command).

#### Step 1: Create external Locations

Create UC external locations by running the [`migration-locations` command](#migrate-locations-command). The command
creates a location for each location found during the [assessment](#assessment-workflow). It uses the credentials
created in the previous steps.

Alternatively, manually create the external locations
- [AWS - Create External Locations](https://docs.databricks.com/en/connect/unity-catalog/external-locations.html)
- [Azure - Create External Locations](https://learn.microsoft.com/en-us/azure/databricks/connect/unity-catalog/external-locations)

##### Step 2: Create Catalogs and Schemas

Create Uniyt Catalog [catalogs](https://learn.microsoft.com/en-us/azure/databricks/catalogs/) and
[schemas](https://learn.microsoft.com/en-us/azure/databricks/schemas/) to organize the destination tables and views in
by running the
[`create-catalogs-schemas` command](#create-catalogs-schemas-command). The command creates the UC catalogs and
schemas based on the [table mapping file](#table-mapping). Additionally, it migrates Hive metastore database
permissions if present.

This step requires considering how to [physically separate data in storage](https://docs.databricks.com/en/data-governance/unity-catalog/best-practices.html#data-is-physically-separated-in-storage)
within UC. As [Databricks recommends storing managed data at the catalog level](https://docs.databricks.com/en/data-governance/unity-catalog/best-practices.html#configure-a-unity-catalog-metastore),
we advise to prepare the external locations for the to-be created catalogs before running the `create-catalogs-schemas`
command. Either, reuse [previously created external locations](#step-23-create-external-locations) or create additional
external locations outside of UCX if data separation restrictions requires that. Note that external locations can be
reused when using subpaths, for example, a folder in a cloud storage
(`abfss://container@storage.dfs.core.windows.net/folder`) can reuse the external location of the cloud storage
(`abfss://container@storage.dfs.core.windows.net/`). (The previous example also holds for other clouds.)

### Migrate Hive metastore data objects

In this step, tables and views are migrated using the following workflows:

```mermaid
flowchart LR
    subgraph CLI
        migrate-tables[migrate-tables]
    end

    subgraph mt_workflow[workflow: migrate-tables]
        dbfs_root_delta_mt_task[migrate_dbfs_root_delta_tables]
        migrate_dbfs_root_non_delta_tables[migrate_dbfs_root_non_delta_tables]
        external_tables_sync_mt_task[migrate_external_tables_sync]
        view_mt_task[migrate_views]
        refresh_migration_status[refresh_migration_status]
        external_tables_sync_mt_task --> view_mt_task
        dbfs_root_delta_mt_task --> view_mt_task
        migrate_dbfs_root_non_delta_tables --> view_mt_task
        view_mt_task --> refresh_migration_status
    end

    subgraph mt_serde_inplace_wf[workflow: migrate-external-hiveserde-tables-in-place-experimental]
        serde_inplace_mt_task[migrate_hive_serde_in_place] --> view_mt_task_inplace[migrate_views]
        view_mt_task_inplace --> refresh_migration_status_hiveserde[refresh_migration_status]
    end

    subgraph mt_ctas_wf[workflow: migrate-external-tables-ctas]
        ctas_mt_task[migrate_other_external_ctas]
        migrate_hive_serde_ctas[migrate_hive_serde_ctas]
        view_mt_ctas_task[migrate_views]
        refresh_migration_status_ctas[refresh_migration_status]
        ctas_mt_task --> view_mt_ctas_task
        migrate_hive_serde_ctas --> view_mt_ctas_task
        view_mt_ctas_task --> refresh_migration_status_ctas
    end

    migrate-tables -- 1st --> mt_workflow
    migrate-tables -- 2nd (optional) table migrated here will be excluded in ctas workflow --> mt_serde_inplace_wf
    migrate-tables -- 3rd --> mt_ctas_wf
```

The [table migration workflows](#table-migration-workflow) can be triggered using the [`migrate-tables` command](#migrate-tables-command) or by
starting the workflows from the workspace UI manually. The table migration workflows are designed to minimize data
copying and to maintain metadata while allowing users to choose preferred strategies where possible. Chose the
strategies for migrating tables using the table below:

| Object Type           | Description                                                                                            | Upgrade Method                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                         |
|-----------------------|--------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| `DBFS_ROOT_DELTA`     | Delta tables persisted on the Databricks file system (dbfs).                                           | Create a copy of the table with the [`DEEP CLONE` command](https://docs.databricks.com/en/sql/language-manual/delta-clone.html).                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                       |
| `DBFS_ROOT_NON_DELTA` | Non-delta tables persisted on the Databricks file system (dbfs).                                       | Create a copy of the table with a [`CREATE TABLE AS SELECT * FROM` command](https://docs.databricks.com/en/sql/language-manual/sql-ref-syntax-ddl-create-table-using.html). The UC table will be a Delta table.                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                        |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                   |
| `MANAGED`             | Managed Hive metastore tables.                                                                         | Depending on the managed table migration strategy chosen during installation:<br/>1. `CLONE`: Create a copy of the table with a [`CREATE TABLE LOCATION '<location>' AS SELECT * FROM` command](https://docs.databricks.com/en/sql/language-manual/sql-ref-syntax-ddl-create-table-using.html).<br/>2. `SYNC_AS_EXTERNAL`, synchronize the table metadata to UC with the [`SYNC` command](https://docs.databricks.com/en/sql/language-manual/sql-ref-syntax-aux-sync.html). **Warning**: If the managed Hive metastore table is dropped, the drop deletes the underlying data affecting the synchronised UC table as well.<br/>3. `CONVERT_TO_EXTERNAL`: First, in-place convert the managed Hive metastore to a non-managed table. Then, synchronize the table metadata to UC with the [`SYNC` command](https://docs.databricks.com/en/sql/language-manual/sql-ref-syntax-aux-sync.html). **Warning**: This strategy has the advantage over `SYNC_AS_EXTERNAL` that dropping the Hive metastore table does not delete the underlying data, however, impact should be carefully validated in existing workloads as the strategy converts the managed Hive metastore to an external table **in-place**. |
| `EXTERNAL_SYNC`       | External tables that the `SYNC` command supports: Delta, Parquet, CSV, JSON, ORC, text or Avro tables. | Synchronize the table metadata to UC with the [`SYNC` command](https://docs.databricks.com/en/sql/language-manual/sql-ref-syntax-aux-sync.html).                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                       |
| `EXTERNAL_NO_SYNC`    | External tables that the `SYNC` command does not support.                                              | Create a copy of the table with a [`CREATE TABLE AS SELECT * FROM` command](https://docs.databricks.com/en/sql/language-manual/sql-ref-syntax-ddl-create-table-using.html). The UC table will be a Delta table.                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                        |
| `EXTERNAL_HIVESERDE`  | External Hive SerDe tables that the `SYNC` command does not support.                                   | Depending on the migration workflow chosen:<br/>1. `migrate-external-tables-ctas` (officially supported) : Create a copy of the table with a [`CREATE TABLE AS SELECT * FROM` command](https://docs.databricks.com/en/sql/language-manual/sql-ref-syntax-ddl-create-table-using.html) The UC table will be a Delta table.<br/>2. `migrate-external-hiveserde-tables-in-place-experimental` (experimental) : Recreate the [Hive SerDe](https://cwiki.apache.org/confluence/display/Hive/SerDe) tables using the serialization and deserialization protocols. **Warning:** Although this strategy is supported, there is a risk that the old files created by Hive SerDe may not be processed correctly by Spark datasource in corner cases.                                                                                                                                                                                                                                                                                                                                                                                                                                                             |
| `VIEW`                | Views                                                                                                  | Recreate [views](https://docs.databricks.com/en/sql/language-manual/sql-ref-syntax-ddl-create-view.html) using their definitions after repointing their dependencies to UC objects. The migration process supports views depending on other views.                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                     |

The workflows may be executed multiple times. Each step is designed as a standalone operation that migrates all
in-scope tables. After migration, each object is marked with an `upgraded_to` property containing the UC identifier to
which the object is migrated. This property signals that the object is out-of-scope for other migration operations and
that the view dependency exists within UC. The created UC objects are marked with an `upgraded_from` property containing
the Hive metastore identifier from which the object was migrated.

Finally, the table migration workflows also migrate Hive metastore permissions to Unity Catalog.

Considerations:
- You may want to run the workflows multiple times to migrate tables in phases.
- If your Delta tables in DBFS root have a large number of files, consider:
  - Setting higher `Min` and `Max workers for auto-scale` when being asked during the UCX installation. More cores in the cluster means more concurrency for calling cloud storage API to copy files when deep cloning the Delta tables.
  - Setting higher `Parallelism for migrating DBFS root Delta tables with deep clone` (default 200) when being asked during the UCX installation. This controls the number of Spark tasks/partitions to be created for deep clone.
- Consider creating an instance pool, and setting its id when prompted during the UCX installation. This instance pool will be specified in the cluster policy used by all UCX workflows job clusters.
- You may also manually edit the job cluster configration per job or per task after the workflows are deployed.

Additional references:
- [Databricks file system (dbfs)](https://docs.databricks.com/en/dbfs/index.html)
- [External tables](https://docs.databricks.com/en/tables/external.html)

### Odds and Ends

The following sections detail how to repair/amend the UC metastore after the upgrade process.

#### Skip migrating schemas, tables or views

```bash
databricks labs ucx skip --schema X [--table Y] [--view Zj]
```

The [`skip` command](#skip-command) marks a schema, table or view as to-be skipped during the migration processes.

####  Move data objects

```bash
databricks labs ucx move --from-catalog A --from-schema B --from-table C --to-catalog D --to-schema E
```

The [`move` command](#move-command) moves the object from the source to the target location. The `upgraded_from`
property are updated to reflect the new location on the source object. Use this command if the data object is migrated
to the wrong destination.

#### Alias data objects

```bash
databricks labs ucx alias --from-catalog A --from-schema B --from-table C --to-catalog D --to-schema E
```

This [`alias` command](#alias-command) creates an alias for the object in the target location by creating a view reading
from the table that needs aliasing. It will create a mirror view to view that is marked as alias.
Use this command if Hive metastore tables point to the same location as UC does not support UC does not support tables
with overlapping data locations.

#### Revert migrated data objects

```bash
databricks labs ucx revert-migrated-tables --schema X --table Y [--delete-managed]
```

The [`revert-migrated-tables` command](#revert-migrated-tables-command) drops the Unity Catalog table or view and reset
the `upgraded_to` property on the source object. Use this command to allow for migrating a table or view again.

[[back to top](#databricks-labs-ucx)]

# Workflows

Part of this application is deployed as [Databricks Workflows](https://docs.databricks.com/en/workflows/index.html).
You can view the status of deployed workflows via the [`workflows` command](#workflows-command).
Failed workflows can be fixed with the [`repair-run` command](#repair-run-command).

[[back to top](#databricks-labs-ucx)]

## Assessment workflow

![ucx_assessment_workflow](docs/ucx_assessment_workflow.png)

The assessment workflow can be triggered using the Databricks UI or via the
[`ensure-assessment-run` command](#ensure-assessment-run-command).

The assessment workflow retrieves - or *crawls* - details of [workspace assets](https://docs.databricks.com/en/workspace/workspace-assets.html)
and [securable objects in the Hive metastore](https://docs.databricks.com/en/data-governance/table-acls/object-privileges.html#securable-objects-in-the-hive-metastore)
relevant for upgrading to UC to assess the compatibility with UC. The `crawl_` tasks retrieve assess and objects. The
`assess_` tasks assess the compatibility with UC. The output of each task is stored in the
[inventory database](#installation-resources) so that it can be used for further analysis and decision-making through
the [assessment report](docs/assessment.md).

1. `crawl_tables`: This task retrieves table definitions from the Hive metastore and persists the definitions in
   the `tables` table. The definitions include information such as:
   - Database/schema name
   - Table name
   - Table type
   - Table location
2. `crawl_udfs`: This task retrieves UDF definitions from the Hive metastore and persists the definitions in
   the `udfs` table.
3. `setup_tacl`: (Optimization) This task starts the `tacl` job cluster in parallel to other tasks.
4. `crawl_grants`: This task retrieves [privileges you can grant on Hive objects](https://docs.databricks.com/en/data-governance/table-acls/object-privileges.html#privileges-you-can-grant-on-hive-metastore-objects)
   and persists the privilege definitions in the `grants` table
   The retrieved permission definitions include information such as:
   - Securable object: schema, table, view, (anonymous) function or any file.
   - Principal: user, service principal or group
   - Action type: grant, revoke or deny
5. `estimate_table_size_for_migration`: This task analyzes the Delta table retrieved by `crawl_tables` to retrieve
   an estimate of their size and persists the table sizes in the `table_size` table. The table size support the decision
   for using the `SYNC` or `CLONE` [table migration strategy](#step-3-upgrade-the-metastore).
6. `crawl_mounts`: This task retrieves mount point definitions and persists the definitions in the `mounts` table.
7. `guess_external_locations`: This task guesses shared mount path prefixes of [external tables](https://docs.databricks.com/en/sql/language-manual/sql-ref-external-tables.html)
   retrieved by `crawl_tables` that use mount points and persists the locations in the `external_locations` table. The
   goal is to identify the to-be created UC [external locations](#step-23-create-external-locations).
8. `assess_jobs`: This task retrieves the job definitions and persists the definitions in the `jobs` table. Job
   definitions may require updating to become UC compatible.
9. `assess_clusters`: This task retrieves the clusters definitions and persists the definitions in the `clusters` table.
    Cluster definitions may require updating to become UC compatible.
10. `assess_pipelines`: This task retrieves the [Delta Live Tables](https://www.databricks.com/product/delta-live-tables)
    (DLT) pipelines definitions and persists the definitions in the `pipelines` table. DLT definitions may require
    updating to become UC compatible.
11. `assess_incompatible_submit_runs` : This task retrieves
    [job runs](https://docs.databricks.com/en/jobs/monitor.html#view-runs-for-a-job), also known as
    [job submit runs](https://docs.databricks.com/api/workspace/jobs/submit), and persists the definitions in the
    `submit_runs` table. Incompatibility with UC is assessed:
    - [Databricks runtime](https://www.databricks.com/glossary/what-is-databricks-runtime) should be version 11.3 or above
    - [Access mode](https://docs.databricks.com/en/compute/configure.html#access-modes) should be set.
12. `crawl_cluster_policies` : This tasks retrieves
    [cluster policies](https://docs.databricks.com/en/admin/clusters/policies.html) and persists the policies in the
    `policies` table. Incompatibility with UC is assessed:
    - [Databricks runtime](https://www.databricks.com/glossary/what-is-databricks-runtime) should be version 11.3 or above
13. `assess_azure_service_principals`: This tasks retrieves Azure service principal
    [authentications information](https://learn.microsoft.com/en-us/azure/databricks/connect/storage/tutorial-azure-storage#connect-adls)
    from Spark configurations to access storage accounts and persists these configuration information
    in `azure_service_principals` table. The Spark configurations from the following places are retrieved:
    - Clusters configurations
    - Cluster policies
    - Job cluster configurations
    - Pipeline configurations
    - Warehouse configuration
14. `assess_global_init_scripts`: This task retrieves
    [global init scripts](https://docs.databricks.com/en/init-scripts/global.html) and persists references to the
    scripts in the `global_init_scripts` table. Again, Azure service principal authentication information might be given
    in those scripts.
15. `workspace_listing` : This tasks lists [workspace files](https://docs.databricks.com/en/workspace/workspace-assets.html#files)
    recursively to compile a collection of directories, notebooks, files, repos and libraries. The task uses multi-threading
    to parallelize the listing process for speeding up execution on big workspaces.
16. `crawl_permissions` : This tasks retrieves workspace-local groups permissions and persists these permissions in the `permissions` table.
17. `crawl_groups`: This tasks retrieves workspace-local groups and persists these permissions in the `groups` table.
18. `assess_dashboards`: This task retrieves the dashboards to analyze their queries for
    [migration problems](#linter-message-codes)
19. `assess_workflows`: This task retrieves the jobs to analyze their notebooks and files for
    [migration problems](#linter-message-codes).

After UCX assessment workflow finished, see the assessment dashboard for findings and recommendations.
See [this guide](docs/assessment.md) for more details.

![report](docs/assessment-report.png)

Proceed to the [group migration workflow](#group-migration-workflow) below or go back to the
[migration process diagram](#migration-process).

The UCX assessment workflow is designed to only run once, re-running will **not** update the existing results. If the
inventory and findings for a workspace need to be updated then first reinstall UCX by [uninstalling](#uninstall-ucx)
and [installing](#install-ucx) it again.

[[back to top](#databricks-labs-ucx)]

## Group migration workflow

> You are required to complete the [assessment workflow](#assessment-workflow) before starting the group migration workflow.

**The group migration workflow does NOT CREATE account groups.** In contrast to account groups, [the (legacy)
workspace-local groups cannot be assigned to additional workspaces or granted access to data in a Unity Catalog
metastore](https://docs.databricks.com/en/admin/users-groups/groups.html#difference-between-account-groups-and-workspace-local-groups).
A Databricks admin [assigns account groups to workspaces](https://docs.databricks.com/en/admin/users-groups/index.html#assign-users-to-workspaces)
using [identity federation](https://docs.databricks.com/en/admin/users-groups/index.html#enable-identity-federation)
to manage groups from a single place: your Databricks account. We expect UCX users to create account groups
centrally while most other Databricks resources that UCX touches are scoped to a single workspace.
If you do not have account groups matching the workspace in which UCX is installed, please
run [`create-account-groups` command](#create-account-groups-command) before running the group migration workflow.

The group migration workflow is designed to migrate workspace-local groups to account-level groups. It verifies if
the necessary groups are available to the workspace with the correct permissions, and removes unnecessary groups and
permissions. The group migration workflow depends on the output of the assessment workflow, thus, should only be
executed after a successful run of the assessment workflow. The group migration workflow may be executed multiple times.

1. `verify_metastore_attached`: Verifies if a metastore is attached. Account level groups are only available when
    a metastore is attached. [See `assign-metastore` command.](#assign-metastore-command)
2. `rename_workspace_local_groups`: This task renames workspace-local groups by adding a `ucx-renamed-` prefix. This
   step is taken to avoid conflicts with account groups that may have the same name as workspace-local groups.
3. `reflect_account_groups_on_workspace`: This task adds matching account groups to this workspace. The matching account
   groups must exist for this step to be successful. This step is necessary to ensure that the account groups
   are available in the workspace for assigning permissions.
4. `apply_permissions_to_account_groups`: This task assigns the full set of permissions of the original group to the
   account-level one. This step is necessary to ensure that the account-level groups have the necessary permissions to
   manage the entities in the workspace. It covers workspace-local permissions for all entities including:
   - Legacy Table ACLs
   - Entitlements
   - AWS instance profiles
   - Clusters
   - Cluster policies
   - Instance Pools
   - Databricks SQL warehouses
   - Delta Live Tables
   - Jobs
   - MLflow experiments
   - MLflow registry
   - SQL Dashboards & Queries
   - SQL Alerts
   - Token and Password usage permissions
   - Secret Scopes
   - Notebooks
   - Directories
   - Repos
   - Files
5. `validate_groups_permissions`: This task validates that all the crawled permissions are applied correctly to the
   destination groups.

After successfully running the group migration workflow:
1. Use [`validate-groups-membership` command](#validate-groups-membership-command) for extra confidence the newly created
   account level groups are considered to be valid.
2. Run the [`remove-workspace-local-backup-grups`](#validate-groups-membership-command) to remove workspace-level backup
   groups, along with their permissions. This should only be executed after confirming that the workspace-local
   migration worked successfully for all the groups involved. This step is necessary to clean up the workspace and
   remove any unnecessary groups and permissions.
3. Proceed to the [table migration process](#Table-Migration).

For additional information see:
- The [detailed design](docs/local-group-migration.md) of thie group migration workflow.
- The [migration process diagram](#migration-process) showing the group migration workflow in context of the whole
  migration process.

[[back to top](#databricks-labs-ucx)]

## Table migration workflows

This section lists the workflows that migrate tables and views. See [this section](#migrate-hive-metastore-data-objects)
for deciding which workflow to run and additional context for migrating tables.

### Migrate tables

The general table migration workflow `migrate-tables` migrates all tables and views using default strategies.

1. `migrate_external_tables_sync` : This step migrates the external tables that are supported by `SYNC` command.
2. `migrate_dbfs_root_delta_tables` : This step migrates delta tables stored in DBFS root using the `DEEP CLONE`
   command.
3. `migrate_dbfs_root_non_delta_tables` : This step migrates non-delta tables stored in DBFS root using the `CREATE
   TABLE AS SELECT * FROM` command.
4. `migrate_views` : This step migrates views using the `CREATE VIEW` command.
5. `update_migration_status` : Refresh the migration status of all data objects.

### Migrate external Hive SerDe tables

The experimental table migration workflow `migrate-external-hiveserde-tables-in-place-experimental` migrates tables that
support the `SYNC AS EXTERNAL` command.

1. `migrate_hive_serde_in_place` : This step migrates the Hive SerDe tables that are supported by `SYNC AS EXTERNAL`
   command.
2. `migrate_views` : This step migrates views using the `CREATE VIEW` command.
3. `update_migration_status` : Refresh the migration status of all data objects.

### Migrate external tables CTAS

The table migration workflow `migrate-external-tables-ctas` migrates tables with the `CREATE TABLE AS SELECT * FROM`
command.

1. `migrate_other_external_ctas` This step migrates the Hive Serde tables using the `CREATE TABLE AS SELECT * FROM`
   command.
2. `migrate_hive_serde_ctas` : This step migrates the Hive Serde tables using the `CREATE TABLE AS SELECT * FROM`
   command.
4. `migrate_views` : This step migrates views using the `CREATE VIEW` command.
4. `update_migration_status` : Refresh the migration status of all data objects.

## Post-migration data reconciliation workflow

The `migrate-data-reconciliation` workflow validates the integrity of the migrated tables and persists its results in
the `recon_results` table. The workflow compares the following between the migrated Hive metastore and its UC
counterpart table:
- `Schema` : See this result in the `schema_matches` column.
- `Column by column` : See this result in the `column_comparison` column.
- `Row counts` : If the row count is within the reconciliation threshold (defaults to 5%), the `data_matches` column is
  set to `true`, otherwise it is set to `false`.
- `Rows` : If the `compare_rows` flag is set to `true`, rows are compared using a hash comparison. Number of missing
  rows are stored in the `source_missing_count` and `target_missing_count` column, respectively.

The output is processed and displayed in the migration dashboard using the in `reconciliation_results` view.

![reconciliation results](docs/recon_results.png)

### [LEGACY] Scan tables in mounts Workflow
#### <b>Always run this workflow AFTER the assessment has finished</b>
- This experimental workflow attempts to find all Tables inside mount points that are present on your workspace.
- If you do not run this workflow, then `migrate-tables-in-mounts-experimental` won't do anything.
- It writes all results to `hive_metastore.<inventory_database>.tables`, you can query those tables found by filtering on database values that starts with `mounted_`
- This command is incremental, meaning that each time you run it, it will overwrite the previous tables in mounts found.
- Current format are supported:
  - DELTA - PARQUET - CSV - JSON
  - Also detects partitioned DELTA and PARQUET
- You can configure these workflows with the following options available on conf.yml:
  - include_mounts : A list of mount points to scans, by default the workflow scans for all mount points
  - exclude_paths_in_mount : A list of paths to exclude in all mount points
  - include_paths_in_mount : A list of paths to include in all mount points

### [LEGACY] Migrate tables in mounts Workflow
- An experimental workflow that migrates tables in mount points using a `CREATE TABLE` command, optinally sets a default tables owner if provided in `default_table_owner` conf parameter.
- You must do the following in order to make this work:
  - run the Assessment [workflow](#assessment-workflow)
  - run the scan tables in mounts [workflow](#EXPERIMENTAL-scan-tables-in-mounts-workflow)
  - run the [`create-table-mapping` command](#create-table-mapping-command)
    - or manually create a `mapping.csv` file in Workspace -> Applications -> ucx


[[back to top](#databricks-labs-ucx)]

## [EXPERIMENTAL] Migration Progress Workflow

The manually triggered `migration-progress-experimental` workflow populates the tables visualized in the
[migration progress dashboard](#dashboards) by updating a **subset** of the [inventory tables](#assessment-workflow)
to [track Unity Catalog compatability](docs/migration-progress.md) of Hive and workspace objects that need to be migrated.

The following pre-requisites need to be fulfilled before running the workflow:
- [UC metastore attached to workspace](../README.md#assign-metastore-command)
- [UCX catalog exists](../README.md#create-ucx-catalog-command)
- [Assessment job ran successfully](../README.md#ensure-assessment-run-command)

[[back to top](#databricks-labs-ucx)]

# Dashboards

The dashboards visualize UCX's outcomes. Each dashboard has text widgets detailing the contents in more detail. Below an
overview with a short description is given.

| Name                                                                                                            | Description                                        |
|-----------------------------------------------------------------------------------------------------------------|----------------------------------------------------|
| [Assessment \[Main\]](./src/databricks/labs/ucx/queries/assessment/main/00_0__assessment_overview.md)           | Assessment overview                                |
| [Assessment \[Estimates\]](./src/databricks/labs/ucx/queries/assessment/estimates/00_0_metastore_assignment.md) | Migration effort estimates based on the assessment |
| [Assessment \[Interactive\]](./src/databricks/labs/ucx/queries/assessment/interactive/00_0_interactive.md)      | Assessment outcomes on interactive clusters        |
| [Assessment \[Azure\]](./src/databricks/labs/ucx/queries/assessment/azure/00_0_azure_service_principals.md)     | Assessment outcomes specific to Azure              |
| [Migration \[Main\]](./src/databricks/labs/ucx/queries/migration/main/00_0_migration_overview.md)               | Migration overview                                 |
| [Migration \[Groups\]](./src/databricks/labs/ucx/queries/migration/groups/00_0_migration_overview.md)           | Group migration outcomes                           |
| [Progress \[Main\]](./src/databricks/labs/ucx/queries/progress/main/00_0_migration_progress.md)                 | [Migration progress](./docs/migration-progress.md) |

[[back to top](#databricks-labs-ucx)]

# Linter message codes

![code compatibility problems](docs/code_compatibility_problems.png)

Here's the detailed explanation of the linter message codes:

## `cannot-autofix-table-reference`

This indicates that the linter has found a table reference that cannot be automatically fixed. The user must manually
update the table reference to point to the correct table in Unity Catalog. This mostly occurs, when table name is
computed dynamically, and it's too complex for our static code analysis to detect it. We detect this problem anywhere
where table name could be used: `spark.sql`, `spark.catalog.*`, `spark.table`, `df.write.*` and many more. Code examples
that trigger this problem:

```python
spark.table(f"foo_{some_table_name}")
# ..
df = spark.range(10)
df.write.saveAsTable(f"foo_{some_table_name}")
# .. or even
df.write.insertInto(f"foo_{some_table_name}")
```

Here the `some_table_name` variable is not defined anywhere in the visible scope. Though, the analyser would
successfully detect table name if it is defined:

```python
some_table_name = 'bar'
spark.table(f"foo_{some_table_name}")
```

We even detect string constants when coming either from `dbutils.widgets.get` (via job named parameters) or through
loop variables. If `old.things` table is migrated to `brand.new.stuff` in Unity Catalog, the following code will
trigger two messages: [`table-migrated-to-uc`](#table-migrated-to-uc) for the first query, as the contents are clearly
analysable, and `cannot-autofix-table-reference` for the second query.

```python
# ucx[table-migrated-to-uc:+4:4:+4:20] Table old.things is migrated to brand.new.stuff in Unity Catalog
# ucx[cannot-autofix-table-reference:+3:4:+3:20] Can't migrate table_name argument in 'spark.sql(query)' because its value cannot be computed
table_name = f"table_{index}"
for query in ["SELECT * FROM old.things", f"SELECT * FROM {table_name}"]:
    spark.sql(query).collect()
```

[[back to top](#databricks-labs-ucx)]

## `catalog-api-in-shared-clusters`

`spark.catalog.*` functions require Databricks Runtime 14.3 LTS or above on Unity Catalog clusters in Shared access
mode, so of your code has `spark.catalog.tableExists("table")` or `spark.catalog.listDatabases()`, you need to ensure
that your cluster is running the correct runtime version and data security mode.

[[back to top](#databricks-labs-ucx)]

## `changed-result-format-in-uc`

Calls to these functions would return a list of `<catalog>.<database>.<table>` instead of `<database>.<table>`. So if
you have code like this:

```python
for table in spark.catalog.listTables():
    do_stuff_with_table(table)
```

you need to make sure that `do_stuff_with_table` can handle the new format.

[[back to top](#databricks-labs-ucx)]

## `direct-filesystem-access-in-sql-query`

Direct filesystem access is deprecated in Unity Catalog.
DBFS is no longer supported, so if you have code like this:

```python
df = spark.sql("SELECT * FROM parquet.`/mnt/foo/path/to/parquet.file`")
```

you need to change it to use UC tables.

[[back to top](#databricks-labs-ucx)]

## `direct-filesystem-access`

Direct filesystem access is deprecated in Unity Catalog.
DBFS is no longer supported, so if you have code like this:

```python
display(spark.read.csv('/mnt/things/data.csv'))
```

or this:

```python
display(spark.read.csv('s3://bucket/folder/data.csv'))
```

You need to change it to use UC tables or UC volumes.

[[back to top](#databricks-labs-ucx)]

## `dependency-not-found`

This message indicates that the linter has found a dependency, like Python source file or a notebook, that is not
available in the workspace. The user must ensure that the dependency is available in the workspace. This usually
means an error in the user code.

[[back to top](#databricks-labs-ucx)]

## `jvm-access-in-shared-clusters`

You cannot access Spark Driver JVM on Unity Catalog clusters in Shared Access mode. If you have code like this:

```python
spark._jspark._jvm.com.my.custom.Name()
```

or like this:

```python
log4jLogger = sc._jvm.org.apache.log4j
LOGGER = log4jLogger.LogManager.getLogger(__name__)
```

you need to change it to use Python equivalents.

[[back to top](#databricks-labs-ucx)]

## `legacy-context-in-shared-clusters`

SparkContext (`sc`) is not supported on Unity Catalog clusters in Shared access mode. Rewrite it using SparkSession
(`spark`). Example code that triggers this message:

```python
df = spark.createDataFrame(sc.emptyRDD(), schema)
```

or this:

```python
sc.parallelize([1, 2, 3])
```

[[back to top](#databricks-labs-ucx)]

## `not-supported`

Installing eggs is no longer supported on Databricks 14.0 or higher.

[[back to top](#databricks-labs-ucx)]

## `notebook-run-cannot-compute-value`

Path for `dbutils.notebook.run` cannot be computed and requires adjusting the notebook path.
It is not clear for automated code analysis where the notebook is located, so you need to simplify the code like:

```python
b = some_function()
dbutils.notebook.run(b)
```

to something like this:

```python
a = "./leaf1.py"
dbutils.notebook.run(a)
```

[[back to top](#databricks-labs-ucx)]

## `python-udf-in-shared-clusters`

`applyInPandas` requires DBR 14.3 LTS or above on Unity Catalog clusters in Shared access mode. Example:

```python
df.groupby("id").applyInPandas(subtract_mean, schema="id long, v double").show()
```

Arrow UDFs require DBR 14.3 LTS or above on Unity Catalog clusters in Shared access mode.

```python
@udf(returnType='int', useArrow=True)
def arrow_slen(s):
    return len(s)
```

It is not possible to register Java UDF from Python code on Unity Catalog clusters in Shared access mode. Use a
`%scala` cell to register the Scala UDF using `spark.udf.register`. Example code that triggers this message:

```python
spark.udf.registerJavaFunction("func", "org.example.func", IntegerType())
```

[[back to top](#databricks-labs-ucx)]

## `rdd-in-shared-clusters`

RDD APIs are not supported on Unity Catalog clusters in Shared access mode. Use mapInArrow() or Pandas UDFs instead.

```python
df.rdd.mapPartitions(myUdf)
```

[[back to top](#databricks-labs-ucx)]

## `spark-logging-in-shared-clusters`

Cannot set Spark log level directly from code on Unity Catalog clusters in Shared access mode. Remove the call and set
the cluster spark conf `spark.log.level` instead:

```python
sc.setLogLevel("INFO")
setLogLevel("WARN")
```

Another example could be:

```python
log4jLogger = sc._jvm.org.apache.log4j
LOGGER = log4jLogger.LogManager.getLogger(__name__)
```

or

```python
sc._jvm.org.apache.log4j.LogManager.getLogger(__name__).info("test")
```

[[back to top](#databricks-labs-ucx)]

## `sql-parse-error`

This is a generic message indicating that the SQL query could not be parsed. The user must manually check the SQL query.

[[back to top](#databricks-labs-ucx)]

## `sys-path-cannot-compute-value`

Path for `sys.path.append` cannot be computed and requires adjusting the path. It is not clear for automated code
analysis where the path is located.

[[back to top](#databricks-labs-ucx)]

## `table-migrated-to-uc`

This message indicates that the linter has found a table that has been migrated to Unity Catalog. The user must ensure
that the table is available in Unity Catalog.

[[back to top](#databricks-labs-ucx)]

## `to-json-in-shared-clusters`

`toJson()` is not available on Unity Catalog clusters in Shared access mode. Use `toSafeJson()` on DBR 13.3 LTS or
above to get a subset of command context information. Example code that triggers this message:

```python
dbutils.notebook.entry_point.getDbutils().notebook().getContext().toSafeJson()
```

[[back to top](#databricks-labs-ucx)]

## `unsupported-magic-line`

This message indicates the code that could not be analysed by UCX. User must check the code manually.

[[back to top](#databricks-labs-ucx)]

# Utility commands

## `logs` command

```text
$ databricks labs ucx logs [--workflow WORKFLOW_NAME] [--debug]
```

This command displays the logs of the last run of the specified workflow. If no workflow is specified, it displays
the logs of the workflow that was run the last. This command is useful for developers and administrators who want to
check the logs of the last run of a workflow and ensure that it was executed as expected. It can also be used for
debugging purposes when a workflow is not behaving as expected. By default, only `INFO`, `WARNING`, and `ERROR` logs
are displayed. To display `DEBUG` logs, use the `--debug` flag.

[[back to top](#databricks-labs-ucx)]

## `ensure-assessment-run` command

```commandline
databricks labs ucx ensure-assessment-run
```

This command ensures that the [assessment workflow](#assessment-workflow) was run on a workspace.
This command will block until job finishes.
Failed workflows can be fixed with the [`repair-run` command](#repair-run-command). Workflows and their status can be
listed with the [`workflows` command](#workflows-command).

[[back to top](#databricks-labs-ucx)]

## `update-migration-progress` command

```commandline
databricks labs ucx update-migration-progress
```

This command runs the [(experimental) migration progress workflow](#experimental-migration-progress-workflow) to update
the migration status of workspace resources that need to be migrated. It does this by triggering
the `migration-progress-experimental` workflow to run on a workspace and waiting for
it to complete.

Workflows and their status can be listed with the [`workflows` command](#workflows-commandr), while failed workflows can
be fixed with the [`repair-run` command](#repair-run-command).

[[back to top](#databricks-labs-ucx)]

## `repair-run` command

```commandline
databricks labs ucx repair-run --step WORKFLOW_NAME
```

This command repairs a failed [UCX Workflow](#workflows). This command is useful for developers and administrators who
want to repair a failed job. It can also be used to debug issues related to job failures. This operation can also be
done via [user interface](https://docs.databricks.com/en/workflows/jobs/repair-job-failures.html). Workflows and their
status can be listed with the [`workflows` command](#workflows-command).

[[back to top](#databricks-labs-ucx)]

## `workflows` command

See the [migration process diagram](#migration-process) to understand the role of each workflow in the migration process.

```text
$ databricks labs ucx workflows
Step                                  State    Started
assessment                            RUNNING  1 hour 2 minutes ago
099-destroy-schema                    UNKNOWN  <never run>
migrate-groups                        UNKNOWN  <never run>
remove-workspace-local-backup-groups  UNKNOWN  <never run>
validate-groups-permissions           UNKNOWN  <never run>
```

This command displays the [deployed workflows](#workflows) and their state in the current workspace. It fetches the latest
job status from the workspace and prints it in a table format. This command is useful for developers and administrators
who want to check the status of UCX workflows and ensure that they have been executed as expected. It can also be used
for debugging purposes when a workflow is not behaving as expected. Failed workflows can be fixed with
the [`repair-run` command](#repair-run-command).

[[back to top](#databricks-labs-ucx)]

## `open-remote-config` command

```commandline
databricks labs ucx open-remote-config
```

This command opens the remote configuration file in the default web browser. It generates a link to the configuration file
and opens it using the `webbrowser.open()` method. This command is useful for developers and administrators who want to view or
edit the remote configuration file without having to manually navigate to it in the workspace. It can also be used to quickly
access the configuration file from the command line. Here's the description of configuration properties:

  * `inventory_database`: A string representing the name of the inventory database.
  * `workspace_group_regex`: An optional string representing the regular expression to match workspace group names.
  * `workspace_group_replace`: An optional string to replace the matched group names with.
  * `account_group_regex`: An optional string representing the regular expression to match account group names.
  * `group_match_by_external_id`: A boolean value indicating whether to match groups by their external IDs.
  * `include_group_names`: An optional list of strings representing the names of groups to include for migration.
  * `renamed_group_prefix`: An optional string representing the prefix to add to renamed group names.
  * `instance_pool_id`: An optional string representing the ID of the instance pool.
  * `warehouse_id`: An optional string representing the ID of the warehouse.
  * `connect`: An optional `Config` object representing the configuration for connecting to the warehouse.
  * `num_threads`: An optional integer representing the number of threads to use for migration.
  * `database_to_catalog_mapping`: An optional dictionary mapping source database names to target catalog names.
  * `default_catalog`: An optional string representing the default catalog name.
  * `skip_tacl_migration`: Optional flag, allow skipping TACL migration when migrating tables or creating catalogs and schemas.
  * `default_owner_group`: Assigns this group to all migrated objects (catalogs, databases, tables, views, etc.). The group has to be an account group and the user running the migration has to be a member of this group.
  * `log_level`: An optional string representing the log level.
  * `workspace_start_path`: A string representing the starting path for notebooks and directories crawler in the workspace.
  * `instance_profile`: An optional string representing the name of the instance profile.
  * `spark_conf`: An optional dictionary of Spark configuration properties.
  * `override_clusters`: An optional dictionary mapping job cluster names to existing cluster IDs.
  * `policy_id`: An optional string representing the ID of the cluster policy.
  * `include_databases`: An optional list of strings representing the names of databases to include for migration.

[[back to top](#databricks-labs-ucx)]

## `installations` command

```text
$ databricks labs ucx installations
...
13:49:16  INFO [d.labs.ucx] Fetching installations...
13:49:17  INFO [d.l.blueprint.parallel][finding_ucx_installations_5] finding ucx installations 10/88, rps: 22.838/sec
13:49:17  INFO [d.l.blueprint.parallel][finding_ucx_installations_9] finding ucx installations 20/88, rps: 35.002/sec
13:49:17  INFO [d.l.blueprint.parallel][finding_ucx_installations_2] finding ucx installations 30/88, rps: 51.556/sec
13:49:18  INFO [d.l.blueprint.parallel][finding_ucx_installations_9] finding ucx installations 40/88, rps: 56.272/sec
13:49:18  INFO [d.l.blueprint.parallel][finding_ucx_installations_19] finding ucx installations 50/88, rps: 67.382/sec
...
Path                                      Database  Warehouse
/Users/serge.smertin@databricks.com/.ucx  ucx       675eaf1ff976aa98
```

This command displays the [installations](#installation) by different users on the same workspace. It fetches all
the installations where the `ucx` package is installed and prints their details in JSON format. This command is useful
for administrators who want to see which users have installed `ucx` and where. It can also be used to debug issues
related to multiple installations of `ucx` on the same workspace.

[[back to top](#databricks-labs-ucx)]


## `report-account-compatibility` command

```text
databricks labs ucx report-account-compatibility --profile labs-azure-account
12:56:09  INFO [databricks.sdk] Using Azure CLI authentication with AAD tokens
12:56:09  INFO [d.l.u.account.aggregate] Generating readiness report
12:56:10  INFO [databricks.sdk] Using Azure CLI authentication with AAD tokens
12:56:10  INFO [databricks.sdk] Using Azure CLI authentication with AAD tokens
12:56:15  INFO [databricks.sdk] Using Azure CLI authentication with AAD tokens
12:56:15  INFO [d.l.u.account.aggregate] Querying Schema ucx
12:56:21  WARN [d.l.u.account.aggregate] Workspace 4045495039142306 does not have UCX installed
12:56:21  INFO [d.l.u.account.aggregate] UC compatibility: 30.303030303030297% (69/99)
12:56:21  INFO [d.l.u.account.aggregate] cluster type not supported : LEGACY_TABLE_ACL: 22 objects
12:56:21  INFO [d.l.u.account.aggregate] cluster type not supported : LEGACY_SINGLE_USER: 24 objects
12:56:21  INFO [d.l.u.account.aggregate] unsupported config: spark.hadoop.javax.jdo.option.ConnectionURL: 10 objects
12:56:21  INFO [d.l.u.account.aggregate] Uses azure service principal credentials config in cluster.: 1 objects
12:56:21  INFO [d.l.u.account.aggregate] No isolation shared clusters not supported in UC: 1 objects
12:56:21  INFO [d.l.u.account.aggregate] Data is in DBFS Root: 23 objects
12:56:21  INFO [d.l.u.account.aggregate] Non-DELTA format: UNKNOWN: 5 objects
```

[[back to top](#databricks-labs-ucx)]
## `export-assessment` command

```commandline
databricks labs ucx export-assessment
```
The export-assessment command is used to export UCX assessment results to a specified location. When you run this command, you will be prompted to provide details on the destination path and the type of report you wish to generate. If you do not specify these details, the command will default to exporting the main results to the current directory. The exported file will be named based on the selection made in the format. Eg: export_{query_choice}_results.zip
- **Choose a path to save the UCX Assessment results:**
    - **Description:** Specify the path where the results should be saved. If not provided, results will be saved in the current directory.

- **Choose which assessment results to export:**
    - **Description:** Select the type of results to export. Options include:
        - `azure`
        - `estimates`
        - `interactive`
        - `main`
    - **Default:** `main`

[[back to top](#databricks-labs-ucx)]

# Metastore related commands

These commands are used to assign a Unity Catalog metastore to a workspace. The metastore assignment is a pre-requisite
for any further migration steps.

[[back to top](#databricks-labs-ucx)]

## `show-all-metastores` command

```text
databricks labs ucx show-all-metastores [--workspace-id <workspace-id>]
```

This command lists all the metastores available to be assigned to a workspace. If no workspace is specified, it lists
all the metastores available in the account. This command is useful when there are multiple metastores available within
a region, and you want to see which ones are available for assignment.

[[back to top](#databricks-labs-ucx)]

## `assign-metastore` command

```text
databricks labs ucx assign-metastore --workspace-id <workspace-id> [--metastore-id <metastore-id>]
```

This command assigns a metastore to a workspace with `--workspace-id`. If there is only a single metastore in the
workspace region, the command automatically assigns that metastore to the workspace. If there are multiple metastores
available, the command prompts for specification of the metastore (id) you want to assign to the workspace.

[[back to top](#databricks-labs-ucx)]

## `create-ucx-catalog` command

```commandline
databricks labs ucx create-ucx-catalog
16:12:59  INFO [d.l.u.hive_metastore.catalog_schema] Validating UC catalog: ucx
Please provide storage location url for catalog: ucx (default: metastore): ...
16:13:01  INFO [d.l.u.hive_metastore.catalog_schema] Creating UC catalog: ucx
```

Create and setup UCX artifact catalog. Amongst other things, the artifacts are used for tracking the migration progress
across workspaces.

# Table migration commands

These commands are vital part of [table migration process](#Table-Migration) process and require
the [assessment workflow](#assessment-workflow) and
[group migration workflow](#group-migration-workflow) to be completed.
See the [migration process diagram](#migration-process) to understand the role of the table migration commands in
the migration process.

The first step is to run the [`principal-prefix-access` command](#principal-prefix-access-command) to identify all
the storage accounts used by tables in the workspace and their permissions on each storage account.

If you don't have any storage credentials and external locations configured,  you'll need to run
the [`migrate-credentials` command](#migrate-credentials-command) to migrate the service principals
and [`migrate-locations` command](#migrate-locations-command) to create the external locations.
If some of the external locations already exist, you should run
the [`validate-external-locations` command](#validate-external-locations-command).
You'll need to create the [uber principal](#create-uber-principal-command) with the _**access to all storage**_ used to tables in
the workspace, so that you can migrate all the tables. If you already have the principal, you can skip this step.

Ask your Databricks Account admin to run the [`sync-workspace-info` command](#sync-workspace-info-command) to sync the
workspace information with the UCX installations. Once the workspace information is synced, you can run the
[`create-table-mapping` command](#create-table-mapping-command) to align your tables with the Unity Catalog,
[create catalogs and schemas](#create-catalogs-schemas-command) and start the migration using [`migrate-tables` command](#migrate-tables-command). During multiple runs of
the table migration workflow, you can use the [`revert-migrated-tables` command](#revert-migrated-tables-command) to
revert the tables that were migrated in the previous run. You can also skip the tables that you don't want to migrate
using the [`skip` command](#skip-command).

Once you're done with the table migration, proceed to the [code migration](#code-migration-commands).

[[back to top](#databricks-labs-ucx)]

## `principal-prefix-access` command

```text
databricks labs ucx principal-prefix-access [--subscription-ids <Azure Subscription ID>] [--aws-profile <AWS CLI profile>]
```

This command depends on results from the [assessment workflow](#assessment-workflow) and requires [AWS CLI](#access-for-aws-s3-buckets)
or [Azure CLI](#access-for-azure-storage-accounts) to be installed and authenticated for the given machine. This command
identifies all the storage accounts used by tables in the workspace and their permissions on each storage account.
Once you're done running this command, proceed to the [`migrate-credentials` command](#migrate-credentials-command).

The "prefix" refers to the start - i.e. prefix - of table locations that point to the cloud storage location.

[[back to top](#databricks-labs-ucx)]

### Access for AWS S3 Buckets

```commandline
databricks labs ucx principal-prefix-access --aws-profile test-profile
```

Use to identify all instance profiles in the workspace, and map their access to S3 buckets.
Also captures the IAM roles which has UC arn listed, and map their access to S3 buckets
This requires `aws` CLI to be installed and configured.

For AWS this command produces a file named `aws_instance_profile_info.csv`.
It has the following format:

| **role_arn**                                         | **resource_type** | **privilege** | **resource_path**     |
|------------------------------------------------------|-------------------|---------------|-----------------------|
| arn:aws:iam::1234:instance-profile/instance-profile1 | s3                | WRITE_FILES   | s3://s3_bucket1/path1 |


Once done, proceed to the [`migrate-credentials` command](#migrate-credentials-command).

[[back to top](#databricks-labs-ucx)]

### Access for Azure Storage Accounts

```commandline
databricks labs ucx principal-prefix-access --subscription-ids test-subscription-id
```

Use to identify all storage account used by tables, identify the relevant Azure service principals and their permissions
on each storage account. The command is used to identify Azure Service Principals, which have `Storage Blob Data Contributor`,
`Storage Blob Data Reader`, `Storage Blob Data Owner` roles, or custom read/write roles on ADLS Gen2 locations that are being
used in Databricks. This requires Azure CLI to be installed and configured via `az login`. It outputs azure_storage_account_info.csv
which will be later used by migrate-credentials command to create UC storage credentials.
Note: This cmd only lists azure storage account gen2, storage format wasb:// or adl:// are not supported in UC and those storage info
will be skipped.

Once done, proceed to the [`migrate-credentials` command](#migrate-credentials-command).

[[back to top](#databricks-labs-ucx)]

## `create-missing-principals` command (AWS Only)
```bash
databricks labs ucx create-missing-principals --aws-profile <aws_profile> --single-role <single_role>
```
This command identifies all the S3 locations that are missing a UC compatible role and creates them.
It takes single-role optional parameter.
If set to True, it will create a single role for all the S3 locations.
Otherwise, it will create a role for each S3 location.

Two optional parameter are available for this command:
`--role-name` - This parameter is used to set the prefix for the role name. The default value is `UCX-ROLE`.
`--role-policy` - This parameter is used to set the prefix for the role policy name. The default value is `UCX-POLICY`.

[[back to top](#databricks-labs-ucx)]

## `delete-missing-principals` command (AWS Only)
```bash
databricks labs ucx delete-missing-principals --aws-profile <aws_profile>
```
This command helps to delete the IAM role created by UCX. It lists all the IAM Roles generated by the principal-prefix-access
command and allows user to select multiple roles to delete. It also checks if selected roles are mapped to any storage credentials
and asks for confirmation from user. Once confirmed, it deletes the role and its associated inline policy.

[[back to top](#databricks-labs-ucx)]

## `create-uber-principal` command

```text
databricks labs ucx create-uber-principal [--subscription-ids X]
```

**Requires Cloud IAM admin privileges.**

Once the [`assessment` workflow](#assessment-workflow) complete, you should run this command to create a service principal with the
_**read-only access to all storage**_ used by tables in this workspace. It will also configure the
[UCX Cluster Policy](#installation) & SQL Warehouse data access configuration to use this service principal for migration
workflows. Once migration is complete, this service principal should be unprovisioned.

On Azure, it creates a principal with `Storage Blob Data Contributor` role assignment on every storage account using
Azure Resource Manager APIs.

This command is one of prerequisites for the [table migration process](#table-migration).

[[back to top](#databricks-labs-ucx)]

## `migrate-credentials` command

```commandline
databricks labs ucx migrate-credentials
```

For Azure, this command prompts to confirm performing the following credential migration steps:
1. [RECOMMENDED] For each storage account, create access connectors with managed identities that have the
   `Storage Blob Data Contributor` role on the respective storage account. A storage credential is created for each
    access connector.
2. Migrate Azure Service Principals, which have `Storage Blob Data Contributor`,
   `Storage Blob Data Reader`, `Storage Blob Data Owner`, or custom roles on ADLS Gen2 locations that are being used in
   Databricks, to UC storage credentials. The Azure Service Principals to location mapping are listed
   in `/Users/{user_name}/.ucx/azure_storage_account_info.csv` which is generated by
   [`principal-prefix-access` command](#principal-prefix-access-command). Please review the file and delete the Service
   Principals you do not want to be migrated. The command will only migrate the Service Principals that have client
   secret stored in Databricks Secret.

  **Warning**: Service principals used to access storage accounts behind firewalls might cause connectivity issues. We
  recommend to use access connectors instead.

For AWS, this command migrates AWS Instance Profiles that are being used in Databricks, to UC storage credentials.
The AWS Instance Profiles to location mapping are listed in
{workspace ucx folder}/aws_instance_profile_info.csv which is generated by principal_prefix_access command.
Please review the file and delete the Instance Profiles you do not want to be migrated. <br/>The aws_profile parameter indicates the aws profile to use.

Once you're done with this command, run [`validate-external-locations` command](#validate-external-locations-command) after this one.

[[back to top](#databricks-labs-ucx)]

## `validate-external-locations` command

```text
databricks labs ucx validate-external-locations
```

Once the [`assessment` workflow](#assessment-workflow) finished successfully, [storage credentials](#migrate-credentials-command) are configured,
run this command to validate and report the missing Unity Catalog external locations to be created.

This command validates and provides mapping to external tables to external locations, also as Terraform configurations.

Once you're done with this command, proceed to the [`migrate-locations` command](#migrate-locations-command).

[[back to top](#databricks-labs-ucx)]

## `migrate-locations` command

```text
databricks labs ucx migrate-locations
```

Once the [`assessment` workflow](#assessment-workflow) finished successfully, and [storage credentials](#migrate-credentials-command) are configured,
run this command to have Unity Catalog external locations created. The candidate locations to be created are extracted from guess_external_locations
task in the assessment job. You can run [`validate-external-locations` command](#validate-external-locations-command) to check the candidate locations.

**Location ACLs:**
`migrate-locations` command applies any location ACL from existing cluster.
For Azure, it checks if there are any interactive cluster or SQL warehouse
which has service principals configured to access storage. It maps the storage account to the external location created and grants `CREATE_EXTERNAL_TABLE`,
`CREATE_EXTERNAL_VOLUME` and `READ_FILES` permission on the location to all the user who have access to the interactive cluster or SQL warehouse
For AWS, it checks any instance profiles mapped to the interactive cluster or SQL warehouse. It checks the mapping of instance profiles to the bucket. It then
maps the bucket to the external locations created and grants `CREATE_EXTERNAL_TABLE`, `CREATE_EXTERNAL_VOLUME` and `READ_FILES` permission on the location to all the user who have access to the interactive cluster
or SQL warehouse

Once you're done with this command, proceed to the [`create-table-mapping` command](#create-table-mapping-command).

[[back to top](#databricks-labs-ucx)]

## `create-table-mapping` command

```text
databricks labs ucx create-table-mapping
```

Once the [`assessment` workflow](#assessment-workflow) finished successfully
[workspace info is synchronized](#sync-workspace-info-command), run this command to create the initial
table mapping for review in CSV format in the Databricks Workspace:

```text
workspace_name,catalog_name,src_schema,dst_schema,src_table,dst_table
labs-azure,labs_azure,default,default,ucx_tybzs,ucx_tybzs
```

The format of the mapping file is as follows:

| **columns:** | **workspace_name**      | **catalog_name** | **src_schema** | **dst_schema** | **src_table** | **dst_table** |
|--------------|---------------------|--------------|----------------|----------------|---------------|---------------|
| values:      | data_engineering_ws | de_catalog   | database1      | database1      | table1        | table1        |

You are supposed to review this mapping and adjust it if necessary. This file is in CSV format, so that you can edit it
easier in your favorite spreadsheet application.

Once you're done with this command, [create catalogs and schemas](#create-catalogs-schemas-command). During
multiple runs of the table migration workflow, you can use the [`revert-migrated-tables` command](#revert-migrated-tables-command)
to revert the tables that were migrated in the previous run. You can also skip the tables that you don't want to migrate
using the [`skip` command](#skip-command).

This command is one of prerequisites for the [table migration process](#Table-Migration).

Once you're done with table migration, proceed to the [code migration](#code-migration-commands).

[[back to top](#databricks-labs-ucx)]

## `skip` command

```text
databricks labs ucx skip --schema X [--table Y] [--view Z]
```

Anytime after [`create-table-mapping` command](#create-table-mapping-command) is executed, you can run this command.

This command allows users to skip certain schemas, tables or views during the [table migration](#table-migration) process.
The command takes `--schema` and, optionally, `--table` and `--view` flags to specify the schema, table or view to skip.
If no `--table` flag is provided, all tables in the specified HMS database are skipped. The `--table` and `--view` can
only be used exclusively. This command is useful to temporarily disable migration of a particular schema, table or view.

Once you're done with table migration, proceed to the [code migration](#code-migration-commands).

[[back to top](#databricks-labs-ucx)]

## `unskip` command

```commandline
databricks labs ucx unskip --schema X [--table Y] [--view Z]
```

This command removes the mark set by the [`skip` command](#skip-command) on the given schema, table or view.

[[back to top](#databricks-labs-ucx)]

## `create-catalogs-schemas` command

```text
databricks labs ucx create-catalogs-schemas
```
After [`create-table-mapping` command](#create-table-mapping-command) is executed, you can run this command to have the required UC catalogs and schemas created.
This command is supposed to be run before migrating tables to UC using [table migration process](#Table-Migration).
Catalog & Schema ACL:
`create-catalogs-schemas` command also applies any catalog and schema ACL from existing clusters.
For Azure it checks if there are any interactive cluster or sql warehouse which has service principals configured to access storage.
It maps the storage account to the tables which has external location on those storage account created and grants `USAGE` access to
the schema and catalog if at least one such table is migrated to it.
For AWS, it checks any instance profiles mapped to the interactive cluster or sql warehouse. It checks the mapping of instance profiles
to the bucket. It then maps the bucket to the tables which has external location on those bucket created and grants `USAGE` access to
the schema and catalog if at least one such table is migrated to it.
[[back to top](#databricks-labs-ucx)]

## `assign-owner-group` command

```text
databricks labs ucx assign-owner-group
```

This command assigns the owner group to the UCX catalog and all migrated tables.
The owner group is an account group that will be designated as an owner to all migrated objects (catalogs, schemas, tables, views).
The principal running the command and later, the migration workflows, is required to be a member of this group.
The command will list all the groups the principal is a member of and allow the selection of the owner group.
It sets the default_owner_group property in the config.yml file.

[[back to top](#databricks-labs-ucx)]

## `migrate-tables` command

```text
databricks labs ucx migrate-tables
```

Anytime after [`create-table-mapping` command](#create-table-mapping-command) is executed, you can run this command.

This command kicks off the [table migration](#Table-Migration) process. It triggers the `migrate-tables` workflow,
and if there are HiveSerDe tables detected, prompt whether to trigger the `migrate-external-hiveserde-tables-in-place-experimental` workflow.

Table and View ACL:
`migrate-tables` command also applies any table and view ACL from existing clusters.
For Azure it checks if there are any interactive cluster or sql warehouse which has service principals configured to access storage.
It maps the storage account to the tables which has external location on those storage account created and grants either `SELECT` permission if
the service principal only has read access on the storage account and `ALL_PRIVILEGES` if the service principal has write access on the storage account
For AWS, it checks any instance profiles mapped to the interactive cluster or sql warehouse. It checks the mapping of instance profiles
to the bucket. It then maps the bucket to the tables which has external location on those bucket created and grants either `SELECT` permission if
the instance profile only has read access on the bucket and `ALL_PRIVILEGES` if the instance profile has write access on the bucket.

[[back to top](#databricks-labs-ucx)]

## `revert-migrated-tables` command

```text
databricks labs ucx revert-migrated-tables --schema X --table Y [--delete-managed]
```

Anytime after [`create-table-mapping` command](#create-table-mapping-command) is executed, you can run this command.

This command removes the `upgraded_from` property on a migrated table for re-migration in the [table migration](#Table-Migration) process.
This command is useful for developers and administrators who want to revert the migration of a table. It can also be used
to debug issues related to table migration.

Go back to the [`create-table-mapping` command](#create-table-mapping-command) after you're done with this command.

[[back to top](#databricks-labs-ucx)]

## `move` command

```text
databricks labs ucx move --from-catalog A --from-schema B --from-table C --to-catalog D --to-schema E
```

This command moves a UC table/tables from one schema to another schema after
the [table migration](#Table-Migration) process. This is useful for developers and administrators who want
to adjust their catalog structure after tables upgrade.

Users will be prompted whether tables/view are dropped after moving to new schema. This only applies to `MANAGED` tables and views.

This command moves different table types differently:
- `MANAGED` tables are deep-cloned to the new schema.
- `EXTERNAL` tables are dropped from the original schema, then created in the target schema using the same location.
This is due to Unity Catalog not supporting multiple tables with overlapping paths
- `VIEW` are recreated using the same view definition.

This command supports moving multiple tables at once, by specifying `*` as the table name.

[[back to top](#databricks-labs-ucx)]

## `alias` command

```text
databricks labs ucx alias --from-catalog A --from-schema B --from-table C --to-catalog D --to-schema E
```

This command aliases a UC table/tables from one schema to another schema in the same or different catalog.
It takes a `WorkspaceClient` object and `from` and `to` parameters as parameters and aliases the tables using
the `TableMove` class. This command is useful for developers and administrators who want to create an alias for a table.
It can also be used to debug issues related to table aliasing.

[[back to top](#databricks-labs-ucx)]

# Code migration commands

See the [migration process diagram](#migration-process) to understand the role of the code migration commands in the migration process.

After you're done with the [table migration](#Table-Migration), you can proceed to the code migration.

Once you're done with the code migration, you can run the [`cluster-remap` command](#cluster-remap-command) to remap the
clusters to be UC compatible.

[[back to top](#databricks-labs-ucx)]

## `lint-local-code` command

```text
databricks labs ucx lint-local-code
```

At any time, you can run this command to assess all migrations required in a local directory or a file. It only takes seconds to run and it
gives you an initial overview of what needs to be migrated without actually performing any migration. A great way to start a migration!

This command detects all dependencies, and analyzes them. It is still experimental and at the moment only supports Python and SQL files.
We expect this command to run within a minute on code bases up to 50.000 lines of code.
Future versions of `ucx` will add support for more source types, and more migration details.

When run from an IDE terminal, this command generates output as follows:
![img.png](docs/lint-local-code-output.png)
With modern IDEs, clicking on the file link opens the file at the problematic line

[[back to top](#databricks-labs-ucx)]

## `migrate-local-code` command

```text
databricks labs ucx migrate-local-code
```

**(Experimental)** Once [table migration](#Table-Migration) is complete, you can run this command to
migrate all python and SQL files in the current working directory. This command is highly experimental and
at the moment only supports Python and SQL files and discards code comments and formatting during
the automated transformation process.

[[back to top](#databricks-labs-ucx)]

## `migrate-dbsql-dashboards` command

```text
databricks labs ucx migrate-dbsql-dashboards [--dashboard-id <dashboard-id>]
```

**(Experimental)** Once [table migration](#Table-Migration) is complete, you can run this command to
migrate all Databricks SQL dashboards in the workspace. At this moment, this command is highly experimental and discards
formatting during the automated transformation process.

This command tags dashboards & queries that have been migrated with `migrated by UCX` tag. The original queries are
also backed up in the ucx installation folder, to allow for easy rollback (see [`revert-dbsql-dashboards` command](#revert-dbsql-dashboards-command)).

This command can be run with `--dashboard-id` flag to migrate a specific dashboard.

This command is incremental and can be run multiple times to migrate new dashboards.

[[back to top](#databricks-labs-ucx)]

## `revert-dbsql-dashboards` command

```text
databricks labs ucx revert-dbsql-dashboards [--dashboard-id <dashboard-id>]
```

**(Experimental)** This command reverts the migration of Databricks SQL dashboards in the workspace, after
`migrate-dbsql-dashboards` command is executed.

This command can be run with `--dashboard-id` flag to migrate a specific dashboard.

[[back to top](#databricks-labs-ucx)]

# Cross-workspace installations

When installing UCX across multiple workspaces, administrators need to keep UCX configurations in sync.
UCX will prompt you to select an account profile that has been defined in `~/.databrickscfg`. If you don't have one,
authenticate your machine with:

* `databricks auth login --host https://accounts.cloud.databricks.com/` (AWS)
* `databricks auth login --host https://accounts.azuredatabricks.net/` (Azure)

Ask your Databricks Account admin to run the [`sync-workspace-info` command](#sync-workspace-info-command) to sync the
workspace information with the UCX installations. Once the workspace information is synced, you can run the
[`create-table-mapping` command](#create-table-mapping-command) to align your tables with the Unity Catalog.

[[back to top](#databricks-labs-ucx)]

## `sync-workspace-info` command

```text
databricks --profile ACCOUNTS labs ucx sync-workspace-info
14:07:07  INFO [databricks.sdk] Using Azure CLI authentication with AAD tokens
14:07:07  INFO [d.labs.ucx] Account ID: ...
14:07:10  INFO [d.l.blueprint.parallel][finding_ucx_installations_16] finding ucx installations 10/88, rps: 16.415/sec
14:07:10  INFO [d.l.blueprint.parallel][finding_ucx_installations_0] finding ucx installations 20/88, rps: 32.110/sec
14:07:11  INFO [d.l.blueprint.parallel][finding_ucx_installations_18] finding ucx installations 30/88, rps: 39.786/sec
...
```

> Requires Databricks Account Administrator privileges. Use `--profile` to select the Databricks cli profile configured
> with access to the Databricks account console (with endpoint "https://accounts.cloud.databricks.com/"
> or "https://accounts.azuredatabricks.net").

This command uploads the workspace config to all workspaces in the account where `ucx` is installed. This command is
necessary to create an immutable default catalog mapping for [table migration](#Table-Migration) process and is the prerequisite
for [`create-table-mapping` command](#create-table-mapping-command).

If you cannot get account administrator privileges in reasonable time, you can take the risk and
run [`manual-workspace-info` command](#manual-workspace-info-command) to enter Databricks Workspace IDs and Databricks
Workspace names.

[[back to top](#databricks-labs-ucx)]

## `manual-workspace-info` command

```text
$ databricks labs ucx manual-workspace-info
14:20:36  WARN [d.l.ucx.account] You are strongly recommended to run "databricks labs ucx sync-workspace-info" by account admin,
 ... otherwise there is a significant risk of inconsistencies between different workspaces. This command will overwrite all UCX
 ... installations on this given workspace. Result may be consistent only within https://adb-987654321.10.azuredatabricks.net
Workspace name for 987654321 (default: workspace-987654321): labs-workspace
Next workspace id (default: stop): 12345
Workspace name for 12345 (default: workspace-12345): other-workspace
Next workspace id (default: stop):
14:21:19  INFO [d.l.blueprint.parallel][finding_ucx_installations_11] finding ucx installations 10/89, rps: 24.577/sec
14:21:19  INFO [d.l.blueprint.parallel][finding_ucx_installations_15] finding ucx installations 20/89, rps: 48.305/sec
...
14:21:20  INFO [d.l.ucx.account] Synchronised workspace id mapping for installations on current workspace
```

This command is only supposed to be run if the [`sync-workspace-info` command](#sync-workspace-info-command) cannot be
run. It prompts the user to enter the required information manually and creates the workspace info. This command is
useful for workspace administrators who are unable to use the `sync-workspace-info` command, because they are not
Databricks Account Administrators. It can also be used to manually create the workspace info in a new workspace.

[[back to top](#databricks-labs-ucx)]

## `create-account-groups` command

```text
$ databricks labs ucx create-account-groups [--workspace-ids 123,456,789]
```

**Requires Databricks Account Administrator privileges.** This command creates account-level groups if a workspace local
group is not present in the account. It crawls all workspaces configured in `--workspace-ids` flag, then creates
account level groups if a WS local group is not present in the account. If `--workspace-ids` flag is not specified, UCX
will create account groups for all workspaces configured in the account.

The following scenarios are supported, if a group X:
- Exist in workspaces A,B,C, and it has same members in there, it will be created in the account
- Exist in workspaces A,B but not in C, it will be created in the account
- Exist in workspaces A,B,C. It has same members in A,B, but not in C. Then, X and C_X will be created in the account

This command is useful for the setups, that don't have SCIM provisioning in place.

Once you're done with this command, proceed to the [group migration workflow](#group-migration-workflow).

[[back to top](#databricks-labs-ucx)]

## `validate-groups-membership` command

```text
$ databricks labs ucx validate-groups-membership
...
14:30:36  INFO [d.l.u.workspace_access.groups] Found 483 account groups
14:30:36  INFO [d.l.u.workspace_access.groups] No group listing provided, all matching groups will be migrated
14:30:36  INFO [d.l.u.workspace_access.groups] There are no groups with different membership between account and workspace
Workspace Group Name  Members Count  Account Group Name  Members Count  Difference
```

This command validates the groups to see if the groups at the account level and workspace level have different membership.
This command is useful for administrators who want to ensure that the groups have the correct membership. It can also be
used to debug issues related to group membership. See [group migration](docs/local-group-migration.md) and
[group migration](#group-migration-workflow) for more details.

Valid group membership is important to ensure users has correct access after legacy table ACL is migrated in [table migration process](#Table-Migration)

[[back to top](#databricks-labs-ucx)]

## `validate-table-locations` command

```text
$ databricks labs ucx validate-table-locations [--workspace-ids 123,456,789]
...
11:39:36  WARN [d.l.u.account.aggregate] Workspace 99999999 does not have UCX installed
11:39:37  WARN [d.l.u.account.aggregate] Overlapping table locations: 123456789:hive_metastore.database.table and 987654321:hive_metastore.database.table
11:39:37  WARN [d.l.u.account.aggregate] Overlapping table locations: 123456789:hive_metastore.database.table and 123456789:hive_metastore.another_database.table
```

This command validates the table locations by checking for overlapping table locations in the workspace and across
workspaces. Unity catalog does not allow overlapping table locations, also not between tables in different catalogs.
Overlapping table locations need to be resolved by the user before running the table migration.

Options to resolve tables with overlapping locations are:
- Move one table and [skip](#skip-command) the other(s).
- Duplicate the tables by copying the data into a managed table and [skip](#skip-command) the original tables.

Considerations when resolving tables with overlapping locations are:
- Migrate the tables one workspace at a time:
  - Let later migrated workspaces read tables from the earlier migrated workspace catalogs.
  - [Move](#move-command) tables between schemas and catalogs when it fits the data management model.
- The tables might have different:
  - Metadata, like:
    - Column schema (names, types, order)
    - Description
    - Tags
  - ACLs

[[back to top](#databricks-labs-ucx)]

## `cluster-remap` command

```text
$ databricks labs ucx cluster-remap
21:29:38  INFO [d.labs.ucx] Remapping the Clusters to UC
Cluster Name                                            Cluster Id
Field Eng Shared UC LTS Cluster                         0601-182128-dcbte59m
Shared Autoscaling Americas cluster                     0329-145545-rugby794
```
```text
Please provide the cluster id's as comma separated value from the above list (default: <ALL>):
```

Once you're done with the [code migration](#code-migration-commands), you can run this command to remap the clusters to UC enabled.

This command will remap the cluster to uc enabled one. When we run this command it will list all the clusters
and its id's and asks to provide the cluster id's as comma separated value which has to be remapped, by default it will take all cluster ids.
Once we provide the cluster id's it will update these clusters to UC enabled.Back up of the existing cluster
config will be stored in backup folder inside the installed location(backup/clusters/cluster_id.json) as a json file.This will help
to revert the cluster remapping.

You can revert the cluster remapping using the [`revert-cluster-remap` command](#revert-cluster-remap-command).

[[back to top](#databricks-labs-ucx)]

## `revert-cluster-remap` command

```text
$ databricks labs ucx revert-cluster-remap
21:31:29  INFO [d.labs.ucx] Reverting the Remapping of the Clusters from UC
21:31:33  INFO [d.labs.ucx] 0301-055912-4ske39iq
21:31:33  INFO [d.labs.ucx] 0306-121015-v1llqff6
Please provide the cluster id's as comma separated value from the above list (default: <ALL>):
```

If a customer want's to revert the cluster remap done using the [`cluster-remap` command](#cluster-remap-command) they can use this command to revert
its configuration from UC to original one.It will iterate through the list of clusters from the backup folder and reverts the
cluster configurations to original one.This will also ask the user to provide the list of clusters that has to be reverted as a prompt.
By default, it will revert all the clusters present in the backup folder

[[back to top](#databricks-labs-ucx)]

## `upload` command

```text
$ databricks labs ucx upload --file <file_path> --run-as-collection True
21:31:29 WARNING [d.labs.ucx] The schema of CSV files is NOT validated, ensure it is correct
21:31:29 INFO [d.labs.ucx] Finished uploading: <file_path>
```

Upload a file to a single workspace (`--run-as-collection False`) or a collection of workspaces
(`--run-as-collection True`). This command is especially useful when uploading the same file to multiple workspaces.

## `download` command

```text
$ databricks labs ucx download --file <file_path> --run-as-collection True
21:31:29 INFO [d.labs.ucx] Finished downloading: <file_path>
```

Download a csv file from a single workspace (`--run-as-collection False`) or a collection of workspaces
(`--run-as-collection True`). This command is especially useful when downloading the same file from multiple workspaces.

## `join-collection` command

```text
$ databricks labs ucx join-collection --workspace-ids <comma seperate list of workspace ids> --profile <account-profile>
```

`join-collection` command joins 2 or more workspaces into a collection. This helps in running supported cli commands as a collection
`join-collection` command updates config.yml file on each workspace ucx installation with installed_workspace_ids attribute.
In order to run `join-collectioon` command a user should:
 - be an Account admin on the Databricks account
 - be a Workspace admin on all the workspaces to be joined as a collection) or a collection of workspaces
 - have installed UCX on the workspace
The `join-collection` command will fail and throw an error msg if the above conditions are not met.

## collection eligible command

Once `join-collection` command is run, it allows user to run multiple cli commands as a collection. The following cli commands
are eligible to be run as a collection. User can run the below commands as collection by passing an additional flag `--run-as-collection=True`
- `ensure-assessment-run`
- `create-table-mapping`
- `principal-prefix-access`
- `migrate-credentials`
- `create-uber-principal`
- `create-missing-principals`
- `validate-external-location`
- `migrate-locations`
- `create-catalog-schemas`
- `migrate-tables`
- `migrate-acls`
- `migrate-dbsql-dashboards`
- `validate-group-membership`
Ex: `databricks labs ucx ensure-assessment-run --run-as-collection=True`


# Common Challenges and the Solutions
Users might encounter some challenges while installing and executing UCX. Please find the listing of some common challenges and the solutions below.

[[back to top](#databricks-labs-ucx)]

### Network Connectivity Issues

**From local machine to the Databricks Account and Workspace:** UCX
installation process has to be run from the local laptop using
Databricks CLI and it will deploy the latest version of UCX into the
Databricks workspace. For this reason, the Databricks account and
workspace needs to be accessible from the laptop. Sometimes, the
workspace might have a network isolation, like it can only be reached
from a VPC, or from a specific IP range.

**Solution:** Please check that your laptop has network connectivity to
the Databricks account and workspace. If not, you might need to be
connected to a VPN or configure an HTTP proxy to access your workspace.

**From local machine to GitHub:** UCX needs internet access to connect to GitHub (https://api.github.com
and https://raw.githubusercontent.com) for downloading the tool from the machine running the installation. The
installation will fail if there is no internet connectivity to these URLs.

**Solution:** Ensure that GitHub is reachable from the local machine. If
not, make necessary changes to the network/firewall settings.

**From Databricks workspace to PyPi:** There are some dependent
libraries which need to be installed from
[<u>pypi.org</u>](https://pypi.org/) to run the UCX workflows from the
Databricks workspace. If the workspace doesn’t have network
connectivity, then the job might fail with
NO_MATCHING_DISTRIBUTION_ERROR.

**Solution:** Version 0.24.0 of UCX supports workspace with no internet
access. Please upgrade UCX and rerun the installation. Reply *yes* to
the question "Does the given workspace block Internet access?" asked
during installation. It will then upload all necessary dependencies to
the workspace. Also, please note that UCX uses both UC and non-UC
enabled clusters. If you have different proxy settings for each, then
please update the necessary proxies (eg. with init scripts) for each
cluster type.

**Local machine to Databricks Account and Workspace connection failed due to proxy and self-signed cert:**
When customer uses web proxy and self-singed certification, UCX may not be able to connect to Account and Workspace
with following errors:
```
File "/Users/userabc/.databricks/labs/ucx/state/venv/lib/python3.10/site-packages/urllib3/connectionpool.py", line 466, in _make_request
    self._validate_conn(conn)
  File "/Users/userabc/.databricks/labs/ucx/state/venv/lib/python3.10/site-packages/urllib3/connectionpool.py", line 1095, in _validate_conn
    conn.connect()
  File "/Users/userabc/.databricks/labs/ucx/state/venv/lib/python3.10/site-packages/urllib3/connection.py", line 652, in connect
    sock_and_verified = _ssl_wrap_socket_and_match_hostname(
  File "/Users/userabc/.databricks/labs/ucx/state/venv/lib/python3.10/site-packages/urllib3/connection.py", line 805, in _ssl_wrap_socket_and_match_hostname
    ssl_sock = ssl_wrap_socket(
  File "/Users/userabc/.databricks/labs/ucx/state/venv/lib/python3.10/site-packages/urllib3/util/ssl_.py", line 465, in ssl_wrap_socket
    ssl_sock = _ssl_wrap_socket_impl(sock, context, tls_in_tls, server_hostname)
  File "/Users/userabc/.databricks/labs/ucx/state/venv/lib/python3.10/site-packages/urllib3/util/ssl_.py", line 509, in _ssl_wrap_socket_impl
    return ssl_context.wrap_socket(sock, server_hostname=server_hostname)
  File "/opt/homebrew/Cellar/python@3.10/3.10.14/Frameworks/Python.framework/Versions/3.10/lib/python3.10/ssl.py", line 513, in wrap_socket
    return self.sslsocket_class._create(
  File "/opt/homebrew/Cellar/python@3.10/3.10.14/Frameworks/Python.framework/Versions/3.10/lib/python3.10/ssl.py", line 1104, in _create
    self.do_handshake()
  File "/opt/homebrew/Cellar/python@3.10/3.10.14/Frameworks/Python.framework/Versions/3.10/lib/python3.10/ssl.py", line 1375, in do_handshake
    self._sslobj.do_handshake()
ssl.SSLCertVerificationError: [SSL: CERTIFICATE_VERIFY_FAILED] certificate verify failed: self-signed certificate in certificate chain (_ssl.c:1007)
```

**Solution:**  set both `REQUESTS_CA_BUNDLE` and `CURL_CA_BUNDLE`
[to force requests library to set verify=False](https://github.com/psf/requests/blob/8c211a96cdbe9fe320d63d9e1ae15c5c07e179f8/requests/sessions.py#L718)
as well as set `SSL_CERT_DIR` env var pointing to the proxy CA cert for the urllib3 library.

[[back to top](#databricks-labs-ucx)]

### Insufficient Privileges

**User is not a Databricks workspace administrator:** User running the
installation needs to be a workspace administrator as the CLI will
deploy the UCX tool into the workspace, create jobs, and dashboards.

**Solution:** Identify a workspace admin from your team and ask them to
install UCX with their authentication, or request a workspace
administrator to grant you temporary administrator privileges to run the
installation. More details on the issues that you can run into if
you are not an admin (and some possible solutions) can be found
[here](docs/troubleshooting.md#resolving-common-errors-on-ucx-install).

**User is not a Cloud IAM Administrator:** Cloud CLI needs to be
installed in the local machine for certain cloud related activities,
like creating an [<u>uber
principal</u>](#create-uber-principal-command).
For this, the user needs Cloud IAM Administrator privileges.

**Solution:** Work with a cloud administrator in your organization to
run the commands that need cloud administrator rights.

Admin privileges required for commands:

| **CLI command**                                                                                  | **Admin privileges** |
|--------------------------------------------------------------------------------------------------|----|
| [<u>install</u>](#install-ucx)                                                                   | Workspace Admin |
| [<u>account install</u>](#advanced-installing-ucx-on-all-workspaces-within-a-databricks-account) | Account Admin |
| [<u>create-account-groups</u>](#create-account-groups-command)                                   | Account Admin |
| [<u>validate-groups-membership</u>](#validate-groups-membership-command)                         | Account Admin |
| [<u>create-uber-principal</u>](#create-uber-principal-command)                                   | Cloud Admin |
| [<u>principal-prefix-access</u>](#principal-prefix-access-command)                               | Cloud Admin |
| [<u>create-missing-principals</u>](#create-missing-principals-command-aws-only)                  | Cloud Admin |
| [<u>delete-missing-principals</u>](#delete-missing-principals-command-aws-only)                  | Cloud Admin |
| [<u>migrate-credentials</u>](#migrate-credentials-command)                                       | Cloud Admin, Account Admin / Metastore Admin / CREATE STORAGE CREDENTIAL privilege |
| [<u>migrate-location</u>](#migrate-locations-command)                                            | Metastore Admin / CREATE EXTERNAL LOCATION privilege |
| [<u>create-catalogs-schemas</u>](#create-catalogs-schemas-command)                               | Metastore Admin / CREATE CATALOG privilege |
| [<u>sync-workspace-info</u>](#sync-workspace-info-command)                                       | Account Admin |
| [<u>manual-workspace-info</u>](#manual-workspace-info-command)                                   | Workspace Admin |

[[back to top](#databricks-labs-ucx)]

### Version Issues

**Python:** UCX needs Python version 3.10 or later.

**Solution:** Check the current version using `python --version`. If the
version is lower than 3.10, upgrade the local Python version to 3.10 or
higher.

**Databricks CLI:** Databricks CLI v0.213 or higher is needed.

**Solution:** Check the current version with `databricks --version`. For
lower versions of CLI,
[<u>update</u>](https://docs.databricks.com/en/dev-tools/cli/install.html#update)
the Databricks CLI on the local machine.

**UCX:** When you install UCX, you get the latest version. But since UCX
is being actively developed, new versions are released frequently. There
might be issues if you have run the assessment with a much earlier
version, and then trying to run the migration workflows with the latest
UCX version.

**Solution:** Upgrade UCX, and rerun the assessment job before running
the migration workflows. For some reason, if you want to install a
specific version of UCX, you can do it using the command
`databricks labs install ucx@\<version\>`, for example,
`databricks labs install ucx@v0.21.0`.

[[back to top](#databricks-labs-ucx)]

### Authentication Issues

**Workspace Level:** If you are facing authentication issues while
setting up Databricks CLI, please refer to the
[Cryptic errors on authentication](docs/troubleshooting.md#cryptic-errors-on-authentication)
section to resolve the common errors related to authentication,
profiles, and tokens.

**Account Level:** Not only workspace, but account level authentication
is also needed for installing UCX. If you do not have an account
configured in .databrickscfg, you will get an error message
".databrickscfg does not contain account profiles; please create one
first".

**Solution:** To authenticate with a Databricks account, consider using
one of the following authentication types: [<u>OAuth machine-to-machine
(M2M)
authentication</u>](https://docs.databricks.com/en/dev-tools/cli/authentication.html#m2m-auth),
[<u>OAuth user-to-machine (U2M)
authentication</u>](https://docs.databricks.com/en/dev-tools/cli/authentication.html#u2m-auth),
[<u>Basic authentication
(legacy)</u>](https://docs.databricks.com/en/dev-tools/cli/authentication.html#basic-auth).

[[back to top](#databricks-labs-ucx)]

### Multiple Profiles in Databricks CLI

**Workspace Level:** More than one workspace profile can be configured
in the .databrickscfg file. For example, you can have profiles set for
Dev and Prod workspaces. You want to install UCX only for the Prod
workspace.

**Solution:** The Databricks CLI provides an option to select the
[<u>profile</u>](https://docs.databricks.com/en/dev-tools/cli/profiles.html)
using `--profile \<profile_name\>` or `-p \<profile_name\>`. You can
test that the correct workspace is getting selected by running any
Databricks CLI command. For example, you can run `databricks clusters
list -p prod` and check that the Prod clusters are being returned. Once
the profile is verified, you can run UCX install for that specific
profile: `databricks labs install ucx -p prod`.

**Account Level:** Multiple account level profiles are set in the
.databrickscfg file.

**Solution:** The installation command `databricks labs install ucx`
will provide an option to select one account profile.

[[back to top](#databricks-labs-ucx)]

### Workspace has an external Hive Metastore (HMS)

**External HMS connectivity from UCX clusters:** If the workspace has an
external HMS, the clusters running the UCX jobs need to have specific
configurations to connect to the external HMS. Otherwise, UCX assessment
will not be able to assess the tables on HMS.

**Solution:** Use a cluster policy before installation to set the
required Spark config for connecting to the external HMS, or manually
edit the cluster post-installation to have the correct configurations.
Detailed steps can be found
[<u>here</u>](docs/external_hms_glue.md).

**External HMS connectivity from UCX SQL warehouse:** UCX requires a SQL
warehouse to create tables, run queries, create and refresh dashboards.
If you already have a Pro or Serverless warehouse connected to the
external HMS, you can select the same warehouse for UCX. You will also
be given an option (during installation) to create a new warehouse for
UCX. If you have never used a warehouse before, the new warehouse
created might not have proper configuration set to connect to the
external HMS.

**Solution:** Set Spark configuration for connecting to external HMS in
the Admin Settings of SQL warehouse. This will only be needed if the
admin settings do not have the configurations already set. For example,
add *spark.hadoop.javax.jdo.option.ConnectionURL \<connectionstring\>*
under Data Access Configuration of SQL Warehouse Admin Settings.

[[back to top](#databricks-labs-ucx)]

### Verify the Installation

Once the UCX command `databricks labs install ucx` has completed
successfully, the installation can be verified with the following steps:

1.  Go to the Databricks Catalog Explorer and check if a new schema for
    ucx is available in Hive Metastore with all empty tables.

2.  Check that the UCX jobs are visible under Workflows.

3.  Run the assessment. This will start the UCX clusters, crawl through
    the workspace, and display results in the UCX dashboards. In case of
    external HMS, verify from the results that the assessment has
    analyzed the external HMS tables.

[[back to top](#databricks-labs-ucx)]

# Star History

[![Star History Chart](https://api.star-history.com/svg?repos=databrickslabs/ucx&type=Date)](https://star-history.com/#databrickslabs/ucx)

[[back to top](#databricks-labs-ucx)]

# Project Support
Please note that all projects in the databrickslabs GitHub account are provided for your exploration only, and are not formally supported by Databricks with Service Level Agreements (SLAs).  They are provided AS-IS, and we do not make any guarantees of any kind.  Please do not submit a support ticket relating to any issues arising from the use of these projects.

Any issues discovered through the use of this project should be filed as GitHub Issues on the Repo.  They will be reviewed as time permits, but there are no formal SLAs for support.

            

Raw data

            {
    "_id": null,
    "home_page": null,
    "name": "databricks-labs-ucx",
    "maintainer": null,
    "docs_url": null,
    "requires_python": ">=3.10",
    "maintainer_email": "Serge Smertin <serge.smertin@databricks.com>, Liran Bareket <liran.bareket@databricks.com>, Marcin Wojtyczka <marcin.wojtyczka@databricks.com>, Ziyuan Qin <ziyuan.qin@databricks.com>, William Conti <william.conti@databricks.com>, Hari Selvarajan <hari.selvarajan@databricks.com>, Vuong Nguyen <vuong.nguyen@databricks.com>",
    "keywords": "Databricks, Unity Catalog",
    "author": null,
    "author_email": "Serge Smertin <serge.smertin@databricks.com>",
    "download_url": "https://files.pythonhosted.org/packages/34/d0/55abb903656625b510479855e44286bea973271b78c14f2f36c901bddbea/databricks_labs_ucx-0.53.1.tar.gz",
    "platform": null,
    "description": "Databricks Labs UCX\n===\n![UCX by Databricks Labs](docs/logo-no-background.png)\n\nThe companion for upgrading to Unity Catalog (UC).\n\nAfter [installation](#install-ucx), ensure to [trigger](#ensure-assessment-run-command) the [assessment workflow](#assessment-workflow),\nso that you'll be able to [scope the migration](docs/assessment.md) and execute the [group migration workflow](#group-migration-workflow).\n\nThe [README notebook](#readme-notebook), which can be found in the installation folder contains further instructions and explanations of the different ucx workflows & dashboards.\nOnce the migration is scoped, you can start with the [table migration process](#Table-Migration).\n\n\nMore workflows, like notebook code migration are coming in future releases.\n\n\nUCX also provides a number of command line utilities accessible via `databricks labs ucx`.\n\nFor questions, troubleshooting or bug fixes, please see our [troubleshooting guide](docs/troubleshooting.md) or submit [an issue](https://github.com/databrickslabs/ucx/issues).\nSee [contributing instructions](CONTRIBUTING.md) to help improve this project.\n\n[![build](https://github.com/databrickslabs/ucx/actions/workflows/push.yml/badge.svg)](https://github.com/databrickslabs/ucx/actions/workflows/push.yml) [![codecov](https://codecov.io/github/databrickslabs/ucx/graph/badge.svg?token=p0WKAfW5HQ)](https://codecov.io/github/databrickslabs/ucx)  ![linesofcode](https://aschey.tech/tokei/github/databrickslabs/ucx?category=code)\n\n<!-- TOC -->\n* [Databricks Labs UCX](#databricks-labs-ucx)\n* [Installation](#installation)\n  * [Installation requirements](#installation-requirements)\n  * [Authenticate Databricks CLI](#authenticate-databricks-cli)\n  * [Install UCX](#install-ucx)\n  * [Installation resources](#installation-resources)\n  * [Installation folder](#installation-folder)\n    * [Readme notebook](#readme-notebook)\n    * [Debug notebook](#debug-notebook)\n    * [Debug logs](#debug-logs)\n    * [Installation configuration](#installation-configuration)\n  * [Advanced installation options](#advanced-installation-options)\n    * [Force install over existing UCX](#force-install-over-existing-ucx)\n    * [Installing UCX on all workspaces within a Databricks account](#installing-ucx-on-all-workspaces-within-a-databricks-account)\n    * [Installing UCX with company hosted PYPI mirror](#installing-ucx-with-company-hosted-pypi-mirror)\n  * [Upgrading UCX for newer versions](#upgrading-ucx-for-newer-versions)\n  * [Uninstall UCX](#uninstall-ucx)\n* [Migration process](#migration-process)\n  * [Table migration process](#table-migration-process)\n    * [Table mapping](#table-mapping)\n      * [Step 1 : Create the mapping file](#step-1--create-the-mapping-file)\n      * [Step 2: Update the mapping file](#step-2-update-the-mapping-file)\n    * [Data access](#data-access)\n      * [Step 1 : Map cloud principals to cloud storage locations](#step-1--map-cloud-principals-to-cloud-storage-locations)\n      * [Step 2 : Create or modify cloud principals and credentials](#step-2--create-or-modify-cloud-principals-and-credentials)\n      * [Step 3: Create the \"uber\" Principal](#step-3-create-the-uber-principal)\n    * [New Unity Catalog resources](#new-unity-catalog-resources)\n      * [Step 0: Attach a metastore](#step-0-attach-a-metastore)\n      * [Step 1: Create external Locations](#step-1-create-external-locations)\n        * [Step 2: Create Catalogs and Schemas](#step-2-create-catalogs-and-schemas)\n    * [Migrate Hive metastore data objects](#migrate-hive-metastore-data-objects)\n    * [Odds and Ends](#odds-and-ends)\n      * [Skip migrating schemas, tables or views](#skip-migrating-schemas-tables-or-views)\n      * [Move data objects](#move-data-objects)\n      * [Alias data objects](#alias-data-objects)\n      * [Revert migrated data objects](#revert-migrated-data-objects)\n* [Workflows](#workflows)\n  * [Assessment workflow](#assessment-workflow)\n  * [Group migration workflow](#group-migration-workflow)\n  * [Table migration workflows](#table-migration-workflows)\n    * [Migrate tables](#migrate-tables)\n    * [Migrate external Hive SerDe tables](#migrate-external-hive-serde-tables)\n    * [Migrate external tables CTAS](#migrate-external-tables-ctas)\n  * [Post-migration data reconciliation workflow](#post-migration-data-reconciliation-workflow)\n    * [[LEGACY] Scan tables in mounts Workflow](#legacy-scan-tables-in-mounts-workflow)\n      * [<b>Always run this workflow AFTER the assessment has finished</b>](#balways-run-this-workflow-after-the-assessment-has-finishedb)\n    * [[LEGACY] Migrate tables in mounts Workflow](#legacy-migrate-tables-in-mounts-workflow)\n  * [[EXPERIMENTAL] Migration Progress Workflow](#experimental-migration-progress-workflow)\n* [Dashboards](#dashboards)\n* [Linter message codes](#linter-message-codes)\n  * [`cannot-autofix-table-reference`](#cannot-autofix-table-reference)\n  * [`catalog-api-in-shared-clusters`](#catalog-api-in-shared-clusters)\n  * [`changed-result-format-in-uc`](#changed-result-format-in-uc)\n  * [`direct-filesystem-access-in-sql-query`](#direct-filesystem-access-in-sql-query)\n  * [`direct-filesystem-access`](#direct-filesystem-access)\n  * [`dependency-not-found`](#dependency-not-found)\n  * [`jvm-access-in-shared-clusters`](#jvm-access-in-shared-clusters)\n  * [`legacy-context-in-shared-clusters`](#legacy-context-in-shared-clusters)\n  * [`not-supported`](#not-supported)\n  * [`notebook-run-cannot-compute-value`](#notebook-run-cannot-compute-value)\n  * [`python-udf-in-shared-clusters`](#python-udf-in-shared-clusters)\n  * [`rdd-in-shared-clusters`](#rdd-in-shared-clusters)\n  * [`spark-logging-in-shared-clusters`](#spark-logging-in-shared-clusters)\n  * [`sql-parse-error`](#sql-parse-error)\n  * [`sys-path-cannot-compute-value`](#sys-path-cannot-compute-value)\n  * [`table-migrated-to-uc`](#table-migrated-to-uc)\n  * [`to-json-in-shared-clusters`](#to-json-in-shared-clusters)\n  * [`unsupported-magic-line`](#unsupported-magic-line)\n* [Utility commands](#utility-commands)\n  * [`logs` command](#logs-command)\n  * [`ensure-assessment-run` command](#ensure-assessment-run-command)\n  * [`update-migration-progress` command](#update-migration-progress-command)\n  * [`repair-run` command](#repair-run-command)\n  * [`workflows` command](#workflows-command)\n  * [`open-remote-config` command](#open-remote-config-command)\n  * [`installations` command](#installations-command)\n  * [`report-account-compatibility` command](#report-account-compatibility-command)\n  * [`export-assessment` command](#export-assessment-command)\n* [Metastore related commands](#metastore-related-commands)\n  * [`show-all-metastores` command](#show-all-metastores-command)\n  * [`assign-metastore` command](#assign-metastore-command)\n  * [`create-ucx-catalog` command](#create-ucx-catalog-command)\n* [Table migration commands](#table-migration-commands)\n  * [`principal-prefix-access` command](#principal-prefix-access-command)\n    * [Access for AWS S3 Buckets](#access-for-aws-s3-buckets)\n    * [Access for Azure Storage Accounts](#access-for-azure-storage-accounts)\n  * [`create-missing-principals` command (AWS Only)](#create-missing-principals-command-aws-only)\n  * [`delete-missing-principals` command (AWS Only)](#delete-missing-principals-command-aws-only)\n  * [`create-uber-principal` command](#create-uber-principal-command)\n  * [`migrate-credentials` command](#migrate-credentials-command)\n  * [`validate-external-locations` command](#validate-external-locations-command)\n  * [`migrate-locations` command](#migrate-locations-command)\n  * [`create-table-mapping` command](#create-table-mapping-command)\n  * [`skip` command](#skip-command)\n  * [`unskip` command](#unskip-command)\n  * [`create-catalogs-schemas` command](#create-catalogs-schemas-command)\n  * [`assign-owner-group` command](#assign-owner-group-command)\n  * [`migrate-tables` command](#migrate-tables-command)\n  * [`revert-migrated-tables` command](#revert-migrated-tables-command)\n  * [`move` command](#move-command)\n  * [`alias` command](#alias-command)\n* [Code migration commands](#code-migration-commands)\n  * [`lint-local-code` command](#lint-local-code-command)\n  * [`migrate-local-code` command](#migrate-local-code-command)\n  * [`migrate-dbsql-dashboards` command](#migrate-dbsql-dashboards-command)\n  * [`revert-dbsql-dashboards` command](#revert-dbsql-dashboards-command)\n* [Cross-workspace installations](#cross-workspace-installations)\n  * [`sync-workspace-info` command](#sync-workspace-info-command)\n  * [`manual-workspace-info` command](#manual-workspace-info-command)\n  * [`create-account-groups` command](#create-account-groups-command)\n  * [`validate-groups-membership` command](#validate-groups-membership-command)\n  * [`validate-table-locations` command](#validate-table-locations-command)\n  * [`cluster-remap` command](#cluster-remap-command)\n  * [`revert-cluster-remap` command](#revert-cluster-remap-command)\n  * [`upload` command](#upload-command)\n  * [`download` command](#download-command)\n  * [`join-collection` command](#join-collection-command)\n  * [collection eligible command](#collection-eligible-command)\n* [Common Challenges and the Solutions](#common-challenges-and-the-solutions)\n    * [Network Connectivity Issues](#network-connectivity-issues)\n    * [Insufficient Privileges](#insufficient-privileges)\n    * [Version Issues](#version-issues)\n    * [Authentication Issues](#authentication-issues)\n    * [Multiple Profiles in Databricks CLI](#multiple-profiles-in-databricks-cli)\n    * [Workspace has an external Hive Metastore (HMS)](#workspace-has-an-external-hive-metastore-hms)\n    * [Verify the Installation](#verify-the-installation)\n* [Star History](#star-history)\n* [Project Support](#project-support)\n<!-- TOC -->\n\n# Installation\n\nUCX installation is covered by this section.\n\n## Installation requirements\n\nUCX has the following installation requirements:\n- Databricks CLI v0.213 or later. See [instructions](#authenticate-databricks-cli).\n- Python 3.10 or later. See [Windows](https://www.python.org/downloads/windows/) instructions.\n- Databricks Premium or Enterprise workspace.\n- Network access to your Databricks Workspace used for the [installation process](#install-ucx).\n- Network access to the Internet for [pypi.org](https://pypi.org) and [github.com](https://github.com) from machine running the installation.\n- Databricks Workspace Administrator privileges for the user, that runs the installation. Running UCX as a Service Principal is not supported.\n- Account level Identity Setup. See instructions for [AWS](https://docs.databricks.com/en/administration-guide/users-groups/best-practices.html), [Azure](https://learn.microsoft.com/en-us/azure/databricks/administration-guide/users-groups/best-practices), and [GCP](https://docs.gcp.databricks.com/administration-guide/users-groups/best-practices.html).\n- Unity Catalog Metastore Created (per region). See instructions for [AWS](https://docs.databricks.com/en/data-governance/unity-catalog/create-metastore.html), [Azure](https://learn.microsoft.com/en-us/azure/databricks/data-governance/unity-catalog/create-metastore), and [GCP](https://docs.gcp.databricks.com/data-governance/unity-catalog/create-metastore.html).\n- If your Databricks Workspace relies on an external Hive Metastore (such as AWS Glue), make sure to read [this guide](docs/external_hms_glue.md).\n- A PRO or Serverless SQL Warehouse to render the [report](docs/assessment.md) for the [assessment workflow](#assessment-workflow).\n\nOnce you [install UCX](#install-ucx), you can proceed to the [assessment workflow](#assessment-workflow) to ensure\nthe compatibility of your workspace with Unity Catalog.\n\n[[back to top](#databricks-labs-ucx)]\n\n## Authenticate Databricks CLI\n\nWe only support installations and upgrades through [Databricks CLI](https://docs.databricks.com/en/dev-tools/cli/index.html), as UCX requires an installation script run\nto make sure all the necessary and correct configurations are in place. Install Databricks CLI on macOS:\n![macos_install_databricks](docs/macos_1_databrickslabsmac_installdatabricks.gif)\n\nInstall Databricks CLI on Windows:\n![windows_install_databricks.png](docs/windows_install_databricks.png)\n\nOnce you install Databricks CLI, authenticate your current machine to a Databricks Workspace:\n\n```commandline\ndatabricks auth login --host WORKSPACE_HOST\n```\n\nTo enable debug logs, simply add `--debug` flag to any command.\n\n[[back to top](#databricks-labs-ucx)]\n\n## Install UCX\n\nInstall UCX via Databricks CLI:\n\n```commandline\ndatabricks labs install ucx\n```\n\nYou'll be prompted to select a [configuration profile](https://docs.databricks.com/en/dev-tools/auth.html#databricks-client-unified-authentication) created by `databricks auth login` command.\n\nOnce you install, proceed to the [assessment workflow](#assessment-workflow) to ensure the compatibility of your workspace with UCX.\n\nThe `WorkspaceInstaller` class is used to create a new configuration for Unity Catalog migration in a Databricks workspace.\nIt guides the user through a series of prompts to gather necessary information, such as selecting an inventory database, choosing\na PRO or SERVERLESS SQL warehouse, specifying a log level and number of threads, and setting up an external Hive Metastore if necessary.\nUpon the first installation, you're prompted for a workspace local [group migration strategy](docs/group_name_conflict.md).\nBased on user input, the class creates a new cluster policy with the specified configuration. The user can review and confirm the configuration,\nwhich is saved to the workspace and can be opened in a web browser.\n\nThe [`WorkspaceInstallation`](src/databricks/labs/ucx/install.py) manages the installation and uninstallation of UCX in a workspace. It handles\nthe configuration and exception management during the installation process. The installation process creates dashboards, databases, and jobs.\nIt also includes the creation of a database with given configuration and the deployment of workflows with specific settings. The installation\nprocess can handle exceptions and infer errors from job runs and task runs. The workspace installation uploads wheels, creates cluster policies,\nand wheel runners to the workspace. It can also handle the creation of job tasks for a given task, such as job dashboard tasks, job notebook tasks,\nand job wheel tasks. The class handles the installation of UCX, including configuring the workspace, installing necessary libraries, and verifying\nthe installation, making it easier for users to migrate their workspaces to UCX.\nAt the end of the installation, the user will be prompted if the current installation needs to join an existing collection (create new collection if none present).\nFor large organization with many workspaces, grouping workspaces into collection helps in managing UCX migration at collection level (instead of workspaces level)\nUser should be an account admin to be able to join a collection.\n\nAfter this, UCX will be installed locally and a number of assets will be deployed in the selected workspace.\nThese assets are available under the installation folder, i.e. `/Applications/ucx` is the default installation folder. Please check [here](#advanced-force-install-over-existing-ucx) for more details.\n\nYou can also install a specific version by specifying it like `@v0.13.2` - `databricks labs install ucx@v0.13.2`.\n\n![macos_install_ucx](docs/macos_2_databrickslabsmac_installucx.gif)\n\n[[back to top](#databricks-labs-ucx)]\n\n## Installation resources\n\nThe following resources are installed by UCX:\n\n| Installed UCX resources                           | Description                                                                                      |\n|---------------------------------------------------|--------------------------------------------------------------------------------------------------|\n| [Inventory database](./docs/table_persistence.md) | A Hive metastore database/schema in which UCX persist inventory required for the upgrade process |\n| [Workflows](#workflows)                           | Workflows to execute UCX                                                                         |\n| [Dashboards](#dashboards)                         | Dashboards to visualize UCX outcomes                                                             |\n| [Installation folder](#installation-folder)       | A workspace folder containing UCX files in `/Applications/ucx/`.                                 |\n\n## Installation folder\n\nUCX is in installed in the workspace folder `/Applications/ucx/`. This folder contains UCX's code resources, like the\n[source code](./src) from this GitHub repository and the [dashboard](#dashboards). Generally, these resources are not\n*directly* used by UCX users. Resources that can be of importance to users are detailed in the subsections below.\n\n### Readme notebook\n\n![readme](docs/readme-notebook.png)\n\nEvery installation creates a `README` notebook with a detailed description of all deployed workflows and their tasks,\nproviding quick links to the relevant workflows and dashboards.\n\n[[back to top](#databricks-labs-ucx)]\n\n### Debug notebook\n\n![debug](docs/debug-notebook.png)\n\nEvery installation creates a `DEBUG` notebook, that initializes UCX as a library for you to execute interactively.\n\n[[back to top](#databricks-labs-ucx)]\n\n### Debug logs\n\n![debug](docs/debug-logs.png)\n\nThe [workflow](#workflows) runs store debug logs in the `logs` folder of the installation folder. The logs are flushed\nevery minute in a separate file. Debug logs for [the command-line interface](#authenticate-databricks-cli) are shown\nby adding the `--debug` flag:\n\n```commandline\ndatabricks --debug labs ucx <command>\n```\n\n[[back to top](#databricks-labs-ucx)]\n\n### Installation configuration\n\nIn the installation folder, the UCX configuration is kept.\n\n## Advanced installation options\n\nAdvanced installation options are detailed below.\n\n### Force install over existing UCX\nUsing an environment variable `UCX_FORCE_INSTALL` you can force the installation of UCX over an existing installation.\nThe values for the environment variable are 'global' and 'user'.\n\nGlobal Install: When UCX is installed at '/Applications/ucx'\nUser Install: When UCX is installed at '/Users/<user>/.ucx'\n\nIf there is an existing global installation of UCX, you can force a user installation of UCX over the existing installation by setting the environment variable `UCX_FORCE_INSTALL` to 'global'.\n\nAt this moment there is no global override over a user installation of UCX. As this requires migration and can break existing installations.\n\n\n| global | user | expected install location | install_folder      | mode                                            |\n|--------|------|---------------------------|---------------------|-------------------------------------------------|\n| no     | no   | default                   | `/Applications/ucx` | install                                         |\n| yes    | no   | default                   | `/Applications/ucx` | upgrade                                         |\n| no     | yes  | default                   | `/Users/X/.ucx`     | upgrade (existing installations must not break) |\n| yes    | yes  | default                   | `/Users/X/.ucx`     | upgrade                                         |\n| yes    | no   | **USER**                  | `/Users/X/.ucx`     | install (show prompt)                           |\n| no     | yes  | **GLOBAL**                | ...                 | migrate                                         |\n\n\n* `UCX_FORCE_INSTALL=user databricks labs install ucx` - will force the installation to be for user only\n* `UCX_FORCE_INSTALL=global databricks labs install ucx` - will force the installation to be for root only\n\n\n[[back to top](#databricks-labs-ucx)]\n\n### Installing UCX on all workspaces within a Databricks account\nSetting the environment variable `UCX_FORCE_INSTALL` to 'account' will install UCX on all workspaces within a Databricks account.\n\n* `UCX_FORCE_INSTALL=account databricks labs install ucx`\n\nAfter the first installation, UCX will prompt the user to confirm whether to install UCX on the remaining workspaces with the same answers. If confirmed, the remaining installations will be completed silently.\n\nThis installation mode will automatically select the following options:\n* Automatically create and enable HMS lineage init script\n* Automatically create a new SQL warehouse for UCX assessment\n\n[[back to top](#databricks-labs-ucx)]\n\n### Installing UCX with company hosted PYPI mirror\n\nSome enterprise block the public PYPI index and host a company controlled PYPI mirror. To install UCX while using a\ncompany hosted PYPI mirror for finding its dependencies, add all UCX dependencies to the company hosted PYPI mirror (see\n\"dependencies\" in [`pyproject.toml`](./pyproject.toml)) and set the environment variable `PIP_INDEX_URL` to the company\nhosted PYPI mirror URL while installing UCX:\n\n```commandline\nPIP_INDEX_URL=\"https://url-to-company-hosted-pypi.internal\" databricks labs install ucx\n```\n\nDuring installation reply *yes* to the question \"Does the given workspace block Internet access\"?\n\n[[back to top](#databricks-labs-ucx)]\n\n## Upgrading UCX for newer versions\n\nVerify that UCX is installed\n\n```commandline\ndatabricks labs installed\n\nName  Description                            Version\nucx   Unity Catalog Migration Toolkit (UCX)  <version>\n```\n\nUpgrade UCX via Databricks CLI:\n\n```commandline\ndatabricks labs upgrade ucx\n```\n\nThe prompts will be similar to [Installation](#install-ucx)\n\n![macos_upgrade_ucx](docs/macos_3_databrickslabsmac_upgradeucx.gif)\n\n[[back to top](#databricks-labs-ucx)]\n\n## Uninstall UCX\n\nUninstall UCX via Databricks CLI:\n\n```commandline\ndatabricks labs uninstall ucx\n```\n\nDatabricks CLI will confirm a few options:\n- Whether you want to remove all ucx artefacts from the workspace as well. Defaults to no.\n- Whether you want to delete the inventory database in `hive_metastore`. Defaults to no.\n\n![macos_uninstall_ucx](docs/macos_4_databrickslabsmac_uninstallucx.gif)\n\n[[back to top](#databricks-labs-ucx)]\n\n# Migration process\n\nOn a high level, the steps in migration process are:\n1. [assessment](#assessment-workflow)\n2. [group migration](#group-migration-workflow)\n3. [table migration](#table-migration-process)\n4. [data reconciliation](#post-migration-data-reconciliation-workflow)\n5. [code migration](#code-migration-commands)\n\nThe migration process can be schematic visualized as:\n\n```mermaid\nflowchart TD\n    subgraph workspace-admin\n        assessment --> group-migration\n        group-migration --> table-migration\n        table-migration --> code-migration\n        assessment --> create-table-mapping\n        create-table-mapping --> table-migration\n        create-table-mapping --> code-migration\n        validate-external-locations --> table-migration\n        assessment --> validate-table-locations\n        validate-table-locations --> table-migration\n        table-migration --> revert-migrated-tables\n        revert-migrated-tables --> table-migration\n    end\n    subgraph account-admin\n        create-account-groups --> group-migration\n        sync-workspace-info --> create-table-mapping\n        group-migration --> validate-groups-membership\n    end\n    subgraph iam-admin\n        setup-account-scim --> create-account-groups\n        assessment --> create-uber-principal\n        create-uber-principal --> table-migration\n        assessment --> principal-prefix-access\n        principal-prefix-access --> migrate-credentials\n        migrate-credentials --> validate-external-locations\n        setup-account-scim\n    end\n```\n\n[[back to top](#databricks-labs-ucx)]\n\n## Table migration process\n\n> You are required to complete the [assessment workflow](#assessment-workflow) before starting the table migration workflow.\n\nThis section explains how to migrate Hive metastore data objects to Unity Catalog. The table migration process consists\nof more steps than only a [workflow](#table-migration-workflow), these steps are:\n\n1. [Table mapping](#table-mapping) : Create a file that maps Hive metastore data objects to Unity Catalog locations.\n2. [Data access](#data-access) : Setup identities to access table data.\n3. [New Unity Catalog resources](#new-unity-catalog-resources) : Create **new** Unity Catalog resources that do not\n   require touching existing Hive metastore resources.\n4. [Migrate Hive metastore data objects](#migrate-hive-metastore-data-objects) : Migrate Hive metastore data objects to UC.\n\n### Table mapping\n\nThis section details how to create the table mapping file. This file points existing Hive metastore tables and views to\nUnity Catalog locations. When [migrating the tables and views](#migrate-hive-metastore-data-objects), the file is read\nto decide where to migrate the tables and views to.\n\n#### Step 1 : Create the mapping file\n\nCreate the mapping file in the [UCX installation folder](#installation-resources) by running the\n[`create-table-mapping` command](#create-table-mapping-command). By default, the file contains all the Hive metastore\ntables and views mapped to a single UC catalog, while maintaining the original schema and table names.\n\n#### Step 2: Update the mapping file\n\nEdit the mapping file from the previous step to:\n- Exclude tables and/or views by removing the lines\n- Change UC location by editing the destination catalog and/or schema\n\nThe mapping file is in CSV format and can be edited using any text editor or Excel. If using Excel, save the file in CSV\nformat.\n\nExample changes:\n\n**Before editing**\n\n| **workspace_name**   | **catalog_name** | **src_schema** | **dst_schema** | **src_table** | **dst_table** |\n|----------------------|--------------|----------------|----------------|---------------|---------------|\n| data_engineering_ws  | 123333333   | sales_analysis      | sales_analysis      | ytd_sales        | ytd_sales        |\n\n**After editing**\n\n| **workspace_name**   | **catalog_name** | **src_schema** | **dst_schema** | **src_table** | **dst_table** |\n|----------------------|------------------|----------------|----------------|---------------|---------------|\n| data_engineering_ws  | data_engineering | sales_analysis      | sales      | ytd_sales        | ytd_analysis        |\n\n### Data access\n\n> Throughout this guide, we refer to IAM roles/instance profiles in AWS & service principals/managed identities in\n> as \"cloud principals\".\n\nThis section creates the cloud principals to access data in Unity Catalog and during the table data during migration. To\nunderstand the motivation for this step, read how Databricks accesses cloud locations:\n- [AWS - Create External Locations](https://docs.databricks.com/en/connect/unity-catalog/external-locations.html)\n- [Azure - Create External Locations](https://learn.microsoft.com/en-us/azure/databricks/connect/unity-catalog/external-locations)\n\n#### Step 1 : Map cloud principals to cloud storage locations\n\nMap the cloud principals to cloud storage by running the\n[`principal-prefix-access` command](#principal-prefix-access-command).\n\n#### Step 2 : Create or modify cloud principals and credentials\n\nManually create the cloud principals to access data from Unity Catalog:\n\n- AWS:\n  - [`create-missing-principals` command](#create-missing-principals-command-aws-only) creates new AWS roles for Unity\n    Catalog to access data.\n  - Or, [Manually create storage credentials](https://docs.databricks.com/en/connect/unity-catalog/storage-credentials.html)\n- Azure:\n  - [Manually create storage Credentials](https://learn.microsoft.com/en-us/azure/databricks/sql/language-manual/sql-ref-storage-credentials)\n\nThen, run the [`migrate-credentials` command](#migrate-credentials-command) to migrate the cloud principal\ncredentials to Unity Catalog.\n\n#### Step 3: Create the \"uber\" Principal\n\nCreate the \"uber\" principal by running the [`create-uber-principal` command](#create-uber-principal-command).\nThe table migration requires a cloud principal with access to table data stored in cloud storage. These tables are known\nas external tables. UCX name this cloud principal the \"uber\" principal as it has access to **all** in-scope cloud\nstorage. This principal is only required for the table migration process and not for ongoing UC usage. Once the upgrade\nis completed, this principal should be deleted.\n\n### New Unity Catalog resources\n\nThis section details the **new** Unity Catalog resources required for migration the data objects. These resources can be\ncreated without touching the existing Hive metastore objecs.\n\n#### Step 0: Attach a metastore\n\nIf skipping the [group migration](#group-migration-workflow), then a metastore should be attached to the workspace by\nfollowing [these instructions](https://docs.databricks.com/en/data-governance/unity-catalog/create-metastore.html) or running the [`assign-metastore` command](#assign-metastore-command).\n\n#### Step 1: Create external Locations\n\nCreate UC external locations by running the [`migration-locations` command](#migrate-locations-command). The command\ncreates a location for each location found during the [assessment](#assessment-workflow). It uses the credentials\ncreated in the previous steps.\n\nAlternatively, manually create the external locations\n- [AWS - Create External Locations](https://docs.databricks.com/en/connect/unity-catalog/external-locations.html)\n- [Azure - Create External Locations](https://learn.microsoft.com/en-us/azure/databricks/connect/unity-catalog/external-locations)\n\n##### Step 2: Create Catalogs and Schemas\n\nCreate Uniyt Catalog [catalogs](https://learn.microsoft.com/en-us/azure/databricks/catalogs/) and\n[schemas](https://learn.microsoft.com/en-us/azure/databricks/schemas/) to organize the destination tables and views in\nby running the\n[`create-catalogs-schemas` command](#create-catalogs-schemas-command). The command creates the UC catalogs and\nschemas based on the [table mapping file](#table-mapping). Additionally, it migrates Hive metastore database\npermissions if present.\n\nThis step requires considering how to [physically separate data in storage](https://docs.databricks.com/en/data-governance/unity-catalog/best-practices.html#data-is-physically-separated-in-storage)\nwithin UC. As [Databricks recommends storing managed data at the catalog level](https://docs.databricks.com/en/data-governance/unity-catalog/best-practices.html#configure-a-unity-catalog-metastore),\nwe advise to prepare the external locations for the to-be created catalogs before running the `create-catalogs-schemas`\ncommand. Either, reuse [previously created external locations](#step-23-create-external-locations) or create additional\nexternal locations outside of UCX if data separation restrictions requires that. Note that external locations can be\nreused when using subpaths, for example, a folder in a cloud storage\n(`abfss://container@storage.dfs.core.windows.net/folder`) can reuse the external location of the cloud storage\n(`abfss://container@storage.dfs.core.windows.net/`). (The previous example also holds for other clouds.)\n\n### Migrate Hive metastore data objects\n\nIn this step, tables and views are migrated using the following workflows:\n\n```mermaid\nflowchart LR\n    subgraph CLI\n        migrate-tables[migrate-tables]\n    end\n\n    subgraph mt_workflow[workflow: migrate-tables]\n        dbfs_root_delta_mt_task[migrate_dbfs_root_delta_tables]\n        migrate_dbfs_root_non_delta_tables[migrate_dbfs_root_non_delta_tables]\n        external_tables_sync_mt_task[migrate_external_tables_sync]\n        view_mt_task[migrate_views]\n        refresh_migration_status[refresh_migration_status]\n        external_tables_sync_mt_task --> view_mt_task\n        dbfs_root_delta_mt_task --> view_mt_task\n        migrate_dbfs_root_non_delta_tables --> view_mt_task\n        view_mt_task --> refresh_migration_status\n    end\n\n    subgraph mt_serde_inplace_wf[workflow: migrate-external-hiveserde-tables-in-place-experimental]\n        serde_inplace_mt_task[migrate_hive_serde_in_place] --> view_mt_task_inplace[migrate_views]\n        view_mt_task_inplace --> refresh_migration_status_hiveserde[refresh_migration_status]\n    end\n\n    subgraph mt_ctas_wf[workflow: migrate-external-tables-ctas]\n        ctas_mt_task[migrate_other_external_ctas]\n        migrate_hive_serde_ctas[migrate_hive_serde_ctas]\n        view_mt_ctas_task[migrate_views]\n        refresh_migration_status_ctas[refresh_migration_status]\n        ctas_mt_task --> view_mt_ctas_task\n        migrate_hive_serde_ctas --> view_mt_ctas_task\n        view_mt_ctas_task --> refresh_migration_status_ctas\n    end\n\n    migrate-tables -- 1st --> mt_workflow\n    migrate-tables -- 2nd (optional) table migrated here will be excluded in ctas workflow --> mt_serde_inplace_wf\n    migrate-tables -- 3rd --> mt_ctas_wf\n```\n\nThe [table migration workflows](#table-migration-workflow) can be triggered using the [`migrate-tables` command](#migrate-tables-command) or by\nstarting the workflows from the workspace UI manually. The table migration workflows are designed to minimize data\ncopying and to maintain metadata while allowing users to choose preferred strategies where possible. Chose the\nstrategies for migrating tables using the table below:\n\n| Object Type           | Description                                                                                            | Upgrade Method                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                         |\n|-----------------------|--------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|\n| `DBFS_ROOT_DELTA`     | Delta tables persisted on the Databricks file system (dbfs).                                           | Create a copy of the table with the [`DEEP CLONE` command](https://docs.databricks.com/en/sql/language-manual/delta-clone.html).                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                       |\n| `DBFS_ROOT_NON_DELTA` | Non-delta tables persisted on the Databricks file system (dbfs).                                       | Create a copy of the table with a [`CREATE TABLE AS SELECT * FROM` command](https://docs.databricks.com/en/sql/language-manual/sql-ref-syntax-ddl-create-table-using.html). The UC table will be a Delta table.                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                        |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                   |\n| `MANAGED`             | Managed Hive metastore tables.                                                                         | Depending on the managed table migration strategy chosen during installation:<br/>1. `CLONE`: Create a copy of the table with a [`CREATE TABLE LOCATION '<location>' AS SELECT * FROM` command](https://docs.databricks.com/en/sql/language-manual/sql-ref-syntax-ddl-create-table-using.html).<br/>2. `SYNC_AS_EXTERNAL`, synchronize the table metadata to UC with the [`SYNC` command](https://docs.databricks.com/en/sql/language-manual/sql-ref-syntax-aux-sync.html). **Warning**: If the managed Hive metastore table is dropped, the drop deletes the underlying data affecting the synchronised UC table as well.<br/>3. `CONVERT_TO_EXTERNAL`: First, in-place convert the managed Hive metastore to a non-managed table. Then, synchronize the table metadata to UC with the [`SYNC` command](https://docs.databricks.com/en/sql/language-manual/sql-ref-syntax-aux-sync.html). **Warning**: This strategy has the advantage over `SYNC_AS_EXTERNAL` that dropping the Hive metastore table does not delete the underlying data, however, impact should be carefully validated in existing workloads as the strategy converts the managed Hive metastore to an external table **in-place**. |\n| `EXTERNAL_SYNC`       | External tables that the `SYNC` command supports: Delta, Parquet, CSV, JSON, ORC, text or Avro tables. | Synchronize the table metadata to UC with the [`SYNC` command](https://docs.databricks.com/en/sql/language-manual/sql-ref-syntax-aux-sync.html).                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                       |\n| `EXTERNAL_NO_SYNC`    | External tables that the `SYNC` command does not support.                                              | Create a copy of the table with a [`CREATE TABLE AS SELECT * FROM` command](https://docs.databricks.com/en/sql/language-manual/sql-ref-syntax-ddl-create-table-using.html). The UC table will be a Delta table.                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                        |\n| `EXTERNAL_HIVESERDE`  | External Hive SerDe tables that the `SYNC` command does not support.                                   | Depending on the migration workflow chosen:<br/>1. `migrate-external-tables-ctas` (officially supported) : Create a copy of the table with a [`CREATE TABLE AS SELECT * FROM` command](https://docs.databricks.com/en/sql/language-manual/sql-ref-syntax-ddl-create-table-using.html) The UC table will be a Delta table.<br/>2. `migrate-external-hiveserde-tables-in-place-experimental` (experimental) : Recreate the [Hive SerDe](https://cwiki.apache.org/confluence/display/Hive/SerDe) tables using the serialization and deserialization protocols. **Warning:** Although this strategy is supported, there is a risk that the old files created by Hive SerDe may not be processed correctly by Spark datasource in corner cases.                                                                                                                                                                                                                                                                                                                                                                                                                                                             |\n| `VIEW`                | Views                                                                                                  | Recreate [views](https://docs.databricks.com/en/sql/language-manual/sql-ref-syntax-ddl-create-view.html) using their definitions after repointing their dependencies to UC objects. The migration process supports views depending on other views.                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                     |\n\nThe workflows may be executed multiple times. Each step is designed as a standalone operation that migrates all\nin-scope tables. After migration, each object is marked with an `upgraded_to` property containing the UC identifier to\nwhich the object is migrated. This property signals that the object is out-of-scope for other migration operations and\nthat the view dependency exists within UC. The created UC objects are marked with an `upgraded_from` property containing\nthe Hive metastore identifier from which the object was migrated.\n\nFinally, the table migration workflows also migrate Hive metastore permissions to Unity Catalog.\n\nConsiderations:\n- You may want to run the workflows multiple times to migrate tables in phases.\n- If your Delta tables in DBFS root have a large number of files, consider:\n  - Setting higher `Min` and `Max workers for auto-scale` when being asked during the UCX installation. More cores in the cluster means more concurrency for calling cloud storage API to copy files when deep cloning the Delta tables.\n  - Setting higher `Parallelism for migrating DBFS root Delta tables with deep clone` (default 200) when being asked during the UCX installation. This controls the number of Spark tasks/partitions to be created for deep clone.\n- Consider creating an instance pool, and setting its id when prompted during the UCX installation. This instance pool will be specified in the cluster policy used by all UCX workflows job clusters.\n- You may also manually edit the job cluster configration per job or per task after the workflows are deployed.\n\nAdditional references:\n- [Databricks file system (dbfs)](https://docs.databricks.com/en/dbfs/index.html)\n- [External tables](https://docs.databricks.com/en/tables/external.html)\n\n### Odds and Ends\n\nThe following sections detail how to repair/amend the UC metastore after the upgrade process.\n\n#### Skip migrating schemas, tables or views\n\n```bash\ndatabricks labs ucx skip --schema X [--table Y] [--view Zj]\n```\n\nThe [`skip` command](#skip-command) marks a schema, table or view as to-be skipped during the migration processes.\n\n####  Move data objects\n\n```bash\ndatabricks labs ucx move --from-catalog A --from-schema B --from-table C --to-catalog D --to-schema E\n```\n\nThe [`move` command](#move-command) moves the object from the source to the target location. The `upgraded_from`\nproperty are updated to reflect the new location on the source object. Use this command if the data object is migrated\nto the wrong destination.\n\n#### Alias data objects\n\n```bash\ndatabricks labs ucx alias --from-catalog A --from-schema B --from-table C --to-catalog D --to-schema E\n```\n\nThis [`alias` command](#alias-command) creates an alias for the object in the target location by creating a view reading\nfrom the table that needs aliasing. It will create a mirror view to view that is marked as alias.\nUse this command if Hive metastore tables point to the same location as UC does not support UC does not support tables\nwith overlapping data locations.\n\n#### Revert migrated data objects\n\n```bash\ndatabricks labs ucx revert-migrated-tables --schema X --table Y [--delete-managed]\n```\n\nThe [`revert-migrated-tables` command](#revert-migrated-tables-command) drops the Unity Catalog table or view and reset\nthe `upgraded_to` property on the source object. Use this command to allow for migrating a table or view again.\n\n[[back to top](#databricks-labs-ucx)]\n\n# Workflows\n\nPart of this application is deployed as [Databricks Workflows](https://docs.databricks.com/en/workflows/index.html).\nYou can view the status of deployed workflows via the [`workflows` command](#workflows-command).\nFailed workflows can be fixed with the [`repair-run` command](#repair-run-command).\n\n[[back to top](#databricks-labs-ucx)]\n\n## Assessment workflow\n\n![ucx_assessment_workflow](docs/ucx_assessment_workflow.png)\n\nThe assessment workflow can be triggered using the Databricks UI or via the\n[`ensure-assessment-run` command](#ensure-assessment-run-command).\n\nThe assessment workflow retrieves - or *crawls* - details of [workspace assets](https://docs.databricks.com/en/workspace/workspace-assets.html)\nand [securable objects in the Hive metastore](https://docs.databricks.com/en/data-governance/table-acls/object-privileges.html#securable-objects-in-the-hive-metastore)\nrelevant for upgrading to UC to assess the compatibility with UC. The `crawl_` tasks retrieve assess and objects. The\n`assess_` tasks assess the compatibility with UC. The output of each task is stored in the\n[inventory database](#installation-resources) so that it can be used for further analysis and decision-making through\nthe [assessment report](docs/assessment.md).\n\n1. `crawl_tables`: This task retrieves table definitions from the Hive metastore and persists the definitions in\n   the `tables` table. The definitions include information such as:\n   - Database/schema name\n   - Table name\n   - Table type\n   - Table location\n2. `crawl_udfs`: This task retrieves UDF definitions from the Hive metastore and persists the definitions in\n   the `udfs` table.\n3. `setup_tacl`: (Optimization) This task starts the `tacl` job cluster in parallel to other tasks.\n4. `crawl_grants`: This task retrieves [privileges you can grant on Hive objects](https://docs.databricks.com/en/data-governance/table-acls/object-privileges.html#privileges-you-can-grant-on-hive-metastore-objects)\n   and persists the privilege definitions in the `grants` table\n   The retrieved permission definitions include information such as:\n   - Securable object: schema, table, view, (anonymous) function or any file.\n   - Principal: user, service principal or group\n   - Action type: grant, revoke or deny\n5. `estimate_table_size_for_migration`: This task analyzes the Delta table retrieved by `crawl_tables` to retrieve\n   an estimate of their size and persists the table sizes in the `table_size` table. The table size support the decision\n   for using the `SYNC` or `CLONE` [table migration strategy](#step-3-upgrade-the-metastore).\n6. `crawl_mounts`: This task retrieves mount point definitions and persists the definitions in the `mounts` table.\n7. `guess_external_locations`: This task guesses shared mount path prefixes of [external tables](https://docs.databricks.com/en/sql/language-manual/sql-ref-external-tables.html)\n   retrieved by `crawl_tables` that use mount points and persists the locations in the `external_locations` table. The\n   goal is to identify the to-be created UC [external locations](#step-23-create-external-locations).\n8. `assess_jobs`: This task retrieves the job definitions and persists the definitions in the `jobs` table. Job\n   definitions may require updating to become UC compatible.\n9. `assess_clusters`: This task retrieves the clusters definitions and persists the definitions in the `clusters` table.\n    Cluster definitions may require updating to become UC compatible.\n10. `assess_pipelines`: This task retrieves the [Delta Live Tables](https://www.databricks.com/product/delta-live-tables)\n    (DLT) pipelines definitions and persists the definitions in the `pipelines` table. DLT definitions may require\n    updating to become UC compatible.\n11. `assess_incompatible_submit_runs` : This task retrieves\n    [job runs](https://docs.databricks.com/en/jobs/monitor.html#view-runs-for-a-job), also known as\n    [job submit runs](https://docs.databricks.com/api/workspace/jobs/submit), and persists the definitions in the\n    `submit_runs` table. Incompatibility with UC is assessed:\n    - [Databricks runtime](https://www.databricks.com/glossary/what-is-databricks-runtime) should be version 11.3 or above\n    - [Access mode](https://docs.databricks.com/en/compute/configure.html#access-modes) should be set.\n12. `crawl_cluster_policies` : This tasks retrieves\n    [cluster policies](https://docs.databricks.com/en/admin/clusters/policies.html) and persists the policies in the\n    `policies` table. Incompatibility with UC is assessed:\n    - [Databricks runtime](https://www.databricks.com/glossary/what-is-databricks-runtime) should be version 11.3 or above\n13. `assess_azure_service_principals`: This tasks retrieves Azure service principal\n    [authentications information](https://learn.microsoft.com/en-us/azure/databricks/connect/storage/tutorial-azure-storage#connect-adls)\n    from Spark configurations to access storage accounts and persists these configuration information\n    in `azure_service_principals` table. The Spark configurations from the following places are retrieved:\n    - Clusters configurations\n    - Cluster policies\n    - Job cluster configurations\n    - Pipeline configurations\n    - Warehouse configuration\n14. `assess_global_init_scripts`: This task retrieves\n    [global init scripts](https://docs.databricks.com/en/init-scripts/global.html) and persists references to the\n    scripts in the `global_init_scripts` table. Again, Azure service principal authentication information might be given\n    in those scripts.\n15. `workspace_listing` : This tasks lists [workspace files](https://docs.databricks.com/en/workspace/workspace-assets.html#files)\n    recursively to compile a collection of directories, notebooks, files, repos and libraries. The task uses multi-threading\n    to parallelize the listing process for speeding up execution on big workspaces.\n16. `crawl_permissions` : This tasks retrieves workspace-local groups permissions and persists these permissions in the `permissions` table.\n17. `crawl_groups`: This tasks retrieves workspace-local groups and persists these permissions in the `groups` table.\n18. `assess_dashboards`: This task retrieves the dashboards to analyze their queries for\n    [migration problems](#linter-message-codes)\n19. `assess_workflows`: This task retrieves the jobs to analyze their notebooks and files for\n    [migration problems](#linter-message-codes).\n\nAfter UCX assessment workflow finished, see the assessment dashboard for findings and recommendations.\nSee [this guide](docs/assessment.md) for more details.\n\n![report](docs/assessment-report.png)\n\nProceed to the [group migration workflow](#group-migration-workflow) below or go back to the\n[migration process diagram](#migration-process).\n\nThe UCX assessment workflow is designed to only run once, re-running will **not** update the existing results. If the\ninventory and findings for a workspace need to be updated then first reinstall UCX by [uninstalling](#uninstall-ucx)\nand [installing](#install-ucx) it again.\n\n[[back to top](#databricks-labs-ucx)]\n\n## Group migration workflow\n\n> You are required to complete the [assessment workflow](#assessment-workflow) before starting the group migration workflow.\n\n**The group migration workflow does NOT CREATE account groups.** In contrast to account groups, [the (legacy)\nworkspace-local groups cannot be assigned to additional workspaces or granted access to data in a Unity Catalog\nmetastore](https://docs.databricks.com/en/admin/users-groups/groups.html#difference-between-account-groups-and-workspace-local-groups).\nA Databricks admin [assigns account groups to workspaces](https://docs.databricks.com/en/admin/users-groups/index.html#assign-users-to-workspaces)\nusing [identity federation](https://docs.databricks.com/en/admin/users-groups/index.html#enable-identity-federation)\nto manage groups from a single place: your Databricks account. We expect UCX users to create account groups\ncentrally while most other Databricks resources that UCX touches are scoped to a single workspace.\nIf you do not have account groups matching the workspace in which UCX is installed, please\nrun [`create-account-groups` command](#create-account-groups-command) before running the group migration workflow.\n\nThe group migration workflow is designed to migrate workspace-local groups to account-level groups. It verifies if\nthe necessary groups are available to the workspace with the correct permissions, and removes unnecessary groups and\npermissions. The group migration workflow depends on the output of the assessment workflow, thus, should only be\nexecuted after a successful run of the assessment workflow. The group migration workflow may be executed multiple times.\n\n1. `verify_metastore_attached`: Verifies if a metastore is attached. Account level groups are only available when\n    a metastore is attached. [See `assign-metastore` command.](#assign-metastore-command)\n2. `rename_workspace_local_groups`: This task renames workspace-local groups by adding a `ucx-renamed-` prefix. This\n   step is taken to avoid conflicts with account groups that may have the same name as workspace-local groups.\n3. `reflect_account_groups_on_workspace`: This task adds matching account groups to this workspace. The matching account\n   groups must exist for this step to be successful. This step is necessary to ensure that the account groups\n   are available in the workspace for assigning permissions.\n4. `apply_permissions_to_account_groups`: This task assigns the full set of permissions of the original group to the\n   account-level one. This step is necessary to ensure that the account-level groups have the necessary permissions to\n   manage the entities in the workspace. It covers workspace-local permissions for all entities including:\n   - Legacy Table ACLs\n   - Entitlements\n   - AWS instance profiles\n   - Clusters\n   - Cluster policies\n   - Instance Pools\n   - Databricks SQL warehouses\n   - Delta Live Tables\n   - Jobs\n   - MLflow experiments\n   - MLflow registry\n   - SQL Dashboards & Queries\n   - SQL Alerts\n   - Token and Password usage permissions\n   - Secret Scopes\n   - Notebooks\n   - Directories\n   - Repos\n   - Files\n5. `validate_groups_permissions`: This task validates that all the crawled permissions are applied correctly to the\n   destination groups.\n\nAfter successfully running the group migration workflow:\n1. Use [`validate-groups-membership` command](#validate-groups-membership-command) for extra confidence the newly created\n   account level groups are considered to be valid.\n2. Run the [`remove-workspace-local-backup-grups`](#validate-groups-membership-command) to remove workspace-level backup\n   groups, along with their permissions. This should only be executed after confirming that the workspace-local\n   migration worked successfully for all the groups involved. This step is necessary to clean up the workspace and\n   remove any unnecessary groups and permissions.\n3. Proceed to the [table migration process](#Table-Migration).\n\nFor additional information see:\n- The [detailed design](docs/local-group-migration.md) of thie group migration workflow.\n- The [migration process diagram](#migration-process) showing the group migration workflow in context of the whole\n  migration process.\n\n[[back to top](#databricks-labs-ucx)]\n\n## Table migration workflows\n\nThis section lists the workflows that migrate tables and views. See [this section](#migrate-hive-metastore-data-objects)\nfor deciding which workflow to run and additional context for migrating tables.\n\n### Migrate tables\n\nThe general table migration workflow `migrate-tables` migrates all tables and views using default strategies.\n\n1. `migrate_external_tables_sync` : This step migrates the external tables that are supported by `SYNC` command.\n2. `migrate_dbfs_root_delta_tables` : This step migrates delta tables stored in DBFS root using the `DEEP CLONE`\n   command.\n3. `migrate_dbfs_root_non_delta_tables` : This step migrates non-delta tables stored in DBFS root using the `CREATE\n   TABLE AS SELECT * FROM` command.\n4. `migrate_views` : This step migrates views using the `CREATE VIEW` command.\n5. `update_migration_status` : Refresh the migration status of all data objects.\n\n### Migrate external Hive SerDe tables\n\nThe experimental table migration workflow `migrate-external-hiveserde-tables-in-place-experimental` migrates tables that\nsupport the `SYNC AS EXTERNAL` command.\n\n1. `migrate_hive_serde_in_place` : This step migrates the Hive SerDe tables that are supported by `SYNC AS EXTERNAL`\n   command.\n2. `migrate_views` : This step migrates views using the `CREATE VIEW` command.\n3. `update_migration_status` : Refresh the migration status of all data objects.\n\n### Migrate external tables CTAS\n\nThe table migration workflow `migrate-external-tables-ctas` migrates tables with the `CREATE TABLE AS SELECT * FROM`\ncommand.\n\n1. `migrate_other_external_ctas` This step migrates the Hive Serde tables using the `CREATE TABLE AS SELECT * FROM`\n   command.\n2. `migrate_hive_serde_ctas` : This step migrates the Hive Serde tables using the `CREATE TABLE AS SELECT * FROM`\n   command.\n4. `migrate_views` : This step migrates views using the `CREATE VIEW` command.\n4. `update_migration_status` : Refresh the migration status of all data objects.\n\n## Post-migration data reconciliation workflow\n\nThe `migrate-data-reconciliation` workflow validates the integrity of the migrated tables and persists its results in\nthe `recon_results` table. The workflow compares the following between the migrated Hive metastore and its UC\ncounterpart table:\n- `Schema` : See this result in the `schema_matches` column.\n- `Column by column` : See this result in the `column_comparison` column.\n- `Row counts` : If the row count is within the reconciliation threshold (defaults to 5%), the `data_matches` column is\n  set to `true`, otherwise it is set to `false`.\n- `Rows` : If the `compare_rows` flag is set to `true`, rows are compared using a hash comparison. Number of missing\n  rows are stored in the `source_missing_count` and `target_missing_count` column, respectively.\n\nThe output is processed and displayed in the migration dashboard using the in `reconciliation_results` view.\n\n![reconciliation results](docs/recon_results.png)\n\n### [LEGACY] Scan tables in mounts Workflow\n#### <b>Always run this workflow AFTER the assessment has finished</b>\n- This experimental workflow attempts to find all Tables inside mount points that are present on your workspace.\n- If you do not run this workflow, then `migrate-tables-in-mounts-experimental` won't do anything.\n- It writes all results to `hive_metastore.<inventory_database>.tables`, you can query those tables found by filtering on database values that starts with `mounted_`\n- This command is incremental, meaning that each time you run it, it will overwrite the previous tables in mounts found.\n- Current format are supported:\n  - DELTA - PARQUET - CSV - JSON\n  - Also detects partitioned DELTA and PARQUET\n- You can configure these workflows with the following options available on conf.yml:\n  - include_mounts : A list of mount points to scans, by default the workflow scans for all mount points\n  - exclude_paths_in_mount : A list of paths to exclude in all mount points\n  - include_paths_in_mount : A list of paths to include in all mount points\n\n### [LEGACY] Migrate tables in mounts Workflow\n- An experimental workflow that migrates tables in mount points using a `CREATE TABLE` command, optinally sets a default tables owner if provided in `default_table_owner` conf parameter.\n- You must do the following in order to make this work:\n  - run the Assessment [workflow](#assessment-workflow)\n  - run the scan tables in mounts [workflow](#EXPERIMENTAL-scan-tables-in-mounts-workflow)\n  - run the [`create-table-mapping` command](#create-table-mapping-command)\n    - or manually create a `mapping.csv` file in Workspace -> Applications -> ucx\n\n\n[[back to top](#databricks-labs-ucx)]\n\n## [EXPERIMENTAL] Migration Progress Workflow\n\nThe manually triggered `migration-progress-experimental` workflow populates the tables visualized in the\n[migration progress dashboard](#dashboards) by updating a **subset** of the [inventory tables](#assessment-workflow)\nto [track Unity Catalog compatability](docs/migration-progress.md) of Hive and workspace objects that need to be migrated.\n\nThe following pre-requisites need to be fulfilled before running the workflow:\n- [UC metastore attached to workspace](../README.md#assign-metastore-command)\n- [UCX catalog exists](../README.md#create-ucx-catalog-command)\n- [Assessment job ran successfully](../README.md#ensure-assessment-run-command)\n\n[[back to top](#databricks-labs-ucx)]\n\n# Dashboards\n\nThe dashboards visualize UCX's outcomes. Each dashboard has text widgets detailing the contents in more detail. Below an\noverview with a short description is given.\n\n| Name                                                                                                            | Description                                        |\n|-----------------------------------------------------------------------------------------------------------------|----------------------------------------------------|\n| [Assessment \\[Main\\]](./src/databricks/labs/ucx/queries/assessment/main/00_0__assessment_overview.md)           | Assessment overview                                |\n| [Assessment \\[Estimates\\]](./src/databricks/labs/ucx/queries/assessment/estimates/00_0_metastore_assignment.md) | Migration effort estimates based on the assessment |\n| [Assessment \\[Interactive\\]](./src/databricks/labs/ucx/queries/assessment/interactive/00_0_interactive.md)      | Assessment outcomes on interactive clusters        |\n| [Assessment \\[Azure\\]](./src/databricks/labs/ucx/queries/assessment/azure/00_0_azure_service_principals.md)     | Assessment outcomes specific to Azure              |\n| [Migration \\[Main\\]](./src/databricks/labs/ucx/queries/migration/main/00_0_migration_overview.md)               | Migration overview                                 |\n| [Migration \\[Groups\\]](./src/databricks/labs/ucx/queries/migration/groups/00_0_migration_overview.md)           | Group migration outcomes                           |\n| [Progress \\[Main\\]](./src/databricks/labs/ucx/queries/progress/main/00_0_migration_progress.md)                 | [Migration progress](./docs/migration-progress.md) |\n\n[[back to top](#databricks-labs-ucx)]\n\n# Linter message codes\n\n![code compatibility problems](docs/code_compatibility_problems.png)\n\nHere's the detailed explanation of the linter message codes:\n\n## `cannot-autofix-table-reference`\n\nThis indicates that the linter has found a table reference that cannot be automatically fixed. The user must manually\nupdate the table reference to point to the correct table in Unity Catalog. This mostly occurs, when table name is\ncomputed dynamically, and it's too complex for our static code analysis to detect it. We detect this problem anywhere\nwhere table name could be used: `spark.sql`, `spark.catalog.*`, `spark.table`, `df.write.*` and many more. Code examples\nthat trigger this problem:\n\n```python\nspark.table(f\"foo_{some_table_name}\")\n# ..\ndf = spark.range(10)\ndf.write.saveAsTable(f\"foo_{some_table_name}\")\n# .. or even\ndf.write.insertInto(f\"foo_{some_table_name}\")\n```\n\nHere the `some_table_name` variable is not defined anywhere in the visible scope. Though, the analyser would\nsuccessfully detect table name if it is defined:\n\n```python\nsome_table_name = 'bar'\nspark.table(f\"foo_{some_table_name}\")\n```\n\nWe even detect string constants when coming either from `dbutils.widgets.get` (via job named parameters) or through\nloop variables. If `old.things` table is migrated to `brand.new.stuff` in Unity Catalog, the following code will\ntrigger two messages: [`table-migrated-to-uc`](#table-migrated-to-uc) for the first query, as the contents are clearly\nanalysable, and `cannot-autofix-table-reference` for the second query.\n\n```python\n# ucx[table-migrated-to-uc:+4:4:+4:20] Table old.things is migrated to brand.new.stuff in Unity Catalog\n# ucx[cannot-autofix-table-reference:+3:4:+3:20] Can't migrate table_name argument in 'spark.sql(query)' because its value cannot be computed\ntable_name = f\"table_{index}\"\nfor query in [\"SELECT * FROM old.things\", f\"SELECT * FROM {table_name}\"]:\n    spark.sql(query).collect()\n```\n\n[[back to top](#databricks-labs-ucx)]\n\n## `catalog-api-in-shared-clusters`\n\n`spark.catalog.*` functions require Databricks Runtime 14.3 LTS or above on Unity Catalog clusters in Shared access\nmode, so of your code has `spark.catalog.tableExists(\"table\")` or `spark.catalog.listDatabases()`, you need to ensure\nthat your cluster is running the correct runtime version and data security mode.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `changed-result-format-in-uc`\n\nCalls to these functions would return a list of `<catalog>.<database>.<table>` instead of `<database>.<table>`. So if\nyou have code like this:\n\n```python\nfor table in spark.catalog.listTables():\n    do_stuff_with_table(table)\n```\n\nyou need to make sure that `do_stuff_with_table` can handle the new format.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `direct-filesystem-access-in-sql-query`\n\nDirect filesystem access is deprecated in Unity Catalog.\nDBFS is no longer supported, so if you have code like this:\n\n```python\ndf = spark.sql(\"SELECT * FROM parquet.`/mnt/foo/path/to/parquet.file`\")\n```\n\nyou need to change it to use UC tables.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `direct-filesystem-access`\n\nDirect filesystem access is deprecated in Unity Catalog.\nDBFS is no longer supported, so if you have code like this:\n\n```python\ndisplay(spark.read.csv('/mnt/things/data.csv'))\n```\n\nor this:\n\n```python\ndisplay(spark.read.csv('s3://bucket/folder/data.csv'))\n```\n\nYou need to change it to use UC tables or UC volumes.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `dependency-not-found`\n\nThis message indicates that the linter has found a dependency, like Python source file or a notebook, that is not\navailable in the workspace. The user must ensure that the dependency is available in the workspace. This usually\nmeans an error in the user code.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `jvm-access-in-shared-clusters`\n\nYou cannot access Spark Driver JVM on Unity Catalog clusters in Shared Access mode. If you have code like this:\n\n```python\nspark._jspark._jvm.com.my.custom.Name()\n```\n\nor like this:\n\n```python\nlog4jLogger = sc._jvm.org.apache.log4j\nLOGGER = log4jLogger.LogManager.getLogger(__name__)\n```\n\nyou need to change it to use Python equivalents.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `legacy-context-in-shared-clusters`\n\nSparkContext (`sc`) is not supported on Unity Catalog clusters in Shared access mode. Rewrite it using SparkSession\n(`spark`). Example code that triggers this message:\n\n```python\ndf = spark.createDataFrame(sc.emptyRDD(), schema)\n```\n\nor this:\n\n```python\nsc.parallelize([1, 2, 3])\n```\n\n[[back to top](#databricks-labs-ucx)]\n\n## `not-supported`\n\nInstalling eggs is no longer supported on Databricks 14.0 or higher.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `notebook-run-cannot-compute-value`\n\nPath for `dbutils.notebook.run` cannot be computed and requires adjusting the notebook path.\nIt is not clear for automated code analysis where the notebook is located, so you need to simplify the code like:\n\n```python\nb = some_function()\ndbutils.notebook.run(b)\n```\n\nto something like this:\n\n```python\na = \"./leaf1.py\"\ndbutils.notebook.run(a)\n```\n\n[[back to top](#databricks-labs-ucx)]\n\n## `python-udf-in-shared-clusters`\n\n`applyInPandas` requires DBR 14.3 LTS or above on Unity Catalog clusters in Shared access mode. Example:\n\n```python\ndf.groupby(\"id\").applyInPandas(subtract_mean, schema=\"id long, v double\").show()\n```\n\nArrow UDFs require DBR 14.3 LTS or above on Unity Catalog clusters in Shared access mode.\n\n```python\n@udf(returnType='int', useArrow=True)\ndef arrow_slen(s):\n    return len(s)\n```\n\nIt is not possible to register Java UDF from Python code on Unity Catalog clusters in Shared access mode. Use a\n`%scala` cell to register the Scala UDF using `spark.udf.register`. Example code that triggers this message:\n\n```python\nspark.udf.registerJavaFunction(\"func\", \"org.example.func\", IntegerType())\n```\n\n[[back to top](#databricks-labs-ucx)]\n\n## `rdd-in-shared-clusters`\n\nRDD APIs are not supported on Unity Catalog clusters in Shared access mode. Use mapInArrow() or Pandas UDFs instead.\n\n```python\ndf.rdd.mapPartitions(myUdf)\n```\n\n[[back to top](#databricks-labs-ucx)]\n\n## `spark-logging-in-shared-clusters`\n\nCannot set Spark log level directly from code on Unity Catalog clusters in Shared access mode. Remove the call and set\nthe cluster spark conf `spark.log.level` instead:\n\n```python\nsc.setLogLevel(\"INFO\")\nsetLogLevel(\"WARN\")\n```\n\nAnother example could be:\n\n```python\nlog4jLogger = sc._jvm.org.apache.log4j\nLOGGER = log4jLogger.LogManager.getLogger(__name__)\n```\n\nor\n\n```python\nsc._jvm.org.apache.log4j.LogManager.getLogger(__name__).info(\"test\")\n```\n\n[[back to top](#databricks-labs-ucx)]\n\n## `sql-parse-error`\n\nThis is a generic message indicating that the SQL query could not be parsed. The user must manually check the SQL query.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `sys-path-cannot-compute-value`\n\nPath for `sys.path.append` cannot be computed and requires adjusting the path. It is not clear for automated code\nanalysis where the path is located.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `table-migrated-to-uc`\n\nThis message indicates that the linter has found a table that has been migrated to Unity Catalog. The user must ensure\nthat the table is available in Unity Catalog.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `to-json-in-shared-clusters`\n\n`toJson()` is not available on Unity Catalog clusters in Shared access mode. Use `toSafeJson()` on DBR 13.3 LTS or\nabove to get a subset of command context information. Example code that triggers this message:\n\n```python\ndbutils.notebook.entry_point.getDbutils().notebook().getContext().toSafeJson()\n```\n\n[[back to top](#databricks-labs-ucx)]\n\n## `unsupported-magic-line`\n\nThis message indicates the code that could not be analysed by UCX. User must check the code manually.\n\n[[back to top](#databricks-labs-ucx)]\n\n# Utility commands\n\n## `logs` command\n\n```text\n$ databricks labs ucx logs [--workflow WORKFLOW_NAME] [--debug]\n```\n\nThis command displays the logs of the last run of the specified workflow. If no workflow is specified, it displays\nthe logs of the workflow that was run the last. This command is useful for developers and administrators who want to\ncheck the logs of the last run of a workflow and ensure that it was executed as expected. It can also be used for\ndebugging purposes when a workflow is not behaving as expected. By default, only `INFO`, `WARNING`, and `ERROR` logs\nare displayed. To display `DEBUG` logs, use the `--debug` flag.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `ensure-assessment-run` command\n\n```commandline\ndatabricks labs ucx ensure-assessment-run\n```\n\nThis command ensures that the [assessment workflow](#assessment-workflow) was run on a workspace.\nThis command will block until job finishes.\nFailed workflows can be fixed with the [`repair-run` command](#repair-run-command). Workflows and their status can be\nlisted with the [`workflows` command](#workflows-command).\n\n[[back to top](#databricks-labs-ucx)]\n\n## `update-migration-progress` command\n\n```commandline\ndatabricks labs ucx update-migration-progress\n```\n\nThis command runs the [(experimental) migration progress workflow](#experimental-migration-progress-workflow) to update\nthe migration status of workspace resources that need to be migrated. It does this by triggering\nthe `migration-progress-experimental` workflow to run on a workspace and waiting for\nit to complete.\n\nWorkflows and their status can be listed with the [`workflows` command](#workflows-commandr), while failed workflows can\nbe fixed with the [`repair-run` command](#repair-run-command).\n\n[[back to top](#databricks-labs-ucx)]\n\n## `repair-run` command\n\n```commandline\ndatabricks labs ucx repair-run --step WORKFLOW_NAME\n```\n\nThis command repairs a failed [UCX Workflow](#workflows). This command is useful for developers and administrators who\nwant to repair a failed job. It can also be used to debug issues related to job failures. This operation can also be\ndone via [user interface](https://docs.databricks.com/en/workflows/jobs/repair-job-failures.html). Workflows and their\nstatus can be listed with the [`workflows` command](#workflows-command).\n\n[[back to top](#databricks-labs-ucx)]\n\n## `workflows` command\n\nSee the [migration process diagram](#migration-process) to understand the role of each workflow in the migration process.\n\n```text\n$ databricks labs ucx workflows\nStep                                  State    Started\nassessment                            RUNNING  1 hour 2 minutes ago\n099-destroy-schema                    UNKNOWN  <never run>\nmigrate-groups                        UNKNOWN  <never run>\nremove-workspace-local-backup-groups  UNKNOWN  <never run>\nvalidate-groups-permissions           UNKNOWN  <never run>\n```\n\nThis command displays the [deployed workflows](#workflows) and their state in the current workspace. It fetches the latest\njob status from the workspace and prints it in a table format. This command is useful for developers and administrators\nwho want to check the status of UCX workflows and ensure that they have been executed as expected. It can also be used\nfor debugging purposes when a workflow is not behaving as expected. Failed workflows can be fixed with\nthe [`repair-run` command](#repair-run-command).\n\n[[back to top](#databricks-labs-ucx)]\n\n## `open-remote-config` command\n\n```commandline\ndatabricks labs ucx open-remote-config\n```\n\nThis command opens the remote configuration file in the default web browser. It generates a link to the configuration file\nand opens it using the `webbrowser.open()` method. This command is useful for developers and administrators who want to view or\nedit the remote configuration file without having to manually navigate to it in the workspace. It can also be used to quickly\naccess the configuration file from the command line. Here's the description of configuration properties:\n\n  * `inventory_database`: A string representing the name of the inventory database.\n  * `workspace_group_regex`: An optional string representing the regular expression to match workspace group names.\n  * `workspace_group_replace`: An optional string to replace the matched group names with.\n  * `account_group_regex`: An optional string representing the regular expression to match account group names.\n  * `group_match_by_external_id`: A boolean value indicating whether to match groups by their external IDs.\n  * `include_group_names`: An optional list of strings representing the names of groups to include for migration.\n  * `renamed_group_prefix`: An optional string representing the prefix to add to renamed group names.\n  * `instance_pool_id`: An optional string representing the ID of the instance pool.\n  * `warehouse_id`: An optional string representing the ID of the warehouse.\n  * `connect`: An optional `Config` object representing the configuration for connecting to the warehouse.\n  * `num_threads`: An optional integer representing the number of threads to use for migration.\n  * `database_to_catalog_mapping`: An optional dictionary mapping source database names to target catalog names.\n  * `default_catalog`: An optional string representing the default catalog name.\n  * `skip_tacl_migration`: Optional flag, allow skipping TACL migration when migrating tables or creating catalogs and schemas.\n  * `default_owner_group`: Assigns this group to all migrated objects (catalogs, databases, tables, views, etc.). The group has to be an account group and the user running the migration has to be a member of this group.\n  * `log_level`: An optional string representing the log level.\n  * `workspace_start_path`: A string representing the starting path for notebooks and directories crawler in the workspace.\n  * `instance_profile`: An optional string representing the name of the instance profile.\n  * `spark_conf`: An optional dictionary of Spark configuration properties.\n  * `override_clusters`: An optional dictionary mapping job cluster names to existing cluster IDs.\n  * `policy_id`: An optional string representing the ID of the cluster policy.\n  * `include_databases`: An optional list of strings representing the names of databases to include for migration.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `installations` command\n\n```text\n$ databricks labs ucx installations\n...\n13:49:16  INFO [d.labs.ucx] Fetching installations...\n13:49:17  INFO [d.l.blueprint.parallel][finding_ucx_installations_5] finding ucx installations 10/88, rps: 22.838/sec\n13:49:17  INFO [d.l.blueprint.parallel][finding_ucx_installations_9] finding ucx installations 20/88, rps: 35.002/sec\n13:49:17  INFO [d.l.blueprint.parallel][finding_ucx_installations_2] finding ucx installations 30/88, rps: 51.556/sec\n13:49:18  INFO [d.l.blueprint.parallel][finding_ucx_installations_9] finding ucx installations 40/88, rps: 56.272/sec\n13:49:18  INFO [d.l.blueprint.parallel][finding_ucx_installations_19] finding ucx installations 50/88, rps: 67.382/sec\n...\nPath                                      Database  Warehouse\n/Users/serge.smertin@databricks.com/.ucx  ucx       675eaf1ff976aa98\n```\n\nThis command displays the [installations](#installation) by different users on the same workspace. It fetches all\nthe installations where the `ucx` package is installed and prints their details in JSON format. This command is useful\nfor administrators who want to see which users have installed `ucx` and where. It can also be used to debug issues\nrelated to multiple installations of `ucx` on the same workspace.\n\n[[back to top](#databricks-labs-ucx)]\n\n\n## `report-account-compatibility` command\n\n```text\ndatabricks labs ucx report-account-compatibility --profile labs-azure-account\n12:56:09  INFO [databricks.sdk] Using Azure CLI authentication with AAD tokens\n12:56:09  INFO [d.l.u.account.aggregate] Generating readiness report\n12:56:10  INFO [databricks.sdk] Using Azure CLI authentication with AAD tokens\n12:56:10  INFO [databricks.sdk] Using Azure CLI authentication with AAD tokens\n12:56:15  INFO [databricks.sdk] Using Azure CLI authentication with AAD tokens\n12:56:15  INFO [d.l.u.account.aggregate] Querying Schema ucx\n12:56:21  WARN [d.l.u.account.aggregate] Workspace 4045495039142306 does not have UCX installed\n12:56:21  INFO [d.l.u.account.aggregate] UC compatibility: 30.303030303030297% (69/99)\n12:56:21  INFO [d.l.u.account.aggregate] cluster type not supported : LEGACY_TABLE_ACL: 22 objects\n12:56:21  INFO [d.l.u.account.aggregate] cluster type not supported : LEGACY_SINGLE_USER: 24 objects\n12:56:21  INFO [d.l.u.account.aggregate] unsupported config: spark.hadoop.javax.jdo.option.ConnectionURL: 10 objects\n12:56:21  INFO [d.l.u.account.aggregate] Uses azure service principal credentials config in cluster.: 1 objects\n12:56:21  INFO [d.l.u.account.aggregate] No isolation shared clusters not supported in UC: 1 objects\n12:56:21  INFO [d.l.u.account.aggregate] Data is in DBFS Root: 23 objects\n12:56:21  INFO [d.l.u.account.aggregate] Non-DELTA format: UNKNOWN: 5 objects\n```\n\n[[back to top](#databricks-labs-ucx)]\n## `export-assessment` command\n\n```commandline\ndatabricks labs ucx export-assessment\n```\nThe export-assessment command is used to export UCX assessment results to a specified location. When you run this command, you will be prompted to provide details on the destination path and the type of report you wish to generate. If you do not specify these details, the command will default to exporting the main results to the current directory. The exported file will be named based on the selection made in the format. Eg: export_{query_choice}_results.zip\n- **Choose a path to save the UCX Assessment results:**\n    - **Description:** Specify the path where the results should be saved. If not provided, results will be saved in the current directory.\n\n- **Choose which assessment results to export:**\n    - **Description:** Select the type of results to export. Options include:\n        - `azure`\n        - `estimates`\n        - `interactive`\n        - `main`\n    - **Default:** `main`\n\n[[back to top](#databricks-labs-ucx)]\n\n# Metastore related commands\n\nThese commands are used to assign a Unity Catalog metastore to a workspace. The metastore assignment is a pre-requisite\nfor any further migration steps.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `show-all-metastores` command\n\n```text\ndatabricks labs ucx show-all-metastores [--workspace-id <workspace-id>]\n```\n\nThis command lists all the metastores available to be assigned to a workspace. If no workspace is specified, it lists\nall the metastores available in the account. This command is useful when there are multiple metastores available within\na region, and you want to see which ones are available for assignment.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `assign-metastore` command\n\n```text\ndatabricks labs ucx assign-metastore --workspace-id <workspace-id> [--metastore-id <metastore-id>]\n```\n\nThis command assigns a metastore to a workspace with `--workspace-id`. If there is only a single metastore in the\nworkspace region, the command automatically assigns that metastore to the workspace. If there are multiple metastores\navailable, the command prompts for specification of the metastore (id) you want to assign to the workspace.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `create-ucx-catalog` command\n\n```commandline\ndatabricks labs ucx create-ucx-catalog\n16:12:59  INFO [d.l.u.hive_metastore.catalog_schema] Validating UC catalog: ucx\nPlease provide storage location url for catalog: ucx (default: metastore): ...\n16:13:01  INFO [d.l.u.hive_metastore.catalog_schema] Creating UC catalog: ucx\n```\n\nCreate and setup UCX artifact catalog. Amongst other things, the artifacts are used for tracking the migration progress\nacross workspaces.\n\n# Table migration commands\n\nThese commands are vital part of [table migration process](#Table-Migration) process and require\nthe [assessment workflow](#assessment-workflow) and\n[group migration workflow](#group-migration-workflow) to be completed.\nSee the [migration process diagram](#migration-process) to understand the role of the table migration commands in\nthe migration process.\n\nThe first step is to run the [`principal-prefix-access` command](#principal-prefix-access-command) to identify all\nthe storage accounts used by tables in the workspace and their permissions on each storage account.\n\nIf you don't have any storage credentials and external locations configured,  you'll need to run\nthe [`migrate-credentials` command](#migrate-credentials-command) to migrate the service principals\nand [`migrate-locations` command](#migrate-locations-command) to create the external locations.\nIf some of the external locations already exist, you should run\nthe [`validate-external-locations` command](#validate-external-locations-command).\nYou'll need to create the [uber principal](#create-uber-principal-command) with the _**access to all storage**_ used to tables in\nthe workspace, so that you can migrate all the tables. If you already have the principal, you can skip this step.\n\nAsk your Databricks Account admin to run the [`sync-workspace-info` command](#sync-workspace-info-command) to sync the\nworkspace information with the UCX installations. Once the workspace information is synced, you can run the\n[`create-table-mapping` command](#create-table-mapping-command) to align your tables with the Unity Catalog,\n[create catalogs and schemas](#create-catalogs-schemas-command) and start the migration using [`migrate-tables` command](#migrate-tables-command). During multiple runs of\nthe table migration workflow, you can use the [`revert-migrated-tables` command](#revert-migrated-tables-command) to\nrevert the tables that were migrated in the previous run. You can also skip the tables that you don't want to migrate\nusing the [`skip` command](#skip-command).\n\nOnce you're done with the table migration, proceed to the [code migration](#code-migration-commands).\n\n[[back to top](#databricks-labs-ucx)]\n\n## `principal-prefix-access` command\n\n```text\ndatabricks labs ucx principal-prefix-access [--subscription-ids <Azure Subscription ID>] [--aws-profile <AWS CLI profile>]\n```\n\nThis command depends on results from the [assessment workflow](#assessment-workflow) and requires [AWS CLI](#access-for-aws-s3-buckets)\nor [Azure CLI](#access-for-azure-storage-accounts) to be installed and authenticated for the given machine. This command\nidentifies all the storage accounts used by tables in the workspace and their permissions on each storage account.\nOnce you're done running this command, proceed to the [`migrate-credentials` command](#migrate-credentials-command).\n\nThe \"prefix\" refers to the start - i.e. prefix - of table locations that point to the cloud storage location.\n\n[[back to top](#databricks-labs-ucx)]\n\n### Access for AWS S3 Buckets\n\n```commandline\ndatabricks labs ucx principal-prefix-access --aws-profile test-profile\n```\n\nUse to identify all instance profiles in the workspace, and map their access to S3 buckets.\nAlso captures the IAM roles which has UC arn listed, and map their access to S3 buckets\nThis requires `aws` CLI to be installed and configured.\n\nFor AWS this command produces a file named `aws_instance_profile_info.csv`.\nIt has the following format:\n\n| **role_arn**                                         | **resource_type** | **privilege** | **resource_path**     |\n|------------------------------------------------------|-------------------|---------------|-----------------------|\n| arn:aws:iam::1234:instance-profile/instance-profile1 | s3                | WRITE_FILES   | s3://s3_bucket1/path1 |\n\n\nOnce done, proceed to the [`migrate-credentials` command](#migrate-credentials-command).\n\n[[back to top](#databricks-labs-ucx)]\n\n### Access for Azure Storage Accounts\n\n```commandline\ndatabricks labs ucx principal-prefix-access --subscription-ids test-subscription-id\n```\n\nUse to identify all storage account used by tables, identify the relevant Azure service principals and their permissions\non each storage account. The command is used to identify Azure Service Principals, which have `Storage Blob Data Contributor`,\n`Storage Blob Data Reader`, `Storage Blob Data Owner` roles, or custom read/write roles on ADLS Gen2 locations that are being\nused in Databricks. This requires Azure CLI to be installed and configured via `az login`. It outputs azure_storage_account_info.csv\nwhich will be later used by migrate-credentials command to create UC storage credentials.\nNote: This cmd only lists azure storage account gen2, storage format wasb:// or adl:// are not supported in UC and those storage info\nwill be skipped.\n\nOnce done, proceed to the [`migrate-credentials` command](#migrate-credentials-command).\n\n[[back to top](#databricks-labs-ucx)]\n\n## `create-missing-principals` command (AWS Only)\n```bash\ndatabricks labs ucx create-missing-principals --aws-profile <aws_profile> --single-role <single_role>\n```\nThis command identifies all the S3 locations that are missing a UC compatible role and creates them.\nIt takes single-role optional parameter.\nIf set to True, it will create a single role for all the S3 locations.\nOtherwise, it will create a role for each S3 location.\n\nTwo optional parameter are available for this command:\n`--role-name` - This parameter is used to set the prefix for the role name. The default value is `UCX-ROLE`.\n`--role-policy` - This parameter is used to set the prefix for the role policy name. The default value is `UCX-POLICY`.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `delete-missing-principals` command (AWS Only)\n```bash\ndatabricks labs ucx delete-missing-principals --aws-profile <aws_profile>\n```\nThis command helps to delete the IAM role created by UCX. It lists all the IAM Roles generated by the principal-prefix-access\ncommand and allows user to select multiple roles to delete. It also checks if selected roles are mapped to any storage credentials\nand asks for confirmation from user. Once confirmed, it deletes the role and its associated inline policy.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `create-uber-principal` command\n\n```text\ndatabricks labs ucx create-uber-principal [--subscription-ids X]\n```\n\n**Requires Cloud IAM admin privileges.**\n\nOnce the [`assessment` workflow](#assessment-workflow) complete, you should run this command to create a service principal with the\n_**read-only access to all storage**_ used by tables in this workspace. It will also configure the\n[UCX Cluster Policy](#installation) & SQL Warehouse data access configuration to use this service principal for migration\nworkflows. Once migration is complete, this service principal should be unprovisioned.\n\nOn Azure, it creates a principal with `Storage Blob Data Contributor` role assignment on every storage account using\nAzure Resource Manager APIs.\n\nThis command is one of prerequisites for the [table migration process](#table-migration).\n\n[[back to top](#databricks-labs-ucx)]\n\n## `migrate-credentials` command\n\n```commandline\ndatabricks labs ucx migrate-credentials\n```\n\nFor Azure, this command prompts to confirm performing the following credential migration steps:\n1. [RECOMMENDED] For each storage account, create access connectors with managed identities that have the\n   `Storage Blob Data Contributor` role on the respective storage account. A storage credential is created for each\n    access connector.\n2. Migrate Azure Service Principals, which have `Storage Blob Data Contributor`,\n   `Storage Blob Data Reader`, `Storage Blob Data Owner`, or custom roles on ADLS Gen2 locations that are being used in\n   Databricks, to UC storage credentials. The Azure Service Principals to location mapping are listed\n   in `/Users/{user_name}/.ucx/azure_storage_account_info.csv` which is generated by\n   [`principal-prefix-access` command](#principal-prefix-access-command). Please review the file and delete the Service\n   Principals you do not want to be migrated. The command will only migrate the Service Principals that have client\n   secret stored in Databricks Secret.\n\n  **Warning**: Service principals used to access storage accounts behind firewalls might cause connectivity issues. We\n  recommend to use access connectors instead.\n\nFor AWS, this command migrates AWS Instance Profiles that are being used in Databricks, to UC storage credentials.\nThe AWS Instance Profiles to location mapping are listed in\n{workspace ucx folder}/aws_instance_profile_info.csv which is generated by principal_prefix_access command.\nPlease review the file and delete the Instance Profiles you do not want to be migrated. <br/>The aws_profile parameter indicates the aws profile to use.\n\nOnce you're done with this command, run [`validate-external-locations` command](#validate-external-locations-command) after this one.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `validate-external-locations` command\n\n```text\ndatabricks labs ucx validate-external-locations\n```\n\nOnce the [`assessment` workflow](#assessment-workflow) finished successfully, [storage credentials](#migrate-credentials-command) are configured,\nrun this command to validate and report the missing Unity Catalog external locations to be created.\n\nThis command validates and provides mapping to external tables to external locations, also as Terraform configurations.\n\nOnce you're done with this command, proceed to the [`migrate-locations` command](#migrate-locations-command).\n\n[[back to top](#databricks-labs-ucx)]\n\n## `migrate-locations` command\n\n```text\ndatabricks labs ucx migrate-locations\n```\n\nOnce the [`assessment` workflow](#assessment-workflow) finished successfully, and [storage credentials](#migrate-credentials-command) are configured,\nrun this command to have Unity Catalog external locations created. The candidate locations to be created are extracted from guess_external_locations\ntask in the assessment job. You can run [`validate-external-locations` command](#validate-external-locations-command) to check the candidate locations.\n\n**Location ACLs:**\n`migrate-locations` command applies any location ACL from existing cluster.\nFor Azure, it checks if there are any interactive cluster or SQL warehouse\nwhich has service principals configured to access storage. It maps the storage account to the external location created and grants `CREATE_EXTERNAL_TABLE`,\n`CREATE_EXTERNAL_VOLUME` and `READ_FILES` permission on the location to all the user who have access to the interactive cluster or SQL warehouse\nFor AWS, it checks any instance profiles mapped to the interactive cluster or SQL warehouse. It checks the mapping of instance profiles to the bucket. It then\nmaps the bucket to the external locations created and grants `CREATE_EXTERNAL_TABLE`, `CREATE_EXTERNAL_VOLUME` and `READ_FILES` permission on the location to all the user who have access to the interactive cluster\nor SQL warehouse\n\nOnce you're done with this command, proceed to the [`create-table-mapping` command](#create-table-mapping-command).\n\n[[back to top](#databricks-labs-ucx)]\n\n## `create-table-mapping` command\n\n```text\ndatabricks labs ucx create-table-mapping\n```\n\nOnce the [`assessment` workflow](#assessment-workflow) finished successfully\n[workspace info is synchronized](#sync-workspace-info-command), run this command to create the initial\ntable mapping for review in CSV format in the Databricks Workspace:\n\n```text\nworkspace_name,catalog_name,src_schema,dst_schema,src_table,dst_table\nlabs-azure,labs_azure,default,default,ucx_tybzs,ucx_tybzs\n```\n\nThe format of the mapping file is as follows:\n\n| **columns:** | **workspace_name**      | **catalog_name** | **src_schema** | **dst_schema** | **src_table** | **dst_table** |\n|--------------|---------------------|--------------|----------------|----------------|---------------|---------------|\n| values:      | data_engineering_ws | de_catalog   | database1      | database1      | table1        | table1        |\n\nYou are supposed to review this mapping and adjust it if necessary. This file is in CSV format, so that you can edit it\neasier in your favorite spreadsheet application.\n\nOnce you're done with this command, [create catalogs and schemas](#create-catalogs-schemas-command). During\nmultiple runs of the table migration workflow, you can use the [`revert-migrated-tables` command](#revert-migrated-tables-command)\nto revert the tables that were migrated in the previous run. You can also skip the tables that you don't want to migrate\nusing the [`skip` command](#skip-command).\n\nThis command is one of prerequisites for the [table migration process](#Table-Migration).\n\nOnce you're done with table migration, proceed to the [code migration](#code-migration-commands).\n\n[[back to top](#databricks-labs-ucx)]\n\n## `skip` command\n\n```text\ndatabricks labs ucx skip --schema X [--table Y] [--view Z]\n```\n\nAnytime after [`create-table-mapping` command](#create-table-mapping-command) is executed, you can run this command.\n\nThis command allows users to skip certain schemas, tables or views during the [table migration](#table-migration) process.\nThe command takes `--schema` and, optionally, `--table` and `--view` flags to specify the schema, table or view to skip.\nIf no `--table` flag is provided, all tables in the specified HMS database are skipped. The `--table` and `--view` can\nonly be used exclusively. This command is useful to temporarily disable migration of a particular schema, table or view.\n\nOnce you're done with table migration, proceed to the [code migration](#code-migration-commands).\n\n[[back to top](#databricks-labs-ucx)]\n\n## `unskip` command\n\n```commandline\ndatabricks labs ucx unskip --schema X [--table Y] [--view Z]\n```\n\nThis command removes the mark set by the [`skip` command](#skip-command) on the given schema, table or view.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `create-catalogs-schemas` command\n\n```text\ndatabricks labs ucx create-catalogs-schemas\n```\nAfter [`create-table-mapping` command](#create-table-mapping-command) is executed, you can run this command to have the required UC catalogs and schemas created.\nThis command is supposed to be run before migrating tables to UC using [table migration process](#Table-Migration).\nCatalog & Schema ACL:\n`create-catalogs-schemas` command also applies any catalog and schema ACL from existing clusters.\nFor Azure it checks if there are any interactive cluster or sql warehouse which has service principals configured to access storage.\nIt maps the storage account to the tables which has external location on those storage account created and grants `USAGE` access to\nthe schema and catalog if at least one such table is migrated to it.\nFor AWS, it checks any instance profiles mapped to the interactive cluster or sql warehouse. It checks the mapping of instance profiles\nto the bucket. It then maps the bucket to the tables which has external location on those bucket created and grants `USAGE` access to\nthe schema and catalog if at least one such table is migrated to it.\n[[back to top](#databricks-labs-ucx)]\n\n## `assign-owner-group` command\n\n```text\ndatabricks labs ucx assign-owner-group\n```\n\nThis command assigns the owner group to the UCX catalog and all migrated tables.\nThe owner group is an account group that will be designated as an owner to all migrated objects (catalogs, schemas, tables, views).\nThe principal running the command and later, the migration workflows, is required to be a member of this group.\nThe command will list all the groups the principal is a member of and allow the selection of the owner group.\nIt sets the default_owner_group property in the config.yml file.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `migrate-tables` command\n\n```text\ndatabricks labs ucx migrate-tables\n```\n\nAnytime after [`create-table-mapping` command](#create-table-mapping-command) is executed, you can run this command.\n\nThis command kicks off the [table migration](#Table-Migration) process. It triggers the `migrate-tables` workflow,\nand if there are HiveSerDe tables detected, prompt whether to trigger the `migrate-external-hiveserde-tables-in-place-experimental` workflow.\n\nTable and View ACL:\n`migrate-tables` command also applies any table and view ACL from existing clusters.\nFor Azure it checks if there are any interactive cluster or sql warehouse which has service principals configured to access storage.\nIt maps the storage account to the tables which has external location on those storage account created and grants either `SELECT` permission if\nthe service principal only has read access on the storage account and `ALL_PRIVILEGES` if the service principal has write access on the storage account\nFor AWS, it checks any instance profiles mapped to the interactive cluster or sql warehouse. It checks the mapping of instance profiles\nto the bucket. It then maps the bucket to the tables which has external location on those bucket created and grants either `SELECT` permission if\nthe instance profile only has read access on the bucket and `ALL_PRIVILEGES` if the instance profile has write access on the bucket.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `revert-migrated-tables` command\n\n```text\ndatabricks labs ucx revert-migrated-tables --schema X --table Y [--delete-managed]\n```\n\nAnytime after [`create-table-mapping` command](#create-table-mapping-command) is executed, you can run this command.\n\nThis command removes the `upgraded_from` property on a migrated table for re-migration in the [table migration](#Table-Migration) process.\nThis command is useful for developers and administrators who want to revert the migration of a table. It can also be used\nto debug issues related to table migration.\n\nGo back to the [`create-table-mapping` command](#create-table-mapping-command) after you're done with this command.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `move` command\n\n```text\ndatabricks labs ucx move --from-catalog A --from-schema B --from-table C --to-catalog D --to-schema E\n```\n\nThis command moves a UC table/tables from one schema to another schema after\nthe [table migration](#Table-Migration) process. This is useful for developers and administrators who want\nto adjust their catalog structure after tables upgrade.\n\nUsers will be prompted whether tables/view are dropped after moving to new schema. This only applies to `MANAGED` tables and views.\n\nThis command moves different table types differently:\n- `MANAGED` tables are deep-cloned to the new schema.\n- `EXTERNAL` tables are dropped from the original schema, then created in the target schema using the same location.\nThis is due to Unity Catalog not supporting multiple tables with overlapping paths\n- `VIEW` are recreated using the same view definition.\n\nThis command supports moving multiple tables at once, by specifying `*` as the table name.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `alias` command\n\n```text\ndatabricks labs ucx alias --from-catalog A --from-schema B --from-table C --to-catalog D --to-schema E\n```\n\nThis command aliases a UC table/tables from one schema to another schema in the same or different catalog.\nIt takes a `WorkspaceClient` object and `from` and `to` parameters as parameters and aliases the tables using\nthe `TableMove` class. This command is useful for developers and administrators who want to create an alias for a table.\nIt can also be used to debug issues related to table aliasing.\n\n[[back to top](#databricks-labs-ucx)]\n\n# Code migration commands\n\nSee the [migration process diagram](#migration-process) to understand the role of the code migration commands in the migration process.\n\nAfter you're done with the [table migration](#Table-Migration), you can proceed to the code migration.\n\nOnce you're done with the code migration, you can run the [`cluster-remap` command](#cluster-remap-command) to remap the\nclusters to be UC compatible.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `lint-local-code` command\n\n```text\ndatabricks labs ucx lint-local-code\n```\n\nAt any time, you can run this command to assess all migrations required in a local directory or a file. It only takes seconds to run and it\ngives you an initial overview of what needs to be migrated without actually performing any migration. A great way to start a migration!\n\nThis command detects all dependencies, and analyzes them. It is still experimental and at the moment only supports Python and SQL files.\nWe expect this command to run within a minute on code bases up to 50.000 lines of code.\nFuture versions of `ucx` will add support for more source types, and more migration details.\n\nWhen run from an IDE terminal, this command generates output as follows:\n![img.png](docs/lint-local-code-output.png)\nWith modern IDEs, clicking on the file link opens the file at the problematic line\n\n[[back to top](#databricks-labs-ucx)]\n\n## `migrate-local-code` command\n\n```text\ndatabricks labs ucx migrate-local-code\n```\n\n**(Experimental)** Once [table migration](#Table-Migration) is complete, you can run this command to\nmigrate all python and SQL files in the current working directory. This command is highly experimental and\nat the moment only supports Python and SQL files and discards code comments and formatting during\nthe automated transformation process.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `migrate-dbsql-dashboards` command\n\n```text\ndatabricks labs ucx migrate-dbsql-dashboards [--dashboard-id <dashboard-id>]\n```\n\n**(Experimental)** Once [table migration](#Table-Migration) is complete, you can run this command to\nmigrate all Databricks SQL dashboards in the workspace. At this moment, this command is highly experimental and discards\nformatting during the automated transformation process.\n\nThis command tags dashboards & queries that have been migrated with `migrated by UCX` tag. The original queries are\nalso backed up in the ucx installation folder, to allow for easy rollback (see [`revert-dbsql-dashboards` command](#revert-dbsql-dashboards-command)).\n\nThis command can be run with `--dashboard-id` flag to migrate a specific dashboard.\n\nThis command is incremental and can be run multiple times to migrate new dashboards.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `revert-dbsql-dashboards` command\n\n```text\ndatabricks labs ucx revert-dbsql-dashboards [--dashboard-id <dashboard-id>]\n```\n\n**(Experimental)** This command reverts the migration of Databricks SQL dashboards in the workspace, after\n`migrate-dbsql-dashboards` command is executed.\n\nThis command can be run with `--dashboard-id` flag to migrate a specific dashboard.\n\n[[back to top](#databricks-labs-ucx)]\n\n# Cross-workspace installations\n\nWhen installing UCX across multiple workspaces, administrators need to keep UCX configurations in sync.\nUCX will prompt you to select an account profile that has been defined in `~/.databrickscfg`. If you don't have one,\nauthenticate your machine with:\n\n* `databricks auth login --host https://accounts.cloud.databricks.com/` (AWS)\n* `databricks auth login --host https://accounts.azuredatabricks.net/` (Azure)\n\nAsk your Databricks Account admin to run the [`sync-workspace-info` command](#sync-workspace-info-command) to sync the\nworkspace information with the UCX installations. Once the workspace information is synced, you can run the\n[`create-table-mapping` command](#create-table-mapping-command) to align your tables with the Unity Catalog.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `sync-workspace-info` command\n\n```text\ndatabricks --profile ACCOUNTS labs ucx sync-workspace-info\n14:07:07  INFO [databricks.sdk] Using Azure CLI authentication with AAD tokens\n14:07:07  INFO [d.labs.ucx] Account ID: ...\n14:07:10  INFO [d.l.blueprint.parallel][finding_ucx_installations_16] finding ucx installations 10/88, rps: 16.415/sec\n14:07:10  INFO [d.l.blueprint.parallel][finding_ucx_installations_0] finding ucx installations 20/88, rps: 32.110/sec\n14:07:11  INFO [d.l.blueprint.parallel][finding_ucx_installations_18] finding ucx installations 30/88, rps: 39.786/sec\n...\n```\n\n> Requires Databricks Account Administrator privileges. Use `--profile` to select the Databricks cli profile configured\n> with access to the Databricks account console (with endpoint \"https://accounts.cloud.databricks.com/\"\n> or \"https://accounts.azuredatabricks.net\").\n\nThis command uploads the workspace config to all workspaces in the account where `ucx` is installed. This command is\nnecessary to create an immutable default catalog mapping for [table migration](#Table-Migration) process and is the prerequisite\nfor [`create-table-mapping` command](#create-table-mapping-command).\n\nIf you cannot get account administrator privileges in reasonable time, you can take the risk and\nrun [`manual-workspace-info` command](#manual-workspace-info-command) to enter Databricks Workspace IDs and Databricks\nWorkspace names.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `manual-workspace-info` command\n\n```text\n$ databricks labs ucx manual-workspace-info\n14:20:36  WARN [d.l.ucx.account] You are strongly recommended to run \"databricks labs ucx sync-workspace-info\" by account admin,\n ... otherwise there is a significant risk of inconsistencies between different workspaces. This command will overwrite all UCX\n ... installations on this given workspace. Result may be consistent only within https://adb-987654321.10.azuredatabricks.net\nWorkspace name for 987654321 (default: workspace-987654321): labs-workspace\nNext workspace id (default: stop): 12345\nWorkspace name for 12345 (default: workspace-12345): other-workspace\nNext workspace id (default: stop):\n14:21:19  INFO [d.l.blueprint.parallel][finding_ucx_installations_11] finding ucx installations 10/89, rps: 24.577/sec\n14:21:19  INFO [d.l.blueprint.parallel][finding_ucx_installations_15] finding ucx installations 20/89, rps: 48.305/sec\n...\n14:21:20  INFO [d.l.ucx.account] Synchronised workspace id mapping for installations on current workspace\n```\n\nThis command is only supposed to be run if the [`sync-workspace-info` command](#sync-workspace-info-command) cannot be\nrun. It prompts the user to enter the required information manually and creates the workspace info. This command is\nuseful for workspace administrators who are unable to use the `sync-workspace-info` command, because they are not\nDatabricks Account Administrators. It can also be used to manually create the workspace info in a new workspace.\n\n[[back to top](#databricks-labs-ucx)]\n\n## `create-account-groups` command\n\n```text\n$ databricks labs ucx create-account-groups [--workspace-ids 123,456,789]\n```\n\n**Requires Databricks Account Administrator privileges.** This command creates account-level groups if a workspace local\ngroup is not present in the account. It crawls all workspaces configured in `--workspace-ids` flag, then creates\naccount level groups if a WS local group is not present in the account. If `--workspace-ids` flag is not specified, UCX\nwill create account groups for all workspaces configured in the account.\n\nThe following scenarios are supported, if a group X:\n- Exist in workspaces A,B,C, and it has same members in there, it will be created in the account\n- Exist in workspaces A,B but not in C, it will be created in the account\n- Exist in workspaces A,B,C. It has same members in A,B, but not in C. Then, X and C_X will be created in the account\n\nThis command is useful for the setups, that don't have SCIM provisioning in place.\n\nOnce you're done with this command, proceed to the [group migration workflow](#group-migration-workflow).\n\n[[back to top](#databricks-labs-ucx)]\n\n## `validate-groups-membership` command\n\n```text\n$ databricks labs ucx validate-groups-membership\n...\n14:30:36  INFO [d.l.u.workspace_access.groups] Found 483 account groups\n14:30:36  INFO [d.l.u.workspace_access.groups] No group listing provided, all matching groups will be migrated\n14:30:36  INFO [d.l.u.workspace_access.groups] There are no groups with different membership between account and workspace\nWorkspace Group Name  Members Count  Account Group Name  Members Count  Difference\n```\n\nThis command validates the groups to see if the groups at the account level and workspace level have different membership.\nThis command is useful for administrators who want to ensure that the groups have the correct membership. It can also be\nused to debug issues related to group membership. See [group migration](docs/local-group-migration.md) and\n[group migration](#group-migration-workflow) for more details.\n\nValid group membership is important to ensure users has correct access after legacy table ACL is migrated in [table migration process](#Table-Migration)\n\n[[back to top](#databricks-labs-ucx)]\n\n## `validate-table-locations` command\n\n```text\n$ databricks labs ucx validate-table-locations [--workspace-ids 123,456,789]\n...\n11:39:36  WARN [d.l.u.account.aggregate] Workspace 99999999 does not have UCX installed\n11:39:37  WARN [d.l.u.account.aggregate] Overlapping table locations: 123456789:hive_metastore.database.table and 987654321:hive_metastore.database.table\n11:39:37  WARN [d.l.u.account.aggregate] Overlapping table locations: 123456789:hive_metastore.database.table and 123456789:hive_metastore.another_database.table\n```\n\nThis command validates the table locations by checking for overlapping table locations in the workspace and across\nworkspaces. Unity catalog does not allow overlapping table locations, also not between tables in different catalogs.\nOverlapping table locations need to be resolved by the user before running the table migration.\n\nOptions to resolve tables with overlapping locations are:\n- Move one table and [skip](#skip-command) the other(s).\n- Duplicate the tables by copying the data into a managed table and [skip](#skip-command) the original tables.\n\nConsiderations when resolving tables with overlapping locations are:\n- Migrate the tables one workspace at a time:\n  - Let later migrated workspaces read tables from the earlier migrated workspace catalogs.\n  - [Move](#move-command) tables between schemas and catalogs when it fits the data management model.\n- The tables might have different:\n  - Metadata, like:\n    - Column schema (names, types, order)\n    - Description\n    - Tags\n  - ACLs\n\n[[back to top](#databricks-labs-ucx)]\n\n## `cluster-remap` command\n\n```text\n$ databricks labs ucx cluster-remap\n21:29:38  INFO [d.labs.ucx] Remapping the Clusters to UC\nCluster Name                                            Cluster Id\nField Eng Shared UC LTS Cluster                         0601-182128-dcbte59m\nShared Autoscaling Americas cluster                     0329-145545-rugby794\n```\n```text\nPlease provide the cluster id's as comma separated value from the above list (default: <ALL>):\n```\n\nOnce you're done with the [code migration](#code-migration-commands), you can run this command to remap the clusters to UC enabled.\n\nThis command will remap the cluster to uc enabled one. When we run this command it will list all the clusters\nand its id's and asks to provide the cluster id's as comma separated value which has to be remapped, by default it will take all cluster ids.\nOnce we provide the cluster id's it will update these clusters to UC enabled.Back up of the existing cluster\nconfig will be stored in backup folder inside the installed location(backup/clusters/cluster_id.json) as a json file.This will help\nto revert the cluster remapping.\n\nYou can revert the cluster remapping using the [`revert-cluster-remap` command](#revert-cluster-remap-command).\n\n[[back to top](#databricks-labs-ucx)]\n\n## `revert-cluster-remap` command\n\n```text\n$ databricks labs ucx revert-cluster-remap\n21:31:29  INFO [d.labs.ucx] Reverting the Remapping of the Clusters from UC\n21:31:33  INFO [d.labs.ucx] 0301-055912-4ske39iq\n21:31:33  INFO [d.labs.ucx] 0306-121015-v1llqff6\nPlease provide the cluster id's as comma separated value from the above list (default: <ALL>):\n```\n\nIf a customer want's to revert the cluster remap done using the [`cluster-remap` command](#cluster-remap-command) they can use this command to revert\nits configuration from UC to original one.It will iterate through the list of clusters from the backup folder and reverts the\ncluster configurations to original one.This will also ask the user to provide the list of clusters that has to be reverted as a prompt.\nBy default, it will revert all the clusters present in the backup folder\n\n[[back to top](#databricks-labs-ucx)]\n\n## `upload` command\n\n```text\n$ databricks labs ucx upload --file <file_path> --run-as-collection True\n21:31:29 WARNING [d.labs.ucx] The schema of CSV files is NOT validated, ensure it is correct\n21:31:29 INFO [d.labs.ucx] Finished uploading: <file_path>\n```\n\nUpload a file to a single workspace (`--run-as-collection False`) or a collection of workspaces\n(`--run-as-collection True`). This command is especially useful when uploading the same file to multiple workspaces.\n\n## `download` command\n\n```text\n$ databricks labs ucx download --file <file_path> --run-as-collection True\n21:31:29 INFO [d.labs.ucx] Finished downloading: <file_path>\n```\n\nDownload a csv file from a single workspace (`--run-as-collection False`) or a collection of workspaces\n(`--run-as-collection True`). This command is especially useful when downloading the same file from multiple workspaces.\n\n## `join-collection` command\n\n```text\n$ databricks labs ucx join-collection --workspace-ids <comma seperate list of workspace ids> --profile <account-profile>\n```\n\n`join-collection` command joins 2 or more workspaces into a collection. This helps in running supported cli commands as a collection\n`join-collection` command updates config.yml file on each workspace ucx installation with installed_workspace_ids attribute.\nIn order to run `join-collectioon` command a user should:\n - be an Account admin on the Databricks account\n - be a Workspace admin on all the workspaces to be joined as a collection) or a collection of workspaces\n - have installed UCX on the workspace\nThe `join-collection` command will fail and throw an error msg if the above conditions are not met.\n\n## collection eligible command\n\nOnce `join-collection` command is run, it allows user to run multiple cli commands as a collection. The following cli commands\nare eligible to be run as a collection. User can run the below commands as collection by passing an additional flag `--run-as-collection=True`\n- `ensure-assessment-run`\n- `create-table-mapping`\n- `principal-prefix-access`\n- `migrate-credentials`\n- `create-uber-principal`\n- `create-missing-principals`\n- `validate-external-location`\n- `migrate-locations`\n- `create-catalog-schemas`\n- `migrate-tables`\n- `migrate-acls`\n- `migrate-dbsql-dashboards`\n- `validate-group-membership`\nEx: `databricks labs ucx ensure-assessment-run --run-as-collection=True`\n\n\n# Common Challenges and the Solutions\nUsers might encounter some challenges while installing and executing UCX. Please find the listing of some common challenges and the solutions below.\n\n[[back to top](#databricks-labs-ucx)]\n\n### Network Connectivity Issues\n\n**From local machine to the Databricks Account and Workspace:** UCX\ninstallation process has to be run from the local laptop using\nDatabricks CLI and it will deploy the latest version of UCX into the\nDatabricks workspace. For this reason, the Databricks account and\nworkspace needs to be accessible from the laptop. Sometimes, the\nworkspace might have a network isolation, like it can only be reached\nfrom a VPC, or from a specific IP range.\n\n**Solution:** Please check that your laptop has network connectivity to\nthe Databricks account and workspace. If not, you might need to be\nconnected to a VPN or configure an HTTP proxy to access your workspace.\n\n**From local machine to GitHub:** UCX needs internet access to connect to GitHub (https://api.github.com\nand https://raw.githubusercontent.com) for downloading the tool from the machine running the installation. The\ninstallation will fail if there is no internet connectivity to these URLs.\n\n**Solution:** Ensure that GitHub is reachable from the local machine. If\nnot, make necessary changes to the network/firewall settings.\n\n**From Databricks workspace to PyPi:** There are some dependent\nlibraries which need to be installed from\n[<u>pypi.org</u>](https://pypi.org/) to run the UCX workflows from the\nDatabricks workspace. If the workspace doesn\u2019t have network\nconnectivity, then the job might fail with\nNO_MATCHING_DISTRIBUTION_ERROR.\n\n**Solution:** Version 0.24.0 of UCX supports workspace with no internet\naccess. Please upgrade UCX and rerun the installation. Reply *yes* to\nthe question \"Does the given workspace block Internet access?\" asked\nduring installation. It will then upload all necessary dependencies to\nthe workspace. Also, please note that UCX uses both UC and non-UC\nenabled clusters. If you have different proxy settings for each, then\nplease update the necessary proxies (eg. with init scripts) for each\ncluster type.\n\n**Local machine to Databricks Account and Workspace connection failed due to proxy and self-signed cert:**\nWhen customer uses web proxy and self-singed certification, UCX may not be able to connect to Account and Workspace\nwith following errors:\n```\nFile \"/Users/userabc/.databricks/labs/ucx/state/venv/lib/python3.10/site-packages/urllib3/connectionpool.py\", line 466, in _make_request\n    self._validate_conn(conn)\n  File \"/Users/userabc/.databricks/labs/ucx/state/venv/lib/python3.10/site-packages/urllib3/connectionpool.py\", line 1095, in _validate_conn\n    conn.connect()\n  File \"/Users/userabc/.databricks/labs/ucx/state/venv/lib/python3.10/site-packages/urllib3/connection.py\", line 652, in connect\n    sock_and_verified = _ssl_wrap_socket_and_match_hostname(\n  File \"/Users/userabc/.databricks/labs/ucx/state/venv/lib/python3.10/site-packages/urllib3/connection.py\", line 805, in _ssl_wrap_socket_and_match_hostname\n    ssl_sock = ssl_wrap_socket(\n  File \"/Users/userabc/.databricks/labs/ucx/state/venv/lib/python3.10/site-packages/urllib3/util/ssl_.py\", line 465, in ssl_wrap_socket\n    ssl_sock = _ssl_wrap_socket_impl(sock, context, tls_in_tls, server_hostname)\n  File \"/Users/userabc/.databricks/labs/ucx/state/venv/lib/python3.10/site-packages/urllib3/util/ssl_.py\", line 509, in _ssl_wrap_socket_impl\n    return ssl_context.wrap_socket(sock, server_hostname=server_hostname)\n  File \"/opt/homebrew/Cellar/python@3.10/3.10.14/Frameworks/Python.framework/Versions/3.10/lib/python3.10/ssl.py\", line 513, in wrap_socket\n    return self.sslsocket_class._create(\n  File \"/opt/homebrew/Cellar/python@3.10/3.10.14/Frameworks/Python.framework/Versions/3.10/lib/python3.10/ssl.py\", line 1104, in _create\n    self.do_handshake()\n  File \"/opt/homebrew/Cellar/python@3.10/3.10.14/Frameworks/Python.framework/Versions/3.10/lib/python3.10/ssl.py\", line 1375, in do_handshake\n    self._sslobj.do_handshake()\nssl.SSLCertVerificationError: [SSL: CERTIFICATE_VERIFY_FAILED] certificate verify failed: self-signed certificate in certificate chain (_ssl.c:1007)\n```\n\n**Solution:**  set both `REQUESTS_CA_BUNDLE` and `CURL_CA_BUNDLE`\n[to force requests library to set verify=False](https://github.com/psf/requests/blob/8c211a96cdbe9fe320d63d9e1ae15c5c07e179f8/requests/sessions.py#L718)\nas well as set `SSL_CERT_DIR` env var pointing to the proxy CA cert for the urllib3 library.\n\n[[back to top](#databricks-labs-ucx)]\n\n### Insufficient Privileges\n\n**User is not a Databricks workspace administrator:** User running the\ninstallation needs to be a workspace administrator as the CLI will\ndeploy the UCX tool into the workspace, create jobs, and dashboards.\n\n**Solution:** Identify a workspace admin from your team and ask them to\ninstall UCX with their authentication, or request a workspace\nadministrator to grant you temporary administrator privileges to run the\ninstallation. More details on the issues that you can run into if\nyou are not an admin (and some possible solutions) can be found\n[here](docs/troubleshooting.md#resolving-common-errors-on-ucx-install).\n\n**User is not a Cloud IAM Administrator:** Cloud CLI needs to be\ninstalled in the local machine for certain cloud related activities,\nlike creating an [<u>uber\nprincipal</u>](#create-uber-principal-command).\nFor this, the user needs Cloud IAM Administrator privileges.\n\n**Solution:** Work with a cloud administrator in your organization to\nrun the commands that need cloud administrator rights.\n\nAdmin privileges required for commands:\n\n| **CLI command**                                                                                  | **Admin privileges** |\n|--------------------------------------------------------------------------------------------------|----|\n| [<u>install</u>](#install-ucx)                                                                   | Workspace Admin |\n| [<u>account install</u>](#advanced-installing-ucx-on-all-workspaces-within-a-databricks-account) | Account Admin |\n| [<u>create-account-groups</u>](#create-account-groups-command)                                   | Account Admin |\n| [<u>validate-groups-membership</u>](#validate-groups-membership-command)                         | Account Admin |\n| [<u>create-uber-principal</u>](#create-uber-principal-command)                                   | Cloud Admin |\n| [<u>principal-prefix-access</u>](#principal-prefix-access-command)                               | Cloud Admin |\n| [<u>create-missing-principals</u>](#create-missing-principals-command-aws-only)                  | Cloud Admin |\n| [<u>delete-missing-principals</u>](#delete-missing-principals-command-aws-only)                  | Cloud Admin |\n| [<u>migrate-credentials</u>](#migrate-credentials-command)                                       | Cloud Admin, Account Admin / Metastore Admin / CREATE STORAGE CREDENTIAL privilege |\n| [<u>migrate-location</u>](#migrate-locations-command)                                            | Metastore Admin / CREATE EXTERNAL LOCATION privilege |\n| [<u>create-catalogs-schemas</u>](#create-catalogs-schemas-command)                               | Metastore Admin / CREATE CATALOG privilege |\n| [<u>sync-workspace-info</u>](#sync-workspace-info-command)                                       | Account Admin |\n| [<u>manual-workspace-info</u>](#manual-workspace-info-command)                                   | Workspace Admin |\n\n[[back to top](#databricks-labs-ucx)]\n\n### Version Issues\n\n**Python:** UCX needs Python version 3.10 or later.\n\n**Solution:** Check the current version using `python --version`. If the\nversion is lower than 3.10, upgrade the local Python version to 3.10 or\nhigher.\n\n**Databricks CLI:** Databricks CLI v0.213 or higher is needed.\n\n**Solution:** Check the current version with `databricks --version`. For\nlower versions of CLI,\n[<u>update</u>](https://docs.databricks.com/en/dev-tools/cli/install.html#update)\nthe Databricks CLI on the local machine.\n\n**UCX:** When you install UCX, you get the latest version. But since UCX\nis being actively developed, new versions are released frequently. There\nmight be issues if you have run the assessment with a much earlier\nversion, and then trying to run the migration workflows with the latest\nUCX version.\n\n**Solution:** Upgrade UCX, and rerun the assessment job before running\nthe migration workflows. For some reason, if you want to install a\nspecific version of UCX, you can do it using the command\n`databricks labs install ucx@\\<version\\>`, for example,\n`databricks labs install ucx@v0.21.0`.\n\n[[back to top](#databricks-labs-ucx)]\n\n### Authentication Issues\n\n**Workspace Level:** If you are facing authentication issues while\nsetting up Databricks CLI, please refer to the\n[Cryptic errors on authentication](docs/troubleshooting.md#cryptic-errors-on-authentication)\nsection to resolve the common errors related to authentication,\nprofiles, and tokens.\n\n**Account Level:** Not only workspace, but account level authentication\nis also needed for installing UCX. If you do not have an account\nconfigured in .databrickscfg, you will get an error message\n\".databrickscfg does not contain account profiles; please create one\nfirst\".\n\n**Solution:** To authenticate with a Databricks account, consider using\none of the following authentication types: [<u>OAuth machine-to-machine\n(M2M)\nauthentication</u>](https://docs.databricks.com/en/dev-tools/cli/authentication.html#m2m-auth),\n[<u>OAuth user-to-machine (U2M)\nauthentication</u>](https://docs.databricks.com/en/dev-tools/cli/authentication.html#u2m-auth),\n[<u>Basic authentication\n(legacy)</u>](https://docs.databricks.com/en/dev-tools/cli/authentication.html#basic-auth).\n\n[[back to top](#databricks-labs-ucx)]\n\n### Multiple Profiles in Databricks CLI\n\n**Workspace Level:** More than one workspace profile can be configured\nin the .databrickscfg file. For example, you can have profiles set for\nDev and Prod workspaces. You want to install UCX only for the Prod\nworkspace.\n\n**Solution:** The Databricks CLI provides an option to select the\n[<u>profile</u>](https://docs.databricks.com/en/dev-tools/cli/profiles.html)\nusing `--profile \\<profile_name\\>` or `-p \\<profile_name\\>`. You can\ntest that the correct workspace is getting selected by running any\nDatabricks CLI command. For example, you can run `databricks clusters\nlist -p prod` and check that the Prod clusters are being returned. Once\nthe profile is verified, you can run UCX install for that specific\nprofile: `databricks labs install ucx -p prod`.\n\n**Account Level:** Multiple account level profiles are set in the\n.databrickscfg file.\n\n**Solution:** The installation command `databricks labs install ucx`\nwill provide an option to select one account profile.\n\n[[back to top](#databricks-labs-ucx)]\n\n### Workspace has an external Hive Metastore (HMS)\n\n**External HMS connectivity from UCX clusters:** If the workspace has an\nexternal HMS, the clusters running the UCX jobs need to have specific\nconfigurations to connect to the external HMS. Otherwise, UCX assessment\nwill not be able to assess the tables on HMS.\n\n**Solution:** Use a cluster policy before installation to set the\nrequired Spark config for connecting to the external HMS, or manually\nedit the cluster post-installation to have the correct configurations.\nDetailed steps can be found\n[<u>here</u>](docs/external_hms_glue.md).\n\n**External HMS connectivity from UCX SQL warehouse:** UCX requires a SQL\nwarehouse to create tables, run queries, create and refresh dashboards.\nIf you already have a Pro or Serverless warehouse connected to the\nexternal HMS, you can select the same warehouse for UCX. You will also\nbe given an option (during installation) to create a new warehouse for\nUCX. If you have never used a warehouse before, the new warehouse\ncreated might not have proper configuration set to connect to the\nexternal HMS.\n\n**Solution:** Set Spark configuration for connecting to external HMS in\nthe Admin Settings of SQL warehouse. This will only be needed if the\nadmin settings do not have the configurations already set. For example,\nadd *spark.hadoop.javax.jdo.option.ConnectionURL \\<connectionstring\\>*\nunder Data Access Configuration of SQL Warehouse Admin Settings.\n\n[[back to top](#databricks-labs-ucx)]\n\n### Verify the Installation\n\nOnce the UCX command `databricks labs install ucx` has completed\nsuccessfully, the installation can be verified with the following steps:\n\n1.  Go to the Databricks Catalog Explorer and check if a new schema for\n    ucx is available in Hive Metastore with all empty tables.\n\n2.  Check that the UCX jobs are visible under Workflows.\n\n3.  Run the assessment. This will start the UCX clusters, crawl through\n    the workspace, and display results in the UCX dashboards. In case of\n    external HMS, verify from the results that the assessment has\n    analyzed the external HMS tables.\n\n[[back to top](#databricks-labs-ucx)]\n\n# Star History\n\n[![Star History Chart](https://api.star-history.com/svg?repos=databrickslabs/ucx&type=Date)](https://star-history.com/#databrickslabs/ucx)\n\n[[back to top](#databricks-labs-ucx)]\n\n# Project Support\nPlease note that all projects in the databrickslabs GitHub account are provided for your exploration only, and are not formally supported by Databricks with Service Level Agreements (SLAs).  They are provided AS-IS, and we do not make any guarantees of any kind.  Please do not submit a support ticket relating to any issues arising from the use of these projects.\n\nAny issues discovered through the use of this project should be filed as GitHub Issues on the Repo.  They will be reviewed as time permits, but there are no formal SLAs for support.\n",
    "bugtrack_url": null,
    "license": null,
    "summary": "UCX - Unity Catalog Migration Toolkit",
    "version": "0.53.1",
    "project_urls": {
        "Issues": "https://github.com/databricks/ucx/issues",
        "Source": "https://github.com/databricks/ucx"
    },
    "split_keywords": [
        "databricks",
        " unity catalog"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "af2f5a74f1c6d34c97074c90d2d4b2e70f1c32d53ff88963ec7e1bdf9c7a5154",
                "md5": "fda2e5ad38c4490f00372249b77f1fc3",
                "sha256": "d94ccdbea8c332d2380354f7bde9c50ab438f68c3b7fcbe40978737c9fc6d90a"
            },
            "downloads": -1,
            "filename": "databricks_labs_ucx-0.53.1-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "fda2e5ad38c4490f00372249b77f1fc3",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.10",
            "size": 593674,
            "upload_time": "2024-12-30T16:58:09",
            "upload_time_iso_8601": "2024-12-30T16:58:09.375120Z",
            "url": "https://files.pythonhosted.org/packages/af/2f/5a74f1c6d34c97074c90d2d4b2e70f1c32d53ff88963ec7e1bdf9c7a5154/databricks_labs_ucx-0.53.1-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "34d055abb903656625b510479855e44286bea973271b78c14f2f36c901bddbea",
                "md5": "aa99adb55356eaf7a550752b03afc67d",
                "sha256": "b8db0eacf963a40a1643bd3cd4b231abcfa8f9cac9e0b9fc69df139ca22547d9"
            },
            "downloads": -1,
            "filename": "databricks_labs_ucx-0.53.1.tar.gz",
            "has_sig": false,
            "md5_digest": "aa99adb55356eaf7a550752b03afc67d",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.10",
            "size": 516670,
            "upload_time": "2024-12-30T16:58:14",
            "upload_time_iso_8601": "2024-12-30T16:58:14.359376Z",
            "url": "https://files.pythonhosted.org/packages/34/d0/55abb903656625b510479855e44286bea973271b78c14f2f36c901bddbea/databricks_labs_ucx-0.53.1.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-12-30 16:58:14",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "databricks",
    "github_project": "ucx",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "lcname": "databricks-labs-ucx"
}
        
Elapsed time: 0.45774s