Skip to content

Commit

Permalink
Tableau Migration App 1.0.0
Browse files Browse the repository at this point in the history
  • Loading branch information
lszinv committed Oct 22, 2024
1 parent 01bc664 commit d502e06
Show file tree
Hide file tree
Showing 102 changed files with 7,028 additions and 0 deletions.
43 changes: 43 additions & 0 deletions BUILD.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
# Tableau ESMB Migration App
Tableau express migration application to migrate data sources from Tableau Server to Tableau Cloud.
Requirements doc can be found [here](https://docs.google.com/document/d/1DXrYdTbS5aGcZeicNVAdD1tvGRwtH1Yj/edit#heading=h.gjdgxs).

# Repo Structure
```
./src
├── Tableau.Migration.App.Core/
└── Tableau.Migration.App.GUI/
```
* Tableau.Migration.App.Core - Core library functionality using the Tableau Migration SDK.
* Tableau.Migration.App.GUI - Gui implementation using Avalonia framework.

# Building
## From Container
The dockerfile is defined in the `Dockerfile` from the project root.
```
docker build .
```
## From Source Root
```
dotnet build Tableau.Migration.App.sln
```

## Documentation
```
cd docs
docfx # Build
docfx --serve # Host the docs locally
```

## Sub Project Build
Individual sub projects can be built from their respective folders.
```
cd src/Tableau.Migration.App.GUI/
dotnet build
dotnet run
```

# Testing
```
dotnet test Tableau.Migration.App.sln
```
37 changes: 37 additions & 0 deletions Dockerfile
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
FROM ubuntu:22.04

# Set environment variables
ENV DEBIAN_FRONTEND=noninteractive
ENV DOTNET_SDK_VERSION=8.0

# Install required packages for C#/Dotnet projects
RUN apt-get update && apt-get install -y \
wget \
apt-transport-https \
ca-certificates \
gnupg \
software-properties-common \
&& wget https://packages.microsoft.com/config/ubuntu/22.04/packages-microsoft-prod.deb -O packages-microsoft-prod.deb \
&& dpkg -i packages-microsoft-prod.deb \
&& rm packages-microsoft-prod.deb \
&& apt-get update && apt-get install -y \
dotnet-sdk-$DOTNET_SDK_VERSION \
&& apt-get clean \
&& rm -rf /var/lib/apt/lists/*

# Set the working directory
WORKDIR /app

# Copy the project files to the container
COPY . .

# # Restore the dependencies
RUN dotnet restore

# # Build the project
RUN dotnet build --configuration Release --no-restore

RUN dotnet test --no-build --configuration Release --verbosity normal

# # Default command to run when the container starts
CMD ["dotnet", "run"]
42 changes: 42 additions & 0 deletions MigrationApp.sln
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio Version 17
VisualStudioVersion = 17.0.31903.59
MinimumVisualStudioVersion = 10.0.40219.1
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Tableau.Migration.App.Core", "src\Tableau.Migration.App.Core\Tableau.Migration.App.Core.csproj", "{BD24867E-3251-4014-B5E7-AD1ABED4EB1A}"
EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Tableau.Migration.App.GUI", "src\Tableau.Migration.App.GUI\Tableau.Migration.App.GUI.csproj", "{AA14EDF6-7994-4CF3-970C-1136DD2491A3}"
EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Tableau.Migration.App.Core.Tests", "tests\Tableau.Migration.App.Core.Tests\Tableau.Migration.App.Core.Tests.csproj", "{E9180159-E6A0-4BF9-BA42-EFF6281D72CC}"
EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Tableau.Migration.App.GUI.Tests", "tests\Tableau.Migration.App.GUI.Tests\Tableau.Migration.App.GUI.Tests.csproj", "{C172CFCB-62C3-46E6-AF02-F0DC55D79DE5}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
Release|Any CPU = Release|Any CPU
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
{BD24867E-3251-4014-B5E7-AD1ABED4EB1A}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{BD24867E-3251-4014-B5E7-AD1ABED4EB1A}.Debug|Any CPU.Build.0 = Debug|Any CPU
{BD24867E-3251-4014-B5E7-AD1ABED4EB1A}.Release|Any CPU.ActiveCfg = Release|Any CPU
{BD24867E-3251-4014-B5E7-AD1ABED4EB1A}.Release|Any CPU.Build.0 = Release|Any CPU
{AA14EDF6-7994-4CF3-970C-1136DD2491A3}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{AA14EDF6-7994-4CF3-970C-1136DD2491A3}.Debug|Any CPU.Build.0 = Debug|Any CPU
{AA14EDF6-7994-4CF3-970C-1136DD2491A3}.Release|Any CPU.ActiveCfg = Release|Any CPU
{AA14EDF6-7994-4CF3-970C-1136DD2491A3}.Release|Any CPU.Build.0 = Release|Any CPU
{E9180159-E6A0-4BF9-BA42-EFF6281D72CC}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{E9180159-E6A0-4BF9-BA42-EFF6281D72CC}.Debug|Any CPU.Build.0 = Debug|Any CPU
{E9180159-E6A0-4BF9-BA42-EFF6281D72CC}.Release|Any CPU.ActiveCfg = Release|Any CPU
{E9180159-E6A0-4BF9-BA42-EFF6281D72CC}.Release|Any CPU.Build.0 = Release|Any CPU
{C172CFCB-62C3-46E6-AF02-F0DC55D79DE5}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{C172CFCB-62C3-46E6-AF02-F0DC55D79DE5}.Debug|Any CPU.Build.0 = Debug|Any CPU
{C172CFCB-62C3-46E6-AF02-F0DC55D79DE5}.Release|Any CPU.ActiveCfg = Release|Any CPU
{C172CFCB-62C3-46E6-AF02-F0DC55D79DE5}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
EndGlobalSection
GlobalSection(ExtensibilityGlobals) = postSolution
SolutionGuid = {8900745D-474C-4BE7-8745-A24DDD64DF2B}
EndGlobalSection
EndGlobal
160 changes: 160 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,160 @@
# Tableau Migration App
### Table of Contents
- [Tableau Migration App](#tableau-migration-app)
- [Table of Contents](#table-of-contents)
- [Features](#features)
- [Usage](#usage)
- [Tableau Server and Cloud URLs](#tableau-server-and-cloud-urls)
- [Personal Access Tokens](#personal-access-tokens)
- [User Mappings](#user-mappings)
- [Username Already in Email format](#username-already-in-email-format)
- [User has an Associated Email](#user-has-an-associated-email)
- [Default Domain Mapping](#default-domain-mapping)
- [CSV Mapping](#csv-mapping)
- [Outputs](#outputs)
- [Logging](#logging)
- [FAQ](#faq)
- [The application is taking a long time, is it stuck?!](#the-application-is-taking-a-long-time-is-it-stuck)
- [How do I migrate items with more specific logic rules?](#how-do-i-migrate-items-with-more-specific-logic-rules)
- [How do I exclude certain items from being migrated?](#how-do-i-exclude-certain-items-from-being-migrated)
- [What order will my resources get migrated in?](#what-order-will-my-resources-get-migrated-in)

## Features
The purpose of ths Tabelau Migration App is to provide users with a method to perform simple migrations from their Tableau Server to Tableau Cloud. This app is meant to accompany the [Tableau Manual Migration Guide](https://help.tableau.com/current/guides/migration/en-us/emg_intro.htm) and replace the work required for the migration steps.

<div align="center">
<img src="./screenshots/TableauMigrationApp.png" alt="Tableau Migration App" width=400>
</div>

* The migration app uses the [Tableau Migration SDK](https://github.com/tableau/tableau-migration-sdk) under the hood, and supports all migration resources that the SDK offers.
* At the time of this writing, this includes:
* Users
* Groups
* Projects
* Data Sources
* Workbooks
* Extract Refresh Tasks
* Custom Views

* Basic mapping options are provided in the app for username migrations.
* A simplified view of ongoing migration progress.

## Usage
### Tableau Server and Cloud URLs
> <div align="center"><img src="./screenshots/UrlParams.png" alt="Tableau Migration App URL Fields" ></div>
The fields in this section are for providing the URIs of the Tableau Server you wish to migrate from, and the Tableau Cloud you wish to migrate to.

These URIs can be found from your browser address bar when logging into the respective Tableau products, or is what you use when connecting from Tableau Desktop.

> The `Base URI` and `Site Name` fields will automatically extract the appropriate information based on the URI provided. Double check that these are correct before moving forward.
**Note**: Multi-site environments will show the `/site/` as part of the URI. If yours does not have one, it is not a problem, and just means that you only have a single site; the `Default` site.

### Personal Access Tokens
> <div align="center"><img src="./screenshots/PatParams.png" alt="Tableau Migration App Token Fields" ></div>
Personal Access Tokens (PATs) are used by the app to programatically access and make changes to the corresponding Tableau Products.

PATs can be generated in your Tableau Environment by going to

> Users > {Your User Here} > Settings > Personal Access Tokens
Where you will be prompted to provide a `Token name` and a `Create Token` button.
> <div align="center"><img src="./screenshots/CreatePAT.png" alt="Tableau Migration PAT" ></div>

Once created, a `secret` will be provided to you for that PAT.

> <div align="center"><img src="./screenshots/CreatePATDetails.png" alt="Tableau PAT Details" ></div>
The `Token Name` is to be filled in the `PAT Name` field of the Migration app, and the `Secret` goes in the `PAT` field.

This will need to be done separately for both the source Tableau Server and the destination Tableau Cloud. Each PAT only provides access to the system it was geneated on.

**Note** The PAT will only provide access to resources of which the user is authorized to have. Resources of which the PAT owner does not have access to will **not** be migrated.

### User Mappings
> **!Important!** Users in Tableau Cloud must have a username in the form of an email. Since Tableau Server does not have this restriction, we've provided different ways for you to designate how the users should be migrated if they don't currently follow this format. The different cases are detailed below.
The order of priority of defined mappings for the migration are as such:
1. CSV Defined mapping
2. Username already in email format
3. User has an associated Email
4. Default Domain Mapping

Meaning that if a defined mapping is found for a user in the CSV file, that mapping will be used even if the user has an associated email.

The following sections will detail more about each mapping option.

#### Username Already in Email format
If a username is already in an email format, then nothing will need to be done and the User will be migrated over as is.

#### User has an Associated Email
If a user has an associated email set to the profile, that email will instead be used for the user when migrating.

#### Default Domain Mapping
> <div align="center"><img src="./screenshots/DefaultDomainMapping.png" alt="Tableau Migration App Token Fields" ></div>
We also provide a default domain mapping. When this is done, all users will have their names appended with that domain to create an email.

e.g. If a User with username `PeterP` is to be migrated, and the `Default User Domain` is set to `DailyBugle.com`, then that user will be migrated to Tableau Cloud as `[email protected]`

This option can be disabled by clicking the checkbox. In which case, users that do not have a mapping found from the other options will **not** be migrated to Tableau Cloud.

**Note** Users containing illegal email characters in their usernames will still fail migration even with a default domain mapped. For example `[][email protected]` is not a valid username for Tableau Cloud.

**Note2** Users with a space in their Tableau Server usernames such as `Renee Montoya` will have their spaces replaced with a `.` instead, becoming `[email protected]`.

#### CSV Mapping
> <div align="center"><img src="./screenshots/UserFileMapping.png" alt="Tableau Migration App Token Fields" ></div>
For more fine-grained control over how individual users are migrated, you can define specific user mappings in a CSV file.

To do so, the file should contain one user mapping per line, separated by a comma (`,`).

i.e. Every line should look like: `{TableauServerUserName},{TableauCloudUsername}`

The `{TableauServerUserName}` **MUST** match the username exactly and is **case sensitive**. Or else the user will not be mapped.

Here is an example of what the file format should be like.

<div style="border: 1px solid #000; padding: 10px; border-radius: 8px;">

`example.csv`:
```
Timmy Turner, [email protected]
Velma Dinkley, [email protected]
Leonardo, [email protected]
...
```
</div>
<br />

**Note** Starting and trailing white spaces will be trimmed. `user1 , [email protected]` is the same as `user1,[email protected]` for the sake of legibility.

### Outputs
Ongoing outputs of the running Migration will be shown in the output window.
> <div align="center"><img src="./screenshots/OutputWindow.png" alt="Tableau Migration App Output" width=400 ></div>
Once migration of a resource is completed, whether successful or not, an associated message will show in the window.

> <div align="center"><img src="./screenshots/OutputWindowFilled.png" alt="Tableau Migration App Output" width=400 ></div>
### Logging
Application logs can be found in the `Logs` folder from where the application is run. These logs contain all the REST requests made by the underlying [Tableau Migration SDK](https://github.com/tableau/tableau-migration-sdk) to the Tableau environments to perform the migration.

These logs can be used to provide further insight to the ongoing status and/or errors of the migration.

### FAQ

#### The application is taking a long time, is it stuck?!
> Check the logs. It could be that the current resource is really large and is taking a long time to migrate.
>
> Or that the application has hit its limit on the number of requests it can make in the current time frame, and is currently waiting until it can send more. In this case, the logs messages should mention how much longer it needs to wait for.
#### How do I migrate items with more specific logic rules?
> If you want finer control over how migration is handled, such as renaming resources or applying special rules such as re-assigning resources, then it is recommended that you use the [Tableau Migration SDK](https://github.com/tableau/tableau-migration-sdk) to programatically perform the migration.
#### How do I exclude certain items from being migrated?
> You can either delete the items after the migration is performed, or you will have to use the [Tableau Migration SDK](https://github.com/tableau/tableau-migration-sdk)
#### What order will my resources get migrated in?
> Resources are first sorted from largest to smallest, and migrated in that order.
Loading

0 comments on commit d502e06

Please sign in to comment.