mirror of
https://github.com/Azure/cosmos-explorer.git
synced 2025-12-23 10:51:30 +00:00
Compare commits
47 Commits
index-arch
...
pixelCorre
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
bfa68a9b00 | ||
|
|
844b6e6b65 | ||
|
|
58e187aeb2 | ||
|
|
5ba7ce2f10 | ||
|
|
e002a4505c | ||
|
|
6483bd146d | ||
|
|
7b437b62ce | ||
|
|
c504d97f7c | ||
|
|
a23a7791d4 | ||
|
|
9bfb6aecc9 | ||
|
|
9227ad379b | ||
|
|
c83f4fc431 | ||
|
|
d924824536 | ||
|
|
cd27814fad | ||
|
|
909957a9a1 | ||
|
|
569e5ed1fc | ||
|
|
a5c3e6bea0 | ||
|
|
76e63818d3 | ||
|
|
cfb5db4df6 | ||
|
|
922ca5c523 | ||
|
|
bafe002fa3 | ||
|
|
0817acf404 | ||
|
|
8e2c46301d | ||
|
|
012d043c78 | ||
|
|
3afd74a957 | ||
|
|
0ef4399ba4 | ||
|
|
870863a723 | ||
|
|
e3815734db | ||
|
|
5ea78f9abf | ||
|
|
8a56214ec2 | ||
|
|
e3ae006100 | ||
|
|
589b61afaf | ||
|
|
eb3f6bc93f | ||
|
|
6ec909a97b | ||
|
|
08a51ca6b1 | ||
|
|
30a3b5c7a4 | ||
|
|
f370507a27 | ||
|
|
e0edaf405c | ||
|
|
f8231600d6 | ||
|
|
45c8d70c77 | ||
|
|
70d7ee755b | ||
|
|
0a4aed4f47 | ||
|
|
a7d007e0dd | ||
|
|
5f4a4e5c4c | ||
|
|
1b64827c24 | ||
|
|
a6ae784a45 | ||
|
|
7458107efd |
@@ -23,8 +23,6 @@ src/Common/MongoUtility.ts
|
|||||||
src/Common/NotificationsClientBase.ts
|
src/Common/NotificationsClientBase.ts
|
||||||
src/Common/QueriesClient.ts
|
src/Common/QueriesClient.ts
|
||||||
src/Common/Splitter.ts
|
src/Common/Splitter.ts
|
||||||
src/Controls/Heatmap/Heatmap.test.ts
|
|
||||||
src/Controls/Heatmap/Heatmap.ts
|
|
||||||
src/Definitions/datatables.d.ts
|
src/Definitions/datatables.d.ts
|
||||||
src/Definitions/gif.d.ts
|
src/Definitions/gif.d.ts
|
||||||
src/Definitions/globals.d.ts
|
src/Definitions/globals.d.ts
|
||||||
|
|||||||
32
.github/workflows/ci.yml
vendored
32
.github/workflows/ci.yml
vendored
@@ -177,9 +177,39 @@ jobs:
|
|||||||
- name: "Az CLI login"
|
- name: "Az CLI login"
|
||||||
uses: Azure/login@v2
|
uses: Azure/login@v2
|
||||||
with:
|
with:
|
||||||
client-id: ${{ secrets.AZURE_CLIENT_ID }}
|
client-id: ${{ secrets.E2E_TESTS_CLIENT_ID }}
|
||||||
tenant-id: ${{ secrets.AZURE_TENANT_ID }}
|
tenant-id: ${{ secrets.AZURE_TENANT_ID }}
|
||||||
subscription-id: ${{ secrets.AZURE_SUBSCRIPTION_ID }}
|
subscription-id: ${{ secrets.AZURE_SUBSCRIPTION_ID }}
|
||||||
|
# We can't use MSAL within playwright so we acquire tokens prior to running the tests
|
||||||
|
- name: "Acquire RBAC tokens for test accounts"
|
||||||
|
uses: azure/cli@v2
|
||||||
|
with:
|
||||||
|
azcliversion: latest
|
||||||
|
inlineScript: |
|
||||||
|
NOSQL_TESTACCOUNT_TOKEN=$(az account get-access-token --scope "https://github-e2etests-sql.documents.azure.com/.default" -o tsv --query accessToken)
|
||||||
|
echo "::add-mask::$NOSQL_TESTACCOUNT_TOKEN"
|
||||||
|
echo NOSQL_TESTACCOUNT_TOKEN=$NOSQL_TESTACCOUNT_TOKEN >> $GITHUB_ENV
|
||||||
|
NOSQL_READONLY_TESTACCOUNT_TOKEN=$(az account get-access-token --scope "https://github-e2etests-sql-readonly.documents.azure.com/.default" -o tsv --query accessToken)
|
||||||
|
echo "::add-mask::$NOSQL_READONLY_TESTACCOUNT_TOKEN"
|
||||||
|
echo NOSQL_READONLY_TESTACCOUNT_TOKEN=$NOSQL_READONLY_TESTACCOUNT_TOKEN >> $GITHUB_ENV
|
||||||
|
TABLE_TESTACCOUNT_TOKEN=$(az account get-access-token --scope "https://github-e2etests-tables.documents.azure.com/.default" -o tsv --query accessToken)
|
||||||
|
echo "::add-mask::$TABLE_TESTACCOUNT_TOKEN"
|
||||||
|
echo TABLE_TESTACCOUNT_TOKEN=$TABLE_TESTACCOUNT_TOKEN >> $GITHUB_ENV
|
||||||
|
GREMLIN_TESTACCOUNT_TOKEN=$(az account get-access-token --scope "https://github-e2etests-gremlin.documents.azure.com/.default" -o tsv --query accessToken)
|
||||||
|
echo "::add-mask::$GREMLIN_TESTACCOUNT_TOKEN"
|
||||||
|
echo GREMLIN_TESTACCOUNT_TOKEN=$GREMLIN_TESTACCOUNT_TOKEN >> $GITHUB_ENV
|
||||||
|
CASSANDRA_TESTACCOUNT_TOKEN=$(az account get-access-token --scope "https://github-e2etests-cassandra.documents.azure.com/.default" -o tsv --query accessToken)
|
||||||
|
echo "::add-mask::$CASSANDRA_TESTACCOUNT_TOKEN"
|
||||||
|
echo CASSANDRA_TESTACCOUNT_TOKEN=$CASSANDRA_TESTACCOUNT_TOKEN >> $GITHUB_ENV
|
||||||
|
MONGO_TESTACCOUNT_TOKEN=$(az account get-access-token --scope "https://github-e2etests-mongo.documents.azure.com/.default" -o tsv --query accessToken)
|
||||||
|
echo "::add-mask::$MONGO_TESTACCOUNT_TOKEN"
|
||||||
|
echo MONGO_TESTACCOUNT_TOKEN=$MONGO_TESTACCOUNT_TOKEN >> $GITHUB_ENV
|
||||||
|
MONGO32_TESTACCOUNT_TOKEN=$(az account get-access-token --scope "https://github-e2etests-mongo32.documents.azure.com/.default" -o tsv --query accessToken)
|
||||||
|
echo "::add-mask::$MONGO32_TESTACCOUNT_TOKEN"
|
||||||
|
echo MONGO32_TESTACCOUNT_TOKEN=$MONGO32_TESTACCOUNT_TOKEN >> $GITHUB_ENV
|
||||||
|
MONGO_READONLY_TESTACCOUNT_TOKEN=$(az account get-access-token --scope "https://github-e2etests-mongo-readonly.documents.azure.com/.default" -o tsv --query accessToken)
|
||||||
|
echo "::add-mask::$MONGO_READONLY_TESTACCOUNT_TOKEN"
|
||||||
|
echo MONGO_READONLY_TESTACCOUNT_TOKEN=$MONGO_READONLY_TESTACCOUNT_TOKEN >> $GITHUB_ENV
|
||||||
- name: Run test shard ${{ matrix['shardIndex'] }} of ${{ matrix['shardTotal']}}
|
- name: Run test shard ${{ matrix['shardIndex'] }} of ${{ matrix['shardTotal']}}
|
||||||
run: npx playwright test --shard=${{ matrix.shardIndex }}/${{ matrix.shardTotal }} --workers=3
|
run: npx playwright test --shard=${{ matrix.shardIndex }}/${{ matrix.shardTotal }} --workers=3
|
||||||
- name: Upload blob report to GitHub Actions Artifacts
|
- name: Upload blob report to GitHub Actions Artifacts
|
||||||
|
|||||||
2
.github/workflows/cleanup.yml
vendored
2
.github/workflows/cleanup.yml
vendored
@@ -27,7 +27,7 @@ jobs:
|
|||||||
- name: "Az CLI login"
|
- name: "Az CLI login"
|
||||||
uses: azure/login@v1
|
uses: azure/login@v1
|
||||||
with:
|
with:
|
||||||
client-id: ${{ secrets.AZURE_CLIENT_ID }}
|
client-id: ${{ secrets.E2E_TESTS_CLIENT_ID }}
|
||||||
tenant-id: ${{ secrets.AZURE_TENANT_ID }}
|
tenant-id: ${{ secrets.AZURE_TENANT_ID }}
|
||||||
subscription-id: ${{ secrets.AZURE_SUBSCRIPTION_ID }}
|
subscription-id: ${{ secrets.AZURE_SUBSCRIPTION_ID }}
|
||||||
|
|
||||||
|
|||||||
2
.npmrc
2
.npmrc
@@ -1,4 +1,4 @@
|
|||||||
save-exact=true
|
save-exact=true
|
||||||
|
|
||||||
# Ignore peer dependency conflicts
|
# Ignore peer dependency conflicts
|
||||||
force=true # TODO: Remove this when we update to React 17 or higher!
|
force=true # TODO: Remove this when we update to React 17 or higher!
|
||||||
|
|||||||
6
.vscode/settings.json
vendored
6
.vscode/settings.json
vendored
@@ -27,5 +27,11 @@
|
|||||||
"editor.defaultFormatter": "esbenp.prettier-vscode",
|
"editor.defaultFormatter": "esbenp.prettier-vscode",
|
||||||
"[typescriptreact]": {
|
"[typescriptreact]": {
|
||||||
"editor.defaultFormatter": "vscode.typescript-language-features"
|
"editor.defaultFormatter": "vscode.typescript-language-features"
|
||||||
|
},
|
||||||
|
"[typescript]": {
|
||||||
|
"editor.defaultFormatter": "vscode.typescript-language-features"
|
||||||
|
},
|
||||||
|
"[json]": {
|
||||||
|
"editor.defaultFormatter": "vscode.json-language-features"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
{
|
{
|
||||||
"JUNO_ENDPOINT": "https://tools.cosmos.azure.com",
|
"JUNO_ENDPOINT": "https://tools.cosmos.azure.com",
|
||||||
"isTerminalEnabled": true,
|
|
||||||
"isPhoenixEnabled": true
|
"isPhoenixEnabled": true
|
||||||
}
|
}
|
||||||
@@ -1,5 +1,4 @@
|
|||||||
{
|
{
|
||||||
"JUNO_ENDPOINT": "https://tools.cosmos.azure.com",
|
"JUNO_ENDPOINT": "https://tools.cosmos.azure.com",
|
||||||
"isTerminalEnabled" : false,
|
"isPhoenixEnabled": false
|
||||||
"isPhoenixEnabled" : false
|
}
|
||||||
}
|
|
||||||
1
images/AzureOpenAi.svg
Normal file
1
images/AzureOpenAi.svg
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<svg id="uuid-adbdae8e-5a41-46d1-8c18-aa73cdbfee32" xmlns="http://www.w3.org/2000/svg" width="18" height="18" viewBox="0 0 18 18"><path d="m0,2.7v12.6c0,1.491,1.209,2.7,2.7,2.7h12.6c1.491,0,2.7-1.209,2.7-2.7V2.7c0-1.491-1.209-2.7-2.7-2.7H2.7C1.209,0,0,1.209,0,2.7ZM10.8,0v3.6c0,3.976,3.224,7.2,7.2,7.2h-3.6c-3.976,0-7.199,3.222-7.2,7.198v-3.598c0-3.976-3.224-7.2-7.2-7.2h3.6c3.976,0,7.2-3.224,7.2-7.2Z" fill="#000000" stroke-width="0" /></svg>
|
||||||
|
After Width: | Height: | Size: 443 B |
17
images/ContainerCopy/copy-jobs.svg
Normal file
17
images/ContainerCopy/copy-jobs.svg
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
<svg width="96" height="104" viewBox="0 0 96 104" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path opacity="0.2" d="M80.5008 81.2203L41.2637 58.2012L35.7705 61.9941L74.6152 84.6208L80.5008 81.2203Z" fill="#AAAAAA"/>
|
||||||
|
<path opacity="0.2" d="M60.2283 92.5992L20.9912 69.5801L15.498 73.373L54.3428 95.9997L60.2283 92.5992Z" fill="#AAAAAA"/>
|
||||||
|
<path d="M63.7596 30.9969L74.8768 37.4057L74.746 82.1359L35.7705 59.7708L35.9013 3.00781L63.7596 19.095V30.9969Z" fill="#C9C9C9"/>
|
||||||
|
<path d="M35.9014 3.00818L41.0022 0L68.8605 16.0872L63.7597 19.0954L35.9014 3.00818Z" fill="#AAAAAA"/>
|
||||||
|
<path d="M74.8769 37.4067L79.9777 34.5293L79.8469 79.2596L74.7461 82.2677L74.8769 37.4067Z" fill="#AAAAAA"/>
|
||||||
|
<path d="M43.4872 42.245L54.6043 48.6537L54.4735 93.384L15.498 71.0188L15.6288 14.2559L43.4872 30.3431V42.245Z" fill="#F4F4F4"/>
|
||||||
|
<path d="M15.6289 14.2562L20.7297 11.248L48.5881 27.3352L43.4872 30.3434L15.6289 14.2562Z" fill="#DCDCDC"/>
|
||||||
|
<path d="M54.6044 48.6547L59.7052 45.7773L59.5745 90.5076L54.4736 93.5158L54.6044 48.6547Z" fill="#DCDCDC"/>
|
||||||
|
<path d="M63.7598 19.0961L68.8606 16.0879L79.9778 34.5293L74.8769 37.4067L63.7598 19.0961Z" fill="#C9C9C9"/>
|
||||||
|
<path d="M63.7598 19.0957L74.8769 37.4063L63.7598 30.9976V19.0957Z" fill="#DCDCDC"/>
|
||||||
|
<path d="M43.4873 30.3441L48.5881 27.3359L59.7053 45.7774L54.6045 48.6548L43.4873 30.3441Z" fill="#F4F4F4"/>
|
||||||
|
<path d="M43.4873 30.3438L54.6045 48.6544L43.4873 42.2457V30.3438Z" fill="#C9C9C9"/>
|
||||||
|
<path d="M46.8751 52.4595V55.9693L23.2275 42.1367V38.627L46.8751 52.4595Z" fill="#C9C9C9"/>
|
||||||
|
<path d="M46.8751 59.0658V62.5756L23.2275 48.6914V45.1816L46.8751 59.0658Z" fill="#C9C9C9"/>
|
||||||
|
<path d="M46.8751 65.3621V68.8719L23.2275 54.9877V51.6328L46.8751 65.3621Z" fill="#C9C9C9"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 1.7 KiB |
3
images/github-black-and-white.svg
Normal file
3
images/github-black-and-white.svg
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<svg width="16" height="16" viewBox="0 0 16 16" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M8 0C8.73438 0 9.44271 0.0960961 10.125 0.288288C10.8073 0.48048 11.4427 0.758091 12.0312 1.12112C12.6198 1.48415 13.1589 1.91124 13.6484 2.4024C14.138 2.89356 14.5573 3.44611 14.9062 4.06006C15.2552 4.67401 15.5234 5.32799 15.7109 6.02202C15.8984 6.71605 15.9948 7.44211 16 8.2002C16 9.08108 15.8698 9.92993 15.6094 10.7467C15.349 11.5636 14.9766 12.3136 14.4922 12.997C14.0078 13.6803 13.4323 14.2783 12.7656 14.7908C12.099 15.3033 11.3542 15.701 10.5312 15.984C10.5156 15.9893 10.4922 15.992 10.4609 15.992C10.4297 15.992 10.4062 15.9947 10.3906 16C10.2656 16 10.1667 15.9626 10.0938 15.8879C10.0208 15.8131 9.98438 15.7144 9.98438 15.5916V14.4705C9.98438 14.1021 9.98698 13.7257 9.99219 13.3413C9.99219 13.0691 9.95312 12.7941 9.875 12.5165C9.79688 12.2389 9.65625 12.0067 9.45312 11.8198C10.0573 11.7504 10.5859 11.625 11.0391 11.4434C11.4922 11.2619 11.8724 11.0057 12.1797 10.6747C12.487 10.3437 12.7161 9.94328 12.8672 9.47347C13.0182 9.00367 13.0964 8.43777 13.1016 7.77578C13.1016 7.35936 13.0339 6.96697 12.8984 6.5986C12.763 6.23023 12.5573 5.88856 12.2812 5.57357C12.3385 5.42409 12.3802 5.26927 12.4062 5.10911C12.4323 4.94895 12.4453 4.78879 12.4453 4.62863C12.4453 4.42042 12.4245 4.21488 12.3828 4.01201C12.3411 3.80914 12.2812 3.60627 12.2031 3.4034C12.1771 3.39273 12.1484 3.38739 12.1172 3.38739C12.0859 3.38739 12.0573 3.38739 12.0312 3.38739C11.8646 3.38739 11.6901 3.41408 11.5078 3.46747C11.3255 3.52085 11.1458 3.59026 10.9688 3.67568C10.7917 3.76109 10.6172 3.85452 10.4453 3.95596C10.2734 4.05739 10.125 4.15349 10 4.24424C9.34896 4.05739 8.68229 3.96396 8 3.96396C7.31771 3.96396 6.65104 4.05739 6 4.24424C5.86979 4.15349 5.72135 4.05739 5.55469 3.95596C5.38802 3.85452 5.21615 3.76376 5.03906 3.68368C4.86198 3.6036 4.67969 3.5342 4.49219 3.47548C4.30469 3.41675 4.13021 3.38739 3.96875 3.38739H3.88281C3.85156 3.38739 3.82292 3.39273 3.79688 3.4034C3.72396 3.60093 3.66667 3.80113 3.625 4.004C3.58333 4.20687 3.5599 4.41508 3.55469 4.62863C3.55469 4.78879 3.56771 4.94895 3.59375 5.10911C3.61979 5.26927 3.66146 5.42409 3.71875 5.57357C3.44271 5.88322 3.23698 6.22222 3.10156 6.59059C2.96615 6.95896 2.89844 7.35402 2.89844 7.77578C2.89844 8.42709 2.97396 8.99032 3.125 9.46547C3.27604 9.94061 3.50521 10.341 3.8125 10.6667C4.11979 10.9923 4.5 11.2513 4.95312 11.4434C5.40625 11.6356 5.9349 11.7638 6.53906 11.8278C6.38802 11.9666 6.27344 12.1321 6.19531 12.3243C6.11719 12.5165 6.0625 12.7167 6.03125 12.9249C5.89062 12.9943 5.74219 13.0477 5.58594 13.0851C5.42969 13.1225 5.27344 13.1411 5.11719 13.1411C4.78385 13.1411 4.50781 13.0611 4.28906 12.9009C4.07031 12.7407 3.875 12.5219 3.70312 12.2442C3.64062 12.1428 3.5651 12.0414 3.47656 11.9399C3.38802 11.8385 3.29167 11.7477 3.1875 11.6677C3.08333 11.5876 2.97135 11.5235 2.85156 11.4755C2.73177 11.4274 2.60677 11.4007 2.47656 11.3954H2.38281C2.34115 11.3954 2.30208 11.4034 2.26562 11.4194C2.22917 11.4354 2.19271 11.4515 2.15625 11.4675C2.11979 11.4835 2.10417 11.5102 2.10938 11.5475C2.10938 11.6116 2.14583 11.673 2.21875 11.7317C2.29167 11.7905 2.35156 11.8385 2.39844 11.8759L2.42188 11.8919C2.53646 11.9826 2.63542 12.0681 2.71875 12.1481C2.80208 12.2282 2.88021 12.3163 2.95312 12.4124C3.02604 12.5085 3.08594 12.6099 3.13281 12.7167C3.17969 12.8235 3.23958 12.9489 3.3125 13.0931C3.48958 13.5095 3.73698 13.8111 4.05469 13.998C4.3724 14.1849 4.75521 14.2809 5.20312 14.2863C5.33854 14.2863 5.47396 14.2783 5.60938 14.2623C5.74479 14.2462 5.88021 14.2222 6.01562 14.1902V15.5836C6.01562 15.7117 5.97917 15.8131 5.90625 15.8879C5.83333 15.9626 5.73177 16 5.60156 16H5.53906C5.51302 16 5.48958 15.9947 5.46875 15.984C4.65104 15.7117 3.90625 15.3193 3.23438 14.8068C2.5625 14.2943 1.98698 13.6937 1.50781 13.005C1.02865 12.3163 0.658854 11.5636 0.398438 10.7467C0.138021 9.92993 0.00520833 9.08108 0 8.2002C0 7.44745 0.09375 6.72139 0.28125 6.02202C0.46875 5.32266 0.739583 4.67134 1.09375 4.06807C1.44792 3.4648 1.86458 2.91225 2.34375 2.41041C2.82292 1.90858 3.36198 1.47881 3.96094 1.12112C4.5599 0.76343 5.19792 0.488488 5.875 0.296296C6.55208 0.104104 7.26042 0.00533867 8 0Z" fill="#000000"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 4.1 KiB |
37
package-lock.json
generated
37
package-lock.json
generated
@@ -10,7 +10,7 @@
|
|||||||
"hasInstallScript": true,
|
"hasInstallScript": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@azure/arm-cosmosdb": "9.1.0",
|
"@azure/arm-cosmosdb": "9.1.0",
|
||||||
"@azure/cosmos": "4.3.0",
|
"@azure/cosmos": "4.5.0",
|
||||||
"@azure/cosmos-language-service": "0.0.5",
|
"@azure/cosmos-language-service": "0.0.5",
|
||||||
"@azure/identity": "4.5.0",
|
"@azure/identity": "4.5.0",
|
||||||
"@azure/msal-browser": "2.14.2",
|
"@azure/msal-browser": "2.14.2",
|
||||||
@@ -391,24 +391,25 @@
|
|||||||
"license": "0BSD"
|
"license": "0BSD"
|
||||||
},
|
},
|
||||||
"node_modules/@azure/cosmos": {
|
"node_modules/@azure/cosmos": {
|
||||||
"version": "4.3.0",
|
"version": "4.5.0",
|
||||||
"resolved": "https://registry.npmjs.org/@azure/cosmos/-/cosmos-4.3.0.tgz",
|
"resolved": "https://registry.npmjs.org/@azure/cosmos/-/cosmos-4.5.0.tgz",
|
||||||
"integrity": "sha512-0Ls3l1uWBBSphx6YRhnM+w7rSvq8qVugBCdO6kSiNuRYXEf6+YWLjbzz4e7L2kkz/6ScFdZIOJYP+XtkiRYOhA==",
|
"integrity": "sha512-JsTh4twb6FcwP7rJwxQiNZQ/LGtuF6gmciaxY9Rnp6/A325Lhsw/SH4R2ArpT0yCvozbZpweIwdPfUkXVBtp5w==",
|
||||||
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@azure/abort-controller": "^2.0.0",
|
"@azure/abort-controller": "^2.1.2",
|
||||||
"@azure/core-auth": "^1.7.1",
|
"@azure/core-auth": "^1.9.0",
|
||||||
"@azure/core-rest-pipeline": "^1.15.1",
|
"@azure/core-rest-pipeline": "^1.19.1",
|
||||||
"@azure/core-tracing": "^1.1.1",
|
"@azure/core-tracing": "^1.2.0",
|
||||||
"@azure/core-util": "^1.8.1",
|
"@azure/core-util": "^1.11.0",
|
||||||
"@azure/keyvault-keys": "^4.8.0",
|
"@azure/keyvault-keys": "^4.9.0",
|
||||||
|
"@azure/logger": "^1.1.4",
|
||||||
"fast-json-stable-stringify": "^2.1.0",
|
"fast-json-stable-stringify": "^2.1.0",
|
||||||
"jsbi": "^4.3.0",
|
|
||||||
"priorityqueuejs": "^2.0.0",
|
"priorityqueuejs": "^2.0.0",
|
||||||
"semaphore": "^1.1.0",
|
"semaphore": "^1.1.0",
|
||||||
"tslib": "^2.6.2"
|
"tslib": "^2.8.1"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=18.0.0"
|
"node": ">=20.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@azure/cosmos-language-service": {
|
"node_modules/@azure/cosmos-language-service": {
|
||||||
@@ -438,8 +439,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@azure/cosmos/node_modules/tslib": {
|
"node_modules/@azure/cosmos/node_modules/tslib": {
|
||||||
"version": "2.6.2",
|
"version": "2.8.1",
|
||||||
"license": "0BSD"
|
"resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz",
|
||||||
|
"integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w=="
|
||||||
},
|
},
|
||||||
"node_modules/@azure/identity": {
|
"node_modules/@azure/identity": {
|
||||||
"version": "4.5.0",
|
"version": "4.5.0",
|
||||||
@@ -27178,11 +27180,6 @@
|
|||||||
"js-yaml": "bin/js-yaml.js"
|
"js-yaml": "bin/js-yaml.js"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/jsbi": {
|
|
||||||
"version": "4.3.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/jsbi/-/jsbi-4.3.0.tgz",
|
|
||||||
"integrity": "sha512-SnZNcinB4RIcnEyZqFPdGPVgrg2AcnykiBy0sHVJQKHYeaLUvi3Exj+iaPpLnFVkDPZIV4U0yvgC9/R4uEAZ9g=="
|
|
||||||
},
|
|
||||||
"node_modules/jsbn": {
|
"node_modules/jsbn": {
|
||||||
"version": "0.1.1",
|
"version": "0.1.1",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@azure/arm-cosmosdb": "9.1.0",
|
"@azure/arm-cosmosdb": "9.1.0",
|
||||||
"@azure/cosmos": "4.3.0",
|
"@azure/cosmos": "4.5.0",
|
||||||
"@azure/cosmos-language-service": "0.0.5",
|
"@azure/cosmos-language-service": "0.0.5",
|
||||||
"@azure/identity": "4.5.0",
|
"@azure/identity": "4.5.0",
|
||||||
"@azure/msal-browser": "2.14.2",
|
"@azure/msal-browser": "2.14.2",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[defaults]
|
[defaults]
|
||||||
group = dataexplorer-preview
|
group = dataexplorer-preview
|
||||||
sku = P1V2
|
sku = P1v2
|
||||||
appserviceplan = dataexplorer-preview
|
appserviceplan = dataexplorer-preview
|
||||||
location = westus2
|
location = westus2
|
||||||
web = dataexplorer-preview
|
web = dataexplorer-preview
|
||||||
|
|||||||
36205
preview/package-lock.json
generated
36205
preview/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@@ -4,16 +4,18 @@
|
|||||||
"description": "",
|
"description": "",
|
||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"deploy": "az webapp up --name \"dataexplorer-preview\" --subscription \"cosmosdb-portalteam-runners\" --resource-group \"dataexplorer-preview\" --runtime \"NODE:18-lts\" --sku P1V2",
|
"deploy": "az webapp up --name \"dataexplorer-preview\" --subscription \"cosmosdb-portalteam-runners\" --resource-group \"dataexplorer-preview\" --runtime \"NODE:20-lts\" --sku P1V2",
|
||||||
"start": "node index.js",
|
"start": "node index.js",
|
||||||
"test": "echo \"Error: no test specified\" && exit 1"
|
"test": "echo \"Error: no test specified\" && exit 1"
|
||||||
},
|
},
|
||||||
"keywords": [],
|
"keywords": [],
|
||||||
"author": "Microsoft Corporation",
|
"author": "Microsoft Corporation",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"express": "^4.17.1",
|
"body-parser": "^1.20.3",
|
||||||
|
"express": "^4.21.2",
|
||||||
"http-proxy-middleware": "^3.0.3",
|
"http-proxy-middleware": "^3.0.3",
|
||||||
"node": "^18.20.6",
|
"node": "^20.19.5",
|
||||||
"node-fetch": "^2.6.1"
|
"node-fetch": "^2.6.1",
|
||||||
|
"path-to-regexp": "^0.1.12"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
65616
sampleData/fabricSampleData.json
Normal file
65616
sampleData/fabricSampleData.json
Normal file
File diff suppressed because it is too large
Load Diff
322616
sampleData/fabricSampleDataVectors.json
Normal file
322616
sampleData/fabricSampleDataVectors.json
Normal file
File diff suppressed because it is too large
Load Diff
@@ -138,13 +138,12 @@ export enum MongoBackendEndpointType {
|
|||||||
remote,
|
remote,
|
||||||
}
|
}
|
||||||
|
|
||||||
export class BackendApi {
|
export class AadScopeEndpoints {
|
||||||
public static readonly GenerateToken: string = "GenerateToken";
|
public static readonly Development: string = "https://cosmos.azure.com";
|
||||||
public static readonly PortalSettings: string = "PortalSettings";
|
public static readonly MPAC: string = "https://cosmos.azure.com";
|
||||||
public static readonly AccountRestrictions: string = "AccountRestrictions";
|
public static readonly Prod: string = "https://cosmos.azure.com";
|
||||||
public static readonly RuntimeProxy: string = "RuntimeProxy";
|
public static readonly Fairfax: string = "https://cosmos.azure.us";
|
||||||
public static readonly DisallowedLocations: string = "DisallowedLocations";
|
public static readonly Mooncake: string = "https://cosmos.azure.cn";
|
||||||
public static readonly SampleData: string = "SampleData";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export class PortalBackendEndpoints {
|
export class PortalBackendEndpoints {
|
||||||
@@ -264,6 +263,7 @@ export class HttpHeaders {
|
|||||||
public static activityId: string = "x-ms-activity-id";
|
public static activityId: string = "x-ms-activity-id";
|
||||||
public static apiType: string = "x-ms-cosmos-apitype";
|
public static apiType: string = "x-ms-cosmos-apitype";
|
||||||
public static authorization: string = "authorization";
|
public static authorization: string = "authorization";
|
||||||
|
public static entraIdToken: string = "x-ms-entraid-token";
|
||||||
public static collectionIndexTransformationProgress: string =
|
public static collectionIndexTransformationProgress: string =
|
||||||
"x-ms-documentdb-collection-index-transformation-progress";
|
"x-ms-documentdb-collection-index-transformation-progress";
|
||||||
public static continuation: string = "x-ms-continuation";
|
public static continuation: string = "x-ms-continuation";
|
||||||
@@ -774,3 +774,10 @@ export const ShortenedQueryCopilotSampleContainerSchema = {
|
|||||||
|
|
||||||
userPrompt: "find all products",
|
userPrompt: "find all products",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export enum MongoGuidRepresentation {
|
||||||
|
Standard = "Standard",
|
||||||
|
CSharpLegacy = "CSharpLegacy",
|
||||||
|
JavaLegacy = "JavaLegacy",
|
||||||
|
PythonLegacy = "PythonLegacy",
|
||||||
|
}
|
||||||
|
|||||||
@@ -4,12 +4,12 @@ import { CosmosDbArtifactType } from "Contracts/FabricMessagesContract";
|
|||||||
import { AuthorizationToken } from "Contracts/FabricMessageTypes";
|
import { AuthorizationToken } from "Contracts/FabricMessageTypes";
|
||||||
import { checkDatabaseResourceTokensValidity, isFabricMirroredKey } from "Platform/Fabric/FabricUtil";
|
import { checkDatabaseResourceTokensValidity, isFabricMirroredKey } from "Platform/Fabric/FabricUtil";
|
||||||
import { LocalStorageUtility, StorageKey } from "Shared/StorageUtility";
|
import { LocalStorageUtility, StorageKey } from "Shared/StorageUtility";
|
||||||
|
import { useDataplaneRbacAuthorization } from "Utils/AuthorizationUtils";
|
||||||
import { AuthType } from "../AuthType";
|
import { AuthType } from "../AuthType";
|
||||||
import { PriorityLevel } from "../Common/Constants";
|
import { PriorityLevel } from "../Common/Constants";
|
||||||
import * as Logger from "../Common/Logger";
|
import * as Logger from "../Common/Logger";
|
||||||
import { Platform, configContext } from "../ConfigContext";
|
import { Platform, configContext } from "../ConfigContext";
|
||||||
import { FabricArtifactInfo, updateUserContext, userContext } from "../UserContext";
|
import { FabricArtifactInfo, updateUserContext, userContext } from "../UserContext";
|
||||||
import { isDataplaneRbacSupported } from "../Utils/APITypeUtils";
|
|
||||||
import { logConsoleError } from "../Utils/NotificationConsoleUtils";
|
import { logConsoleError } from "../Utils/NotificationConsoleUtils";
|
||||||
import * as PriorityBasedExecutionUtils from "../Utils/PriorityBasedExecutionUtils";
|
import * as PriorityBasedExecutionUtils from "../Utils/PriorityBasedExecutionUtils";
|
||||||
import { EmulatorMasterKey, HttpHeaders } from "./Constants";
|
import { EmulatorMasterKey, HttpHeaders } from "./Constants";
|
||||||
@@ -20,8 +20,7 @@ const _global = typeof self === "undefined" ? window : self;
|
|||||||
export const tokenProvider = async (requestInfo: Cosmos.RequestInfo) => {
|
export const tokenProvider = async (requestInfo: Cosmos.RequestInfo) => {
|
||||||
const { verb, resourceId, resourceType, headers } = requestInfo;
|
const { verb, resourceId, resourceType, headers } = requestInfo;
|
||||||
|
|
||||||
const dataPlaneRBACOptionEnabled = userContext.dataPlaneRbacEnabled && isDataplaneRbacSupported(userContext.apiType);
|
if (useDataplaneRbacAuthorization(userContext)) {
|
||||||
if (userContext.features.enableAadDataPlane || dataPlaneRBACOptionEnabled) {
|
|
||||||
Logger.logInfo(
|
Logger.logInfo(
|
||||||
`AAD Data Plane Feature flag set to ${userContext.features.enableAadDataPlane} for account with disable local auth ${userContext.databaseAccount.properties.disableLocalAuth} `,
|
`AAD Data Plane Feature flag set to ${userContext.features.enableAadDataPlane} for account with disable local auth ${userContext.databaseAccount.properties.disableLocalAuth} `,
|
||||||
"Explorer/tokenProvider",
|
"Explorer/tokenProvider",
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { TagNames, WorkloadType } from "Common/Constants";
|
import { TagNames, WorkloadType } from "Common/Constants";
|
||||||
import { Tags } from "Contracts/DataModels";
|
import { Tags } from "Contracts/DataModels";
|
||||||
import { isFabric } from "Platform/Fabric/FabricUtil";
|
import { isFabric } from "Platform/Fabric/FabricUtil";
|
||||||
import { userContext } from "../UserContext";
|
import { ApiType, userContext } from "../UserContext";
|
||||||
|
|
||||||
function isVirtualNetworkFilterEnabled() {
|
function isVirtualNetworkFilterEnabled() {
|
||||||
return userContext.databaseAccount?.properties?.isVirtualNetworkFilterEnabled;
|
return userContext.databaseAccount?.properties?.isVirtualNetworkFilterEnabled;
|
||||||
@@ -33,3 +33,33 @@ export function isGlobalSecondaryIndexEnabled(): boolean {
|
|||||||
!isFabric() && userContext.apiType === "SQL" && userContext.databaseAccount?.properties?.enableMaterializedViews
|
!isFabric() && userContext.apiType === "SQL" && userContext.databaseAccount?.properties?.enableMaterializedViews
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const getDatabaseEndpoint = (apiType: ApiType): string => {
|
||||||
|
switch (apiType) {
|
||||||
|
case "Mongo":
|
||||||
|
return "mongodbDatabases";
|
||||||
|
case "Cassandra":
|
||||||
|
return "cassandraKeyspaces";
|
||||||
|
case "Gremlin":
|
||||||
|
return "gremlinDatabases";
|
||||||
|
case "Tables":
|
||||||
|
return "tables";
|
||||||
|
default:
|
||||||
|
case "SQL":
|
||||||
|
return "sqlDatabases";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getCollectionEndpoint = (apiType: ApiType): string => {
|
||||||
|
switch (apiType) {
|
||||||
|
case "Mongo":
|
||||||
|
return "collections";
|
||||||
|
case "Cassandra":
|
||||||
|
return "tables";
|
||||||
|
case "Gremlin":
|
||||||
|
return "graphs";
|
||||||
|
default:
|
||||||
|
case "SQL":
|
||||||
|
return "containers";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|||||||
@@ -28,3 +28,39 @@ describe("Environment Utility Test", () => {
|
|||||||
expect(EnvironmentUtility.getEnvironment()).toBe(EnvironmentUtility.Environment.Development);
|
expect(EnvironmentUtility.getEnvironment()).toBe(EnvironmentUtility.Environment.Development);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
describe("normalizeArmEndpoint", () => {
|
||||||
|
it("should append '/' if not present", () => {
|
||||||
|
expect(EnvironmentUtility.normalizeArmEndpoint("https://example.com")).toBe("https://example.com/");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return the same uri if '/' is present at the end", () => {
|
||||||
|
expect(EnvironmentUtility.normalizeArmEndpoint("https://example.com/")).toBe("https://example.com/");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should handle empty string", () => {
|
||||||
|
expect(EnvironmentUtility.normalizeArmEndpoint("")).toBe("");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("getEnvironment", () => {
|
||||||
|
it("should return Prod environment", () => {
|
||||||
|
updateConfigContext({
|
||||||
|
PORTAL_BACKEND_ENDPOINT: PortalBackendEndpoints.Prod,
|
||||||
|
});
|
||||||
|
expect(EnvironmentUtility.getEnvironment()).toBe(EnvironmentUtility.Environment.Prod);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return Fairfax environment", () => {
|
||||||
|
updateConfigContext({
|
||||||
|
PORTAL_BACKEND_ENDPOINT: PortalBackendEndpoints.Fairfax,
|
||||||
|
});
|
||||||
|
expect(EnvironmentUtility.getEnvironment()).toBe(EnvironmentUtility.Environment.Fairfax);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return Mooncake environment", () => {
|
||||||
|
updateConfigContext({
|
||||||
|
PORTAL_BACKEND_ENDPOINT: PortalBackendEndpoints.Mooncake,
|
||||||
|
});
|
||||||
|
expect(EnvironmentUtility.getEnvironment()).toBe(EnvironmentUtility.Environment.Mooncake);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import { PortalBackendEndpoints } from "Common/Constants";
|
import { AadScopeEndpoints, PortalBackendEndpoints } from "Common/Constants";
|
||||||
|
import * as Logger from "Common/Logger";
|
||||||
import { configContext } from "ConfigContext";
|
import { configContext } from "ConfigContext";
|
||||||
|
|
||||||
export function normalizeArmEndpoint(uri: string): string {
|
export function normalizeArmEndpoint(uri: string): string {
|
||||||
@@ -27,3 +28,17 @@ export const getEnvironment = (): Environment => {
|
|||||||
|
|
||||||
return environmentMap[configContext.PORTAL_BACKEND_ENDPOINT];
|
return environmentMap[configContext.PORTAL_BACKEND_ENDPOINT];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const getEnvironmentScopeEndpoint = (): string => {
|
||||||
|
const environment = getEnvironment();
|
||||||
|
const endpoint = AadScopeEndpoints[environment];
|
||||||
|
if (!endpoint) {
|
||||||
|
throw new Error("Cannot determine AAD scope endpoint");
|
||||||
|
}
|
||||||
|
const hrefEndpoint = new URL(endpoint).href.replace(/\/+$/, "/.default");
|
||||||
|
Logger.logInfo(
|
||||||
|
`Using AAD scope endpoint: ${hrefEndpoint}, Environment: ${environment}`,
|
||||||
|
"EnvironmentUtility/getEnvironmentScopeEndpoint",
|
||||||
|
);
|
||||||
|
return hrefEndpoint;
|
||||||
|
};
|
||||||
|
|||||||
@@ -65,7 +65,6 @@ describe("MongoProxyClient", () => {
|
|||||||
});
|
});
|
||||||
updateConfigContext({
|
updateConfigContext({
|
||||||
MONGO_PROXY_ENDPOINT: MongoProxyEndpoints.Prod,
|
MONGO_PROXY_ENDPOINT: MongoProxyEndpoints.Prod,
|
||||||
globallyEnabledMongoAPIs: [],
|
|
||||||
});
|
});
|
||||||
window.fetch = jest.fn().mockImplementation(fetchMock);
|
window.fetch = jest.fn().mockImplementation(fetchMock);
|
||||||
});
|
});
|
||||||
@@ -84,7 +83,6 @@ describe("MongoProxyClient", () => {
|
|||||||
it("builds the correct proxy URL in development", () => {
|
it("builds the correct proxy URL in development", () => {
|
||||||
updateConfigContext({
|
updateConfigContext({
|
||||||
MONGO_PROXY_ENDPOINT: "https://localhost:1234",
|
MONGO_PROXY_ENDPOINT: "https://localhost:1234",
|
||||||
globallyEnabledMongoAPIs: [],
|
|
||||||
});
|
});
|
||||||
queryDocuments(databaseId, collection, true, "{}");
|
queryDocuments(databaseId, collection, true, "{}");
|
||||||
expect(window.fetch).toHaveBeenCalledWith(
|
expect(window.fetch).toHaveBeenCalledWith(
|
||||||
@@ -101,7 +99,6 @@ describe("MongoProxyClient", () => {
|
|||||||
});
|
});
|
||||||
updateConfigContext({
|
updateConfigContext({
|
||||||
MONGO_PROXY_ENDPOINT: MongoProxyEndpoints.Prod,
|
MONGO_PROXY_ENDPOINT: MongoProxyEndpoints.Prod,
|
||||||
globallyEnabledMongoAPIs: [],
|
|
||||||
});
|
});
|
||||||
window.fetch = jest.fn().mockImplementation(fetchMock);
|
window.fetch = jest.fn().mockImplementation(fetchMock);
|
||||||
});
|
});
|
||||||
@@ -120,7 +117,6 @@ describe("MongoProxyClient", () => {
|
|||||||
it("builds the correct proxy URL in development", () => {
|
it("builds the correct proxy URL in development", () => {
|
||||||
updateConfigContext({
|
updateConfigContext({
|
||||||
MONGO_PROXY_ENDPOINT: "https://localhost:1234",
|
MONGO_PROXY_ENDPOINT: "https://localhost:1234",
|
||||||
globallyEnabledMongoAPIs: [],
|
|
||||||
});
|
});
|
||||||
readDocument(databaseId, collection, documentId);
|
readDocument(databaseId, collection, documentId);
|
||||||
expect(window.fetch).toHaveBeenCalledWith(
|
expect(window.fetch).toHaveBeenCalledWith(
|
||||||
@@ -137,7 +133,6 @@ describe("MongoProxyClient", () => {
|
|||||||
});
|
});
|
||||||
updateConfigContext({
|
updateConfigContext({
|
||||||
MONGO_PROXY_ENDPOINT: MongoProxyEndpoints.Prod,
|
MONGO_PROXY_ENDPOINT: MongoProxyEndpoints.Prod,
|
||||||
globallyEnabledMongoAPIs: [],
|
|
||||||
});
|
});
|
||||||
window.fetch = jest.fn().mockImplementation(fetchMock);
|
window.fetch = jest.fn().mockImplementation(fetchMock);
|
||||||
});
|
});
|
||||||
@@ -156,7 +151,6 @@ describe("MongoProxyClient", () => {
|
|||||||
it("builds the correct proxy URL in development", () => {
|
it("builds the correct proxy URL in development", () => {
|
||||||
updateConfigContext({
|
updateConfigContext({
|
||||||
MONGO_PROXY_ENDPOINT: "https://localhost:1234",
|
MONGO_PROXY_ENDPOINT: "https://localhost:1234",
|
||||||
globallyEnabledMongoAPIs: [],
|
|
||||||
});
|
});
|
||||||
readDocument(databaseId, collection, documentId);
|
readDocument(databaseId, collection, documentId);
|
||||||
expect(window.fetch).toHaveBeenCalledWith(
|
expect(window.fetch).toHaveBeenCalledWith(
|
||||||
@@ -173,7 +167,6 @@ describe("MongoProxyClient", () => {
|
|||||||
});
|
});
|
||||||
updateConfigContext({
|
updateConfigContext({
|
||||||
MONGO_PROXY_ENDPOINT: MongoProxyEndpoints.Prod,
|
MONGO_PROXY_ENDPOINT: MongoProxyEndpoints.Prod,
|
||||||
globallyEnabledMongoAPIs: [],
|
|
||||||
});
|
});
|
||||||
window.fetch = jest.fn().mockImplementation(fetchMock);
|
window.fetch = jest.fn().mockImplementation(fetchMock);
|
||||||
});
|
});
|
||||||
@@ -197,7 +190,6 @@ describe("MongoProxyClient", () => {
|
|||||||
});
|
});
|
||||||
updateConfigContext({
|
updateConfigContext({
|
||||||
MONGO_PROXY_ENDPOINT: MongoProxyEndpoints.Prod,
|
MONGO_PROXY_ENDPOINT: MongoProxyEndpoints.Prod,
|
||||||
globallyEnabledMongoAPIs: [],
|
|
||||||
});
|
});
|
||||||
window.fetch = jest.fn().mockImplementation(fetchMock);
|
window.fetch = jest.fn().mockImplementation(fetchMock);
|
||||||
});
|
});
|
||||||
@@ -216,7 +208,6 @@ describe("MongoProxyClient", () => {
|
|||||||
it("builds the correct proxy URL in development", () => {
|
it("builds the correct proxy URL in development", () => {
|
||||||
updateConfigContext({
|
updateConfigContext({
|
||||||
MONGO_PROXY_ENDPOINT: "https://localhost:1234",
|
MONGO_PROXY_ENDPOINT: "https://localhost:1234",
|
||||||
globallyEnabledMongoAPIs: [],
|
|
||||||
});
|
});
|
||||||
deleteDocuments(databaseId, collection, [documentId]);
|
deleteDocuments(databaseId, collection, [documentId]);
|
||||||
expect(window.fetch).toHaveBeenCalledWith(
|
expect(window.fetch).toHaveBeenCalledWith(
|
||||||
@@ -233,7 +224,6 @@ describe("MongoProxyClient", () => {
|
|||||||
});
|
});
|
||||||
updateConfigContext({
|
updateConfigContext({
|
||||||
MONGO_PROXY_ENDPOINT: MongoProxyEndpoints.Prod,
|
MONGO_PROXY_ENDPOINT: MongoProxyEndpoints.Prod,
|
||||||
globallyEnabledMongoAPIs: [],
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import { Constants as CosmosSDKConstants } from "@azure/cosmos";
|
import { Constants as CosmosSDKConstants } from "@azure/cosmos";
|
||||||
|
import { getMongoGuidRepresentation } from "Shared/StorageUtility";
|
||||||
import { AuthType } from "../AuthType";
|
import { AuthType } from "../AuthType";
|
||||||
import { configContext } from "../ConfigContext";
|
import { configContext } from "../ConfigContext";
|
||||||
import * as DataModels from "../Contracts/DataModels";
|
import * as DataModels from "../Contracts/DataModels";
|
||||||
@@ -6,6 +7,7 @@ import { MessageTypes } from "../Contracts/ExplorerContracts";
|
|||||||
import { Collection } from "../Contracts/ViewModels";
|
import { Collection } from "../Contracts/ViewModels";
|
||||||
import DocumentId from "../Explorer/Tree/DocumentId";
|
import DocumentId from "../Explorer/Tree/DocumentId";
|
||||||
import { userContext } from "../UserContext";
|
import { userContext } from "../UserContext";
|
||||||
|
import { isDataplaneRbacEnabledForProxyApi } from "../Utils/AuthorizationUtils";
|
||||||
import { logConsoleError } from "../Utils/NotificationConsoleUtils";
|
import { logConsoleError } from "../Utils/NotificationConsoleUtils";
|
||||||
import { ApiType, ContentType, HttpHeaders, HttpStatusCodes } from "./Constants";
|
import { ApiType, ContentType, HttpHeaders, HttpStatusCodes } from "./Constants";
|
||||||
import { MinimalQueryIterator } from "./IteratorUtilities";
|
import { MinimalQueryIterator } from "./IteratorUtilities";
|
||||||
@@ -21,7 +23,13 @@ function authHeaders() {
|
|||||||
if (userContext.authType === AuthType.EncryptedToken) {
|
if (userContext.authType === AuthType.EncryptedToken) {
|
||||||
return { [HttpHeaders.guestAccessToken]: userContext.accessToken };
|
return { [HttpHeaders.guestAccessToken]: userContext.accessToken };
|
||||||
} else {
|
} else {
|
||||||
return { [HttpHeaders.authorization]: userContext.authorizationToken };
|
const headers: { [key: string]: string } = {
|
||||||
|
[HttpHeaders.authorization]: userContext.authorizationToken,
|
||||||
|
};
|
||||||
|
if (isDataplaneRbacEnabledForProxyApi(userContext)) {
|
||||||
|
headers[HttpHeaders.entraIdToken] = userContext.aadToken;
|
||||||
|
}
|
||||||
|
return headers;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -139,6 +147,9 @@ export function readDocument(
|
|||||||
documentId && documentId.partitionKey && !documentId.partitionKey.systemKey
|
documentId && documentId.partitionKey && !documentId.partitionKey.systemKey
|
||||||
? documentId.partitionKeyProperties?.[0]
|
? documentId.partitionKeyProperties?.[0]
|
||||||
: "",
|
: "",
|
||||||
|
clientSettings: {
|
||||||
|
guidRepresentation: getMongoGuidRepresentation(),
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const endpoint = getEndpoint(configContext.MONGO_PROXY_ENDPOINT);
|
const endpoint = getEndpoint(configContext.MONGO_PROXY_ENDPOINT);
|
||||||
@@ -181,6 +192,9 @@ export function createDocument(
|
|||||||
partitionKey:
|
partitionKey:
|
||||||
collection && collection.partitionKey && !collection.partitionKey.systemKey ? partitionKeyProperty : "",
|
collection && collection.partitionKey && !collection.partitionKey.systemKey ? partitionKeyProperty : "",
|
||||||
documentContent: JSON.stringify(documentContent),
|
documentContent: JSON.stringify(documentContent),
|
||||||
|
clientSettings: {
|
||||||
|
guidRepresentation: getMongoGuidRepresentation(),
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const endpoint = getEndpoint(configContext.MONGO_PROXY_ENDPOINT);
|
const endpoint = getEndpoint(configContext.MONGO_PROXY_ENDPOINT);
|
||||||
@@ -228,6 +242,9 @@ export function updateDocument(
|
|||||||
? documentId.partitionKeyProperties?.[0]
|
? documentId.partitionKeyProperties?.[0]
|
||||||
: "",
|
: "",
|
||||||
documentContent,
|
documentContent,
|
||||||
|
clientSettings: {
|
||||||
|
guidRepresentation: getMongoGuidRepresentation(),
|
||||||
|
},
|
||||||
};
|
};
|
||||||
const endpoint = getEndpoint(configContext.MONGO_PROXY_ENDPOINT);
|
const endpoint = getEndpoint(configContext.MONGO_PROXY_ENDPOINT);
|
||||||
|
|
||||||
@@ -274,6 +291,9 @@ export function deleteDocuments(
|
|||||||
subscriptionID: userContext.subscriptionId,
|
subscriptionID: userContext.subscriptionId,
|
||||||
resourceGroup: userContext.resourceGroup,
|
resourceGroup: userContext.resourceGroup,
|
||||||
databaseAccountName: databaseAccount.name,
|
databaseAccountName: databaseAccount.name,
|
||||||
|
clientSettings: {
|
||||||
|
guidRepresentation: getMongoGuidRepresentation(),
|
||||||
|
},
|
||||||
};
|
};
|
||||||
const endpoint = getEndpoint(configContext.MONGO_PROXY_ENDPOINT);
|
const endpoint = getEndpoint(configContext.MONGO_PROXY_ENDPOINT);
|
||||||
|
|
||||||
|
|||||||
40
src/Common/ShimmerTree/index.tsx
Normal file
40
src/Common/ShimmerTree/index.tsx
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
import { Shimmer, ShimmerElementType, Stack } from "@fluentui/react";
|
||||||
|
import * as React from "react";
|
||||||
|
|
||||||
|
export interface IndentLevel {
|
||||||
|
level: number;
|
||||||
|
width?: string;
|
||||||
|
}
|
||||||
|
interface ShimmerTreeProps {
|
||||||
|
indentLevels: IndentLevel[];
|
||||||
|
style?: React.CSSProperties;
|
||||||
|
}
|
||||||
|
|
||||||
|
const ShimmerTree = ({ indentLevels, style = {} }: ShimmerTreeProps) => {
|
||||||
|
/**
|
||||||
|
* indentLevels - Array of indent levels for shimmer tree
|
||||||
|
* 0 - Root
|
||||||
|
* 1 - Level 1
|
||||||
|
* 2 - Level 2
|
||||||
|
* 3 - Level 3
|
||||||
|
* n - Level n
|
||||||
|
* */
|
||||||
|
const renderShimmers = (indent: IndentLevel) => (
|
||||||
|
<Shimmer
|
||||||
|
key={Math.random()}
|
||||||
|
shimmerElements={[
|
||||||
|
{ type: ShimmerElementType.gap, width: `${indent.level * 20}px` }, // Indent for hierarchy
|
||||||
|
{ type: ShimmerElementType.line, height: 16, width: indent.width || "100%" },
|
||||||
|
]}
|
||||||
|
style={{ marginBottom: 8 }}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Stack tokens={{ childrenGap: 8 }} style={{ width: "50%", ...style }} data-testid="shimmer-stack">
|
||||||
|
{indentLevels.map((indentLevel: IndentLevel) => renderShimmers(indentLevel))}
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default ShimmerTree;
|
||||||
@@ -2,7 +2,6 @@
|
|||||||
|
|
||||||
exports[`getCommonQueryOptions builds the correct default options objects 1`] = `
|
exports[`getCommonQueryOptions builds the correct default options objects 1`] = `
|
||||||
{
|
{
|
||||||
"disableNonStreamingOrderByQuery": true,
|
|
||||||
"enableQueryControl": false,
|
"enableQueryControl": false,
|
||||||
"enableScanInQuery": true,
|
"enableScanInQuery": true,
|
||||||
"forceQueryPlan": true,
|
"forceQueryPlan": true,
|
||||||
@@ -14,7 +13,6 @@ exports[`getCommonQueryOptions builds the correct default options objects 1`] =
|
|||||||
|
|
||||||
exports[`getCommonQueryOptions reads from localStorage 1`] = `
|
exports[`getCommonQueryOptions reads from localStorage 1`] = `
|
||||||
{
|
{
|
||||||
"disableNonStreamingOrderByQuery": true,
|
|
||||||
"enableQueryControl": false,
|
"enableQueryControl": false,
|
||||||
"enableScanInQuery": true,
|
"enableScanInQuery": true,
|
||||||
"forceQueryPlan": true,
|
"forceQueryPlan": true,
|
||||||
|
|||||||
@@ -1,4 +1,6 @@
|
|||||||
import { ContainerRequest, ContainerResponse, DatabaseRequest, DatabaseResponse, RequestOptions } from "@azure/cosmos";
|
import { ContainerRequest, ContainerResponse, DatabaseRequest, DatabaseResponse, RequestOptions } from "@azure/cosmos";
|
||||||
|
import { sendMessage } from "Common/MessageHandler";
|
||||||
|
import { FabricMessageTypes } from "Contracts/FabricMessageTypes";
|
||||||
import { isFabricNative } from "Platform/Fabric/FabricUtil";
|
import { isFabricNative } from "Platform/Fabric/FabricUtil";
|
||||||
import { AuthType } from "../../AuthType";
|
import { AuthType } from "../../AuthType";
|
||||||
import * as DataModels from "../../Contracts/DataModels";
|
import * as DataModels from "../../Contracts/DataModels";
|
||||||
@@ -43,6 +45,14 @@ export const createCollection = async (params: DataModels.CreateCollectionParams
|
|||||||
}
|
}
|
||||||
|
|
||||||
logConsoleInfo(`Successfully created container ${params.collectionId}`);
|
logConsoleInfo(`Successfully created container ${params.collectionId}`);
|
||||||
|
|
||||||
|
if (isFabricNative()) {
|
||||||
|
sendMessage({
|
||||||
|
type: FabricMessageTypes.ContainerUpdated,
|
||||||
|
params: { updateType: "created" },
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
return collection;
|
return collection;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
handleError(error, "CreateCollection", `Error while creating container ${params.collectionId}`);
|
handleError(error, "CreateCollection", `Error while creating container ${params.collectionId}`);
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
import { sendMessage } from "Common/MessageHandler";
|
||||||
|
import { FabricMessageTypes } from "Contracts/FabricMessageTypes";
|
||||||
import { isFabric } from "Platform/Fabric/FabricUtil";
|
import { isFabric } from "Platform/Fabric/FabricUtil";
|
||||||
import { AuthType } from "../../AuthType";
|
import { AuthType } from "../../AuthType";
|
||||||
import { userContext } from "../../UserContext";
|
import { userContext } from "../../UserContext";
|
||||||
@@ -19,6 +21,11 @@ export async function deleteCollection(databaseId: string, collectionId: string)
|
|||||||
await client().database(databaseId).container(collectionId).delete();
|
await client().database(databaseId).container(collectionId).delete();
|
||||||
}
|
}
|
||||||
logConsoleInfo(`Successfully deleted container ${collectionId}`);
|
logConsoleInfo(`Successfully deleted container ${collectionId}`);
|
||||||
|
|
||||||
|
sendMessage({
|
||||||
|
type: FabricMessageTypes.ContainerUpdated,
|
||||||
|
params: { updateType: "deleted" },
|
||||||
|
});
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
handleError(error, "DeleteCollection", `Error while deleting container ${collectionId}`);
|
handleError(error, "DeleteCollection", `Error while deleting container ${collectionId}`);
|
||||||
throw error;
|
throw error;
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import { isFabricNative } from "Platform/Fabric/FabricUtil";
|
||||||
import { AuthType } from "../../AuthType";
|
import { AuthType } from "../../AuthType";
|
||||||
import { configContext } from "../../ConfigContext";
|
import { configContext } from "../../ConfigContext";
|
||||||
import { userContext } from "../../UserContext";
|
import { userContext } from "../../UserContext";
|
||||||
@@ -41,7 +42,7 @@ interface MetricsResponse {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const getCollectionUsageSizeInKB = async (databaseName: string, containerName: string): Promise<number> => {
|
export const getCollectionUsageSizeInKB = async (databaseName: string, containerName: string): Promise<number> => {
|
||||||
if (userContext.authType !== AuthType.AAD) {
|
if (userContext.authType !== AuthType.AAD || isFabricNative()) {
|
||||||
return undefined;
|
return undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
import { FeedOptions, ItemDefinition, QueryIterator, Resource } from "@azure/cosmos";
|
import { FeedOptions, ItemDefinition, QueryIterator, Resource } from "@azure/cosmos";
|
||||||
import { isVectorSearchEnabled } from "Utils/CapabilityUtils";
|
|
||||||
import { LocalStorageUtility, StorageKey } from "../../Shared/StorageUtility";
|
import { LocalStorageUtility, StorageKey } from "../../Shared/StorageUtility";
|
||||||
import { Queries } from "../Constants";
|
import { Queries } from "../Constants";
|
||||||
import { client } from "../CosmosClient";
|
import { client } from "../CosmosClient";
|
||||||
@@ -28,6 +27,5 @@ export const getCommonQueryOptions = (options: FeedOptions): FeedOptions => {
|
|||||||
Queries.itemsPerPage;
|
Queries.itemsPerPage;
|
||||||
options.enableQueryControl = LocalStorageUtility.getEntryBoolean(StorageKey.QueryControlEnabled);
|
options.enableQueryControl = LocalStorageUtility.getEntryBoolean(StorageKey.QueryControlEnabled);
|
||||||
options.maxDegreeOfParallelism = LocalStorageUtility.getEntryNumber(StorageKey.MaxDegreeOfParellism);
|
options.maxDegreeOfParallelism = LocalStorageUtility.getEntryNumber(StorageKey.MaxDegreeOfParellism);
|
||||||
options.disableNonStreamingOrderByQuery = !isVectorSearchEnabled();
|
|
||||||
return options;
|
return options;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -126,12 +126,5 @@ async function readCollectionsWithARM(databaseId: string): Promise<DataModels.Co
|
|||||||
throw new Error(`Unsupported default experience type: ${apiType}`);
|
throw new Error(`Unsupported default experience type: ${apiType}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
// TO DO: Remove when we get RP API Spec with materializedViews
|
return rpResponse?.value?.map((collection) => collection.properties?.resource as DataModels.Collection);
|
||||||
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
||||||
return rpResponse?.value?.map((collection: any) => {
|
|
||||||
const collectionDataModel: DataModels.Collection = collection.properties?.resource as DataModels.Collection;
|
|
||||||
collectionDataModel.materializedViews = collection.properties?.resource?.materializedViews;
|
|
||||||
collectionDataModel.materializedViewDefinition = collection.properties?.resource?.materializedViewDefinition;
|
|
||||||
return collectionDataModel;
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,21 +1,15 @@
|
|||||||
|
import { CassandraProxyEndpoints, JunoEndpoints, MongoProxyEndpoints, PortalBackendEndpoints } from "Common/Constants";
|
||||||
import {
|
import {
|
||||||
BackendApi,
|
|
||||||
CassandraProxyEndpoints,
|
|
||||||
JunoEndpoints,
|
|
||||||
MongoProxyEndpoints,
|
|
||||||
PortalBackendEndpoints,
|
|
||||||
} from "Common/Constants";
|
|
||||||
import {
|
|
||||||
allowedAadEndpoints,
|
|
||||||
allowedArcadiaEndpoints,
|
allowedArcadiaEndpoints,
|
||||||
allowedEmulatorEndpoints,
|
allowedEmulatorEndpoints,
|
||||||
allowedGraphEndpoints,
|
|
||||||
allowedHostedExplorerEndpoints,
|
allowedHostedExplorerEndpoints,
|
||||||
allowedJunoOrigins,
|
allowedJunoOrigins,
|
||||||
allowedMsalRedirectEndpoints,
|
allowedMsalRedirectEndpoints,
|
||||||
|
defaultAllowedAadEndpoints,
|
||||||
defaultAllowedArmEndpoints,
|
defaultAllowedArmEndpoints,
|
||||||
defaultAllowedBackendEndpoints,
|
defaultAllowedBackendEndpoints,
|
||||||
defaultAllowedCassandraProxyEndpoints,
|
defaultAllowedCassandraProxyEndpoints,
|
||||||
|
defaultAllowedGraphEndpoints,
|
||||||
defaultAllowedMongoProxyEndpoints,
|
defaultAllowedMongoProxyEndpoints,
|
||||||
validateEndpoint,
|
validateEndpoint,
|
||||||
} from "Utils/EndpointUtils";
|
} from "Utils/EndpointUtils";
|
||||||
@@ -29,6 +23,8 @@ export enum Platform {
|
|||||||
|
|
||||||
export interface ConfigContext {
|
export interface ConfigContext {
|
||||||
platform: Platform;
|
platform: Platform;
|
||||||
|
allowedAadEndpoints: ReadonlyArray<string>;
|
||||||
|
allowedGraphEndpoints: ReadonlyArray<string>;
|
||||||
allowedArmEndpoints: ReadonlyArray<string>;
|
allowedArmEndpoints: ReadonlyArray<string>;
|
||||||
allowedBackendEndpoints: ReadonlyArray<string>;
|
allowedBackendEndpoints: ReadonlyArray<string>;
|
||||||
allowedCassandraProxyEndpoints: ReadonlyArray<string>;
|
allowedCassandraProxyEndpoints: ReadonlyArray<string>;
|
||||||
@@ -37,10 +33,8 @@ export interface ConfigContext {
|
|||||||
gitSha?: string;
|
gitSha?: string;
|
||||||
proxyPath?: string;
|
proxyPath?: string;
|
||||||
AAD_ENDPOINT: string;
|
AAD_ENDPOINT: string;
|
||||||
ARM_AUTH_AREA: string;
|
|
||||||
ARM_ENDPOINT: string;
|
ARM_ENDPOINT: string;
|
||||||
EMULATOR_ENDPOINT?: string;
|
EMULATOR_ENDPOINT?: string;
|
||||||
ARM_API_VERSION: string;
|
|
||||||
GRAPH_ENDPOINT: string;
|
GRAPH_ENDPOINT: string;
|
||||||
GRAPH_API_VERSION: string;
|
GRAPH_API_VERSION: string;
|
||||||
// This is the endpoint to get offering Ids to be used to fetch prices. Refer to this doc: https://learn.microsoft.com/en-us/rest/api/marketplacecatalog/dataplane/skus/list?view=rest-marketplacecatalog-dataplane-2023-05-01-preview&tabs=HTTP
|
// This is the endpoint to get offering Ids to be used to fetch prices. Refer to this doc: https://learn.microsoft.com/en-us/rest/api/marketplacecatalog/dataplane/skus/list?view=rest-marketplacecatalog-dataplane-2023-05-01-preview&tabs=HTTP
|
||||||
@@ -50,27 +44,24 @@ export interface ConfigContext {
|
|||||||
ARCADIA_ENDPOINT: string;
|
ARCADIA_ENDPOINT: string;
|
||||||
ARCADIA_LIVY_ENDPOINT_DNS_ZONE: string;
|
ARCADIA_LIVY_ENDPOINT_DNS_ZONE: string;
|
||||||
PORTAL_BACKEND_ENDPOINT: string;
|
PORTAL_BACKEND_ENDPOINT: string;
|
||||||
NEW_BACKEND_APIS?: BackendApi[];
|
|
||||||
MONGO_PROXY_ENDPOINT: string;
|
MONGO_PROXY_ENDPOINT: string;
|
||||||
CASSANDRA_PROXY_ENDPOINT: string;
|
CASSANDRA_PROXY_ENDPOINT: string;
|
||||||
NEW_CASSANDRA_APIS?: string[];
|
|
||||||
PROXY_PATH?: string;
|
PROXY_PATH?: string;
|
||||||
JUNO_ENDPOINT: string;
|
JUNO_ENDPOINT: string;
|
||||||
GITHUB_CLIENT_ID: string;
|
GITHUB_CLIENT_ID: string;
|
||||||
GITHUB_TEST_ENV_CLIENT_ID: string;
|
GITHUB_TEST_ENV_CLIENT_ID: string;
|
||||||
GITHUB_CLIENT_SECRET?: string; // No need to inject secret for prod. Juno already knows it.
|
GITHUB_CLIENT_SECRET?: string; // No need to inject secret for prod. Juno already knows it.
|
||||||
isTerminalEnabled: boolean;
|
|
||||||
isPhoenixEnabled: boolean;
|
isPhoenixEnabled: boolean;
|
||||||
hostedExplorerURL: string;
|
hostedExplorerURL: string;
|
||||||
armAPIVersion?: string;
|
armAPIVersion?: string;
|
||||||
msalRedirectURI?: string;
|
msalRedirectURI?: string;
|
||||||
globallyEnabledCassandraAPIs?: string[];
|
|
||||||
globallyEnabledMongoAPIs?: string[];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Default configuration
|
// Default configuration
|
||||||
let configContext: Readonly<ConfigContext> = {
|
let configContext: Readonly<ConfigContext> = {
|
||||||
platform: Platform.Portal,
|
platform: Platform.Portal,
|
||||||
|
allowedAadEndpoints: defaultAllowedAadEndpoints,
|
||||||
|
allowedGraphEndpoints: defaultAllowedGraphEndpoints,
|
||||||
allowedArmEndpoints: defaultAllowedArmEndpoints,
|
allowedArmEndpoints: defaultAllowedArmEndpoints,
|
||||||
allowedBackendEndpoints: defaultAllowedBackendEndpoints,
|
allowedBackendEndpoints: defaultAllowedBackendEndpoints,
|
||||||
allowedCassandraProxyEndpoints: defaultAllowedCassandraProxyEndpoints,
|
allowedCassandraProxyEndpoints: defaultAllowedCassandraProxyEndpoints,
|
||||||
@@ -85,17 +76,12 @@ let configContext: Readonly<ConfigContext> = {
|
|||||||
`^https:\\/\\/cosmos-db-dataexplorer-germanycentral\\.azurewebsites\\.de$`,
|
`^https:\\/\\/cosmos-db-dataexplorer-germanycentral\\.azurewebsites\\.de$`,
|
||||||
`^https:\\/\\/.*\\.fabric\\.microsoft\\.com$`,
|
`^https:\\/\\/.*\\.fabric\\.microsoft\\.com$`,
|
||||||
`^https:\\/\\/.*\\.powerbi\\.com$`,
|
`^https:\\/\\/.*\\.powerbi\\.com$`,
|
||||||
`^https:\\/\\/.*\\.analysis-df\\.net$`,
|
|
||||||
`^https:\\/\\/.*\\.analysis-df\\.windows\\.net$`,
|
|
||||||
`^https:\\/\\/.*\\.azure-test\\.net$`,
|
|
||||||
`^https:\\/\\/dataexplorer-preview\\.azurewebsites\\.net$`,
|
`^https:\\/\\/dataexplorer-preview\\.azurewebsites\\.net$`,
|
||||||
], // Webpack injects this at build time
|
], // Webpack injects this at build time
|
||||||
gitSha: process.env.GIT_SHA,
|
gitSha: process.env.GIT_SHA,
|
||||||
hostedExplorerURL: "https://cosmos.azure.com/",
|
hostedExplorerURL: "https://cosmos.azure.com/",
|
||||||
AAD_ENDPOINT: "https://login.microsoftonline.com/",
|
AAD_ENDPOINT: "https://login.microsoftonline.com/",
|
||||||
ARM_AUTH_AREA: "https://management.azure.com/",
|
|
||||||
ARM_ENDPOINT: "https://management.azure.com/",
|
ARM_ENDPOINT: "https://management.azure.com/",
|
||||||
ARM_API_VERSION: "2016-06-01",
|
|
||||||
GRAPH_ENDPOINT: "https://graph.microsoft.com",
|
GRAPH_ENDPOINT: "https://graph.microsoft.com",
|
||||||
GRAPH_API_VERSION: "1.6",
|
GRAPH_API_VERSION: "1.6",
|
||||||
CATALOG_ENDPOINT: "https://catalogapi.azure.com/",
|
CATALOG_ENDPOINT: "https://catalogapi.azure.com/",
|
||||||
@@ -109,11 +95,7 @@ let configContext: Readonly<ConfigContext> = {
|
|||||||
PORTAL_BACKEND_ENDPOINT: PortalBackendEndpoints.Prod,
|
PORTAL_BACKEND_ENDPOINT: PortalBackendEndpoints.Prod,
|
||||||
MONGO_PROXY_ENDPOINT: MongoProxyEndpoints.Prod,
|
MONGO_PROXY_ENDPOINT: MongoProxyEndpoints.Prod,
|
||||||
CASSANDRA_PROXY_ENDPOINT: CassandraProxyEndpoints.Prod,
|
CASSANDRA_PROXY_ENDPOINT: CassandraProxyEndpoints.Prod,
|
||||||
NEW_CASSANDRA_APIS: ["postQuery", "createOrDelete", "getKeys", "getSchema"],
|
|
||||||
isTerminalEnabled: false,
|
|
||||||
isPhoenixEnabled: false,
|
isPhoenixEnabled: false,
|
||||||
globallyEnabledCassandraAPIs: [],
|
|
||||||
globallyEnabledMongoAPIs: [],
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export function resetConfigContext(): void {
|
export function resetConfigContext(): void {
|
||||||
@@ -128,19 +110,38 @@ export function updateConfigContext(newContext: Partial<ConfigContext>): void {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!validateEndpoint(newContext.ARM_ENDPOINT, configContext.allowedArmEndpoints || defaultAllowedArmEndpoints)) {
|
if (newContext.allowedAadEndpoints) {
|
||||||
delete newContext.ARM_ENDPOINT;
|
Object.assign(configContext, { allowedAadEndpoints: newContext.allowedAadEndpoints });
|
||||||
|
}
|
||||||
|
if (newContext.allowedArmEndpoints) {
|
||||||
|
Object.assign(configContext, { allowedArmEndpoints: newContext.allowedArmEndpoints });
|
||||||
|
}
|
||||||
|
if (newContext.allowedGraphEndpoints) {
|
||||||
|
Object.assign(configContext, { allowedGraphEndpoints: newContext.allowedGraphEndpoints });
|
||||||
|
}
|
||||||
|
if (newContext.allowedBackendEndpoints) {
|
||||||
|
Object.assign(configContext, { allowedBackendEndpoints: newContext.allowedBackendEndpoints });
|
||||||
|
}
|
||||||
|
if (newContext.allowedMongoProxyEndpoints) {
|
||||||
|
Object.assign(configContext, { allowedMongoProxyEndpoints: newContext.allowedMongoProxyEndpoints });
|
||||||
|
}
|
||||||
|
if (newContext.allowedCassandraProxyEndpoints) {
|
||||||
|
Object.assign(configContext, { allowedCassandraProxyEndpoints: newContext.allowedCassandraProxyEndpoints });
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!validateEndpoint(newContext.AAD_ENDPOINT, allowedAadEndpoints)) {
|
if (!validateEndpoint(newContext.AAD_ENDPOINT, configContext.allowedAadEndpoints)) {
|
||||||
delete newContext.AAD_ENDPOINT;
|
delete newContext.AAD_ENDPOINT;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!validateEndpoint(newContext.ARM_ENDPOINT, configContext.allowedArmEndpoints)) {
|
||||||
|
delete newContext.ARM_ENDPOINT;
|
||||||
|
}
|
||||||
|
|
||||||
if (!validateEndpoint(newContext.EMULATOR_ENDPOINT, allowedEmulatorEndpoints)) {
|
if (!validateEndpoint(newContext.EMULATOR_ENDPOINT, allowedEmulatorEndpoints)) {
|
||||||
delete newContext.EMULATOR_ENDPOINT;
|
delete newContext.EMULATOR_ENDPOINT;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!validateEndpoint(newContext.GRAPH_ENDPOINT, allowedGraphEndpoints)) {
|
if (!validateEndpoint(newContext.GRAPH_ENDPOINT, configContext.allowedGraphEndpoints)) {
|
||||||
delete newContext.GRAPH_ENDPOINT;
|
delete newContext.GRAPH_ENDPOINT;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -148,21 +149,15 @@ export function updateConfigContext(newContext: Partial<ConfigContext>): void {
|
|||||||
delete newContext.ARCADIA_ENDPOINT;
|
delete newContext.ARCADIA_ENDPOINT;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (
|
if (!validateEndpoint(newContext.PORTAL_BACKEND_ENDPOINT, configContext.allowedBackendEndpoints)) {
|
||||||
!validateEndpoint(
|
delete newContext.PORTAL_BACKEND_ENDPOINT;
|
||||||
newContext.MONGO_PROXY_ENDPOINT,
|
}
|
||||||
configContext.allowedMongoProxyEndpoints || defaultAllowedMongoProxyEndpoints,
|
|
||||||
)
|
if (!validateEndpoint(newContext.MONGO_PROXY_ENDPOINT, configContext.allowedMongoProxyEndpoints)) {
|
||||||
) {
|
|
||||||
delete newContext.MONGO_PROXY_ENDPOINT;
|
delete newContext.MONGO_PROXY_ENDPOINT;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (
|
if (!validateEndpoint(newContext.CASSANDRA_PROXY_ENDPOINT, configContext.allowedCassandraProxyEndpoints)) {
|
||||||
!validateEndpoint(
|
|
||||||
newContext.CASSANDRA_PROXY_ENDPOINT,
|
|
||||||
configContext.allowedCassandraProxyEndpoints || defaultAllowedCassandraProxyEndpoints,
|
|
||||||
)
|
|
||||||
) {
|
|
||||||
delete newContext.CASSANDRA_PROXY_ENDPOINT;
|
delete newContext.CASSANDRA_PROXY_ENDPOINT;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -23,6 +23,7 @@ export enum PaneKind {
|
|||||||
GlobalSettings,
|
GlobalSettings,
|
||||||
AdHocAccess,
|
AdHocAccess,
|
||||||
SwitchDirectory,
|
SwitchDirectory,
|
||||||
|
QuickStart,
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import { FabricMessageTypes } from "./FabricMessageTypes";
|
import { FabricMessageTypes } from "./FabricMessageTypes";
|
||||||
|
import { MessageTypes } from "./MessageTypes";
|
||||||
|
|
||||||
// This is the current version of these messages
|
// This is the current version of these messages
|
||||||
export const DATA_EXPLORER_RPC_VERSION = "3";
|
export const DATA_EXPLORER_RPC_VERSION = "3";
|
||||||
@@ -19,9 +20,32 @@ export type DataExploreMessageV3 =
|
|||||||
type: FabricMessageTypes.GetAllResourceTokens;
|
type: FabricMessageTypes.GetAllResourceTokens;
|
||||||
id: string;
|
id: string;
|
||||||
}
|
}
|
||||||
|
| {
|
||||||
|
type: FabricMessageTypes.GetAccessToken;
|
||||||
|
id: string;
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: MessageTypes.TelemetryInfo;
|
||||||
|
data: {
|
||||||
|
action: string;
|
||||||
|
actionModifier: string;
|
||||||
|
data: unknown;
|
||||||
|
timestamp: number;
|
||||||
|
};
|
||||||
|
}
|
||||||
| {
|
| {
|
||||||
type: FabricMessageTypes.OpenSettings;
|
type: FabricMessageTypes.OpenSettings;
|
||||||
settingsId: string;
|
params: [{ settingsId?: "About" | "Connection" }];
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: FabricMessageTypes.RestoreContainer;
|
||||||
|
params: [];
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: FabricMessageTypes.ContainerUpdated;
|
||||||
|
params: {
|
||||||
|
updateType: "created" | "deleted" | "settings";
|
||||||
|
};
|
||||||
};
|
};
|
||||||
export interface GetCosmosTokenMessageOptions {
|
export interface GetCosmosTokenMessageOptions {
|
||||||
verb: "connect" | "delete" | "get" | "head" | "options" | "patch" | "post" | "put" | "trace";
|
verb: "connect" | "delete" | "get" | "head" | "options" | "patch" | "post" | "put" | "trace";
|
||||||
|
|||||||
@@ -7,17 +7,45 @@ export interface ArmEntity {
|
|||||||
type: string;
|
type: string;
|
||||||
kind: string;
|
kind: string;
|
||||||
tags?: Tags;
|
tags?: Tags;
|
||||||
|
resourceGroup?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface DatabaseAccountUserAssignedIdentity {
|
||||||
|
[key: string]: {
|
||||||
|
principalId: string;
|
||||||
|
clientId: string;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface DatabaseAccountIdentity {
|
||||||
|
type: string;
|
||||||
|
principalId?: string;
|
||||||
|
tenantId?: string;
|
||||||
|
userAssignedIdentities?: DatabaseAccountUserAssignedIdentity;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DatabaseAccount extends ArmEntity {
|
export interface DatabaseAccount extends ArmEntity {
|
||||||
properties: DatabaseAccountExtendedProperties;
|
properties: DatabaseAccountExtendedProperties;
|
||||||
systemData?: DatabaseAccountSystemData;
|
systemData?: DatabaseAccountSystemData;
|
||||||
|
identity?: DatabaseAccountIdentity | null;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DatabaseAccountSystemData {
|
export interface DatabaseAccountSystemData {
|
||||||
createdAt: string;
|
createdAt: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface DatabaseAccountBackupPolicy {
|
||||||
|
type: string;
|
||||||
|
/* periodicModeProperties?: {
|
||||||
|
backupIntervalInMinutes: number;
|
||||||
|
backupRetentionIntervalInHours: number;
|
||||||
|
backupStorageRedundancy: string;
|
||||||
|
};
|
||||||
|
continuousModeProperties?: {
|
||||||
|
tier: string;
|
||||||
|
}; */
|
||||||
|
}
|
||||||
|
|
||||||
export interface DatabaseAccountExtendedProperties {
|
export interface DatabaseAccountExtendedProperties {
|
||||||
documentEndpoint?: string;
|
documentEndpoint?: string;
|
||||||
disableLocalAuth?: boolean;
|
disableLocalAuth?: boolean;
|
||||||
@@ -28,6 +56,8 @@ export interface DatabaseAccountExtendedProperties {
|
|||||||
capabilities?: Capability[];
|
capabilities?: Capability[];
|
||||||
enableMultipleWriteLocations?: boolean;
|
enableMultipleWriteLocations?: boolean;
|
||||||
mongoEndpoint?: string;
|
mongoEndpoint?: string;
|
||||||
|
backupPolicy?: DatabaseAccountBackupPolicy;
|
||||||
|
defaultIdentity?: string;
|
||||||
readLocations?: DatabaseAccountResponseLocation[];
|
readLocations?: DatabaseAccountResponseLocation[];
|
||||||
writeLocations?: DatabaseAccountResponseLocation[];
|
writeLocations?: DatabaseAccountResponseLocation[];
|
||||||
enableFreeTier?: boolean;
|
enableFreeTier?: boolean;
|
||||||
@@ -100,6 +130,24 @@ export interface Subscription {
|
|||||||
authorizationSource?: string;
|
authorizationSource?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface DatabaseModel extends ArmEntity {
|
||||||
|
properties: DatabaseGetProperties;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface DatabaseGetProperties {
|
||||||
|
resource: DatabaseResource & ExtendedResourceProperties;
|
||||||
|
}
|
||||||
|
export interface DatabaseResource {
|
||||||
|
id: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ExtendedResourceProperties {
|
||||||
|
readonly _rid?: string;
|
||||||
|
readonly _self?: string;
|
||||||
|
readonly _ts?: number;
|
||||||
|
readonly _etag?: string;
|
||||||
|
}
|
||||||
|
|
||||||
export interface SubscriptionPolicies {
|
export interface SubscriptionPolicies {
|
||||||
locationPlacementId: string;
|
locationPlacementId: string;
|
||||||
quotaId: string;
|
quotaId: string;
|
||||||
@@ -388,7 +436,7 @@ export interface VectorEmbeddingPolicy {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface VectorEmbedding {
|
export interface VectorEmbedding {
|
||||||
dataType: "float16" | "float32" | "uint8" | "int8";
|
dataType: "float32" | "uint8" | "int8";
|
||||||
dimensions: number;
|
dimensions: number;
|
||||||
distanceFunction: "euclidean" | "cosine" | "dotproduct";
|
distanceFunction: "euclidean" | "cosine" | "dotproduct";
|
||||||
path: string;
|
path: string;
|
||||||
|
|||||||
@@ -7,6 +7,8 @@ export enum FabricMessageTypes {
|
|||||||
GetAccessToken = "GetAccessToken",
|
GetAccessToken = "GetAccessToken",
|
||||||
Ready = "Ready",
|
Ready = "Ready",
|
||||||
OpenSettings = "OpenSettings",
|
OpenSettings = "OpenSettings",
|
||||||
|
RestoreContainer = "RestoreContainer",
|
||||||
|
ContainerUpdated = "ContainerUpdated",
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface AuthorizationToken {
|
export interface AuthorizationToken {
|
||||||
|
|||||||
@@ -443,6 +443,8 @@ export interface DataExplorerInputsFrame {
|
|||||||
[key: string]: string;
|
[key: string]: string;
|
||||||
};
|
};
|
||||||
feedbackPolicies?: any;
|
feedbackPolicies?: any;
|
||||||
|
aadToken?: string;
|
||||||
|
containerCopyEnabled?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface SelfServeFrameInputs {
|
export interface SelfServeFrameInputs {
|
||||||
|
|||||||
@@ -1,11 +0,0 @@
|
|||||||
<!doctype html>
|
|
||||||
<html class="no-js" lang="en">
|
|
||||||
<head>
|
|
||||||
<meta charset="utf-8" />
|
|
||||||
<link rel="icon" href="data:," />
|
|
||||||
</head>
|
|
||||||
|
|
||||||
<body>
|
|
||||||
<div id="heatmap"></div>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
||||||
@@ -1,55 +0,0 @@
|
|||||||
@import "../../../less/Common/Constants";
|
|
||||||
html {
|
|
||||||
font-family: @DataExplorerFont;
|
|
||||||
padding: 0px;
|
|
||||||
margin: 0px;
|
|
||||||
border: 0px;
|
|
||||||
overflow: hidden;
|
|
||||||
position: fixed;
|
|
||||||
width: 100%;
|
|
||||||
height: 100%;
|
|
||||||
}
|
|
||||||
|
|
||||||
body {
|
|
||||||
font-family: @DataExplorerFont;
|
|
||||||
padding: 0px;
|
|
||||||
margin: 0px;
|
|
||||||
border: 0px;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
#heatmap {
|
|
||||||
.dark-theme {
|
|
||||||
color: @BaseLight;
|
|
||||||
}
|
|
||||||
|
|
||||||
.chartTitle {
|
|
||||||
position: absolute;
|
|
||||||
top: 5px;
|
|
||||||
left: 3px;
|
|
||||||
font-size: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.noDataMessage {
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
align-items: center;
|
|
||||||
position: absolute;
|
|
||||||
z-index: 10000;
|
|
||||||
height: 100%;
|
|
||||||
width: 100%;
|
|
||||||
top: 0;
|
|
||||||
left: 0;
|
|
||||||
opacity: 0.97;
|
|
||||||
div {
|
|
||||||
border-color: rgba(204, 204, 204, 0.8);
|
|
||||||
box-shadow: 0 4px 6px 0 rgba(0, 0, 0, 0.12);
|
|
||||||
padding: 15px 10px;
|
|
||||||
width: calc(55% - 40px);
|
|
||||||
font-size: 13px;
|
|
||||||
text-align: center;
|
|
||||||
border-width: 1px;
|
|
||||||
border-style: solid;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,143 +0,0 @@
|
|||||||
import dayjs from "dayjs";
|
|
||||||
import { handleMessage, Heatmap, isDarkTheme } from "./Heatmap";
|
|
||||||
import { PortalTheme } from "./HeatmapDatatypes";
|
|
||||||
|
|
||||||
describe("The Heatmap Control", () => {
|
|
||||||
const dataPoints = {
|
|
||||||
"1": {
|
|
||||||
"2019-06-19T00:59:10Z": {
|
|
||||||
"Normalized Throughput": 0.35,
|
|
||||||
},
|
|
||||||
"2019-06-19T00:48:10Z": {
|
|
||||||
"Normalized Throughput": 0.25,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
const chartCaptions = {
|
|
||||||
chartTitle: "chart title",
|
|
||||||
yAxisTitle: "YAxisTitle",
|
|
||||||
tooltipText: "Tooltip text",
|
|
||||||
timeWindow: 123456789,
|
|
||||||
};
|
|
||||||
|
|
||||||
let heatmap: Heatmap;
|
|
||||||
const theme: PortalTheme = 1;
|
|
||||||
const divElement = `<div id="${Heatmap.elementId}"></div>`;
|
|
||||||
|
|
||||||
describe("drawHeatmap rendering", () => {
|
|
||||||
beforeEach(() => {
|
|
||||||
heatmap = new Heatmap(dataPoints, chartCaptions, theme);
|
|
||||||
document.body.innerHTML = divElement;
|
|
||||||
});
|
|
||||||
|
|
||||||
afterEach(() => {
|
|
||||||
document.body.innerHTML = ``;
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should call _getChartSettings when drawHeatmap is invoked", () => {
|
|
||||||
const _getChartSettings = jest.spyOn(heatmap, "_getChartSettings");
|
|
||||||
heatmap.drawHeatmap();
|
|
||||||
expect(_getChartSettings).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should call _getLayoutSettings when drawHeatmap is invoked", () => {
|
|
||||||
const _getLayoutSettings = jest.spyOn(heatmap, "_getLayoutSettings");
|
|
||||||
heatmap.drawHeatmap();
|
|
||||||
expect(_getLayoutSettings).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should call _getChartDisplaySettings when drawHeatmap is invoked", () => {
|
|
||||||
const _getChartDisplaySettings = jest.spyOn(heatmap, "_getChartDisplaySettings");
|
|
||||||
heatmap.drawHeatmap();
|
|
||||||
expect(_getChartDisplaySettings).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
|
|
||||||
it("drawHeatmap should render a Heatmap inside the div element", () => {
|
|
||||||
heatmap.drawHeatmap();
|
|
||||||
expect(document.body.innerHTML).not.toEqual(divElement);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe("generateMatrixFromMap", () => {
|
|
||||||
it("should massage input data to match output expected", () => {
|
|
||||||
expect(heatmap.generateMatrixFromMap(dataPoints).yAxisPoints).toEqual(["1"]);
|
|
||||||
expect(heatmap.generateMatrixFromMap(dataPoints).dataPoints).toEqual([[0.25, 0.35]]);
|
|
||||||
expect(heatmap.generateMatrixFromMap(dataPoints).xAxisPoints.length).toEqual(2);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should output the date format to ISO8601 string format", () => {
|
|
||||||
expect(heatmap.generateMatrixFromMap(dataPoints).xAxisPoints[0].slice(10, 11)).toEqual("T");
|
|
||||||
expect(heatmap.generateMatrixFromMap(dataPoints).xAxisPoints[0].slice(-1)).toEqual("Z");
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should convert the time to the user's local time", () => {
|
|
||||||
if (dayjs().utcOffset()) {
|
|
||||||
expect(heatmap.generateMatrixFromMap(dataPoints).xAxisPoints).not.toEqual([
|
|
||||||
"2019-06-19T00:48:10Z",
|
|
||||||
"2019-06-19T00:59:10Z",
|
|
||||||
]);
|
|
||||||
} else {
|
|
||||||
expect(heatmap.generateMatrixFromMap(dataPoints).xAxisPoints).toEqual([
|
|
||||||
"2019-06-19T00:48:10Z",
|
|
||||||
"2019-06-19T00:59:10Z",
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe("isDarkTheme", () => {
|
|
||||||
it("isDarkTheme should return the correct result", () => {
|
|
||||||
expect(isDarkTheme(PortalTheme.dark)).toEqual(true);
|
|
||||||
expect(isDarkTheme(PortalTheme.azure)).not.toEqual(true);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe("iframe rendering when there is no data", () => {
|
|
||||||
afterEach(() => {
|
|
||||||
document.body.innerHTML = ``;
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should show a no data message with a dark theme", () => {
|
|
||||||
const data = {
|
|
||||||
data: {
|
|
||||||
signature: "pcIframe",
|
|
||||||
data: {
|
|
||||||
chartData: {},
|
|
||||||
chartSettings: {},
|
|
||||||
theme: 4,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
origin: "http://localhost",
|
|
||||||
};
|
|
||||||
|
|
||||||
const divElement = `<div id="${Heatmap.elementId}"></div>`;
|
|
||||||
document.body.innerHTML = divElement;
|
|
||||||
|
|
||||||
handleMessage(data as MessageEvent);
|
|
||||||
expect(document.body.innerHTML).toContain("dark-theme");
|
|
||||||
expect(document.body.innerHTML).toContain("noDataMessage");
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should show a no data message with a white theme", () => {
|
|
||||||
const data = {
|
|
||||||
data: {
|
|
||||||
signature: "pcIframe",
|
|
||||||
data: {
|
|
||||||
chartData: {},
|
|
||||||
chartSettings: {},
|
|
||||||
theme: 2,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
origin: "http://localhost",
|
|
||||||
};
|
|
||||||
|
|
||||||
const divElement = `<div id="${Heatmap.elementId}"></div>`;
|
|
||||||
document.body.innerHTML = divElement;
|
|
||||||
|
|
||||||
handleMessage(data as MessageEvent);
|
|
||||||
expect(document.body.innerHTML).not.toContain("dark-theme");
|
|
||||||
expect(document.body.innerHTML).toContain("noDataMessage");
|
|
||||||
});
|
|
||||||
});
|
|
||||||
@@ -1,272 +0,0 @@
|
|||||||
import dayjs from "dayjs";
|
|
||||||
import * as Plotly from "plotly.js-cartesian-dist-min";
|
|
||||||
import { sendCachedDataMessage, sendReadyMessage } from "../../Common/MessageHandler";
|
|
||||||
import { StyleConstants } from "../../Common/StyleConstants";
|
|
||||||
import { MessageTypes } from "../../Contracts/ExplorerContracts";
|
|
||||||
import { isInvalidParentFrameOrigin } from "../../Utils/MessageValidation";
|
|
||||||
import "./Heatmap.less";
|
|
||||||
import {
|
|
||||||
ChartSettings,
|
|
||||||
DataPayload,
|
|
||||||
DisplaySettings,
|
|
||||||
FontSettings,
|
|
||||||
HeatmapCaptions,
|
|
||||||
HeatmapData,
|
|
||||||
LayoutSettings,
|
|
||||||
PartitionTimeStampToData,
|
|
||||||
PortalTheme,
|
|
||||||
} from "./HeatmapDatatypes";
|
|
||||||
|
|
||||||
export class Heatmap {
|
|
||||||
public static readonly elementId: string = "heatmap";
|
|
||||||
|
|
||||||
private _chartData: HeatmapData;
|
|
||||||
private _heatmapCaptions: HeatmapCaptions;
|
|
||||||
private _theme: PortalTheme;
|
|
||||||
private _defaultFontColor: string;
|
|
||||||
|
|
||||||
constructor(data: DataPayload, heatmapCaptions: HeatmapCaptions, theme: PortalTheme) {
|
|
||||||
this._theme = theme;
|
|
||||||
this._defaultFontColor = StyleConstants.BaseDark;
|
|
||||||
this._setThemeColorForChart();
|
|
||||||
this._chartData = this.generateMatrixFromMap(data);
|
|
||||||
this._heatmapCaptions = heatmapCaptions;
|
|
||||||
}
|
|
||||||
|
|
||||||
private _setThemeColorForChart() {
|
|
||||||
if (isDarkTheme(this._theme)) {
|
|
||||||
this._defaultFontColor = StyleConstants.BaseLight;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private _getFontStyles(size: number = StyleConstants.MediumFontSize, color = "#838383"): FontSettings {
|
|
||||||
return {
|
|
||||||
family: StyleConstants.DataExplorerFont,
|
|
||||||
size,
|
|
||||||
color,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
public generateMatrixFromMap(data: DataPayload): HeatmapData {
|
|
||||||
// all keys in data payload, sorted...
|
|
||||||
const rows: string[] = Object.keys(data).sort((a: string, b: string) => {
|
|
||||||
if (parseInt(a) < parseInt(b)) {
|
|
||||||
return -1;
|
|
||||||
} else {
|
|
||||||
if (parseInt(a) > parseInt(b)) {
|
|
||||||
return 1;
|
|
||||||
} else {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
const output: HeatmapData = {
|
|
||||||
yAxisPoints: [],
|
|
||||||
dataPoints: [],
|
|
||||||
xAxisPoints: Object.keys(data[rows[0]]).sort((a: string, b: string) => {
|
|
||||||
if (a < b) {
|
|
||||||
return -1;
|
|
||||||
} else {
|
|
||||||
if (a > b) {
|
|
||||||
return 1;
|
|
||||||
} else {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}),
|
|
||||||
};
|
|
||||||
// go thru all rows and create 2d matrix for heatmap...
|
|
||||||
for (let i = 0; i < rows.length; i++) {
|
|
||||||
output.yAxisPoints.push(rows[i]);
|
|
||||||
const dataPoints: number[] = [];
|
|
||||||
for (let a = 0; a < output.xAxisPoints.length; a++) {
|
|
||||||
const row: PartitionTimeStampToData = data[rows[i]];
|
|
||||||
dataPoints.push(row[output.xAxisPoints[a]]["Normalized Throughput"]);
|
|
||||||
}
|
|
||||||
output.dataPoints.push(dataPoints);
|
|
||||||
}
|
|
||||||
for (let a = 0; a < output.xAxisPoints.length; a++) {
|
|
||||||
const dateTime = output.xAxisPoints[a];
|
|
||||||
// convert to local users timezone...
|
|
||||||
const day = dayjs(new Date(dateTime)).format("YYYY-MM-DD");
|
|
||||||
const hour = dayjs(new Date(dateTime)).format("HH:mm:ss");
|
|
||||||
// coerce to ISOString format since that is what plotly wants...
|
|
||||||
output.xAxisPoints[a] = `${day}T${hour}Z`;
|
|
||||||
}
|
|
||||||
return output;
|
|
||||||
}
|
|
||||||
|
|
||||||
// public for testing purposes
|
|
||||||
public _getChartSettings(): ChartSettings[] {
|
|
||||||
return [
|
|
||||||
{
|
|
||||||
z: this._chartData.dataPoints,
|
|
||||||
type: "heatmap",
|
|
||||||
zmin: 0,
|
|
||||||
zmid: 50,
|
|
||||||
zmax: 100,
|
|
||||||
colorscale: [
|
|
||||||
[0.0, "#1FD338"],
|
|
||||||
[0.1, "#1CAD2F"],
|
|
||||||
[0.2, "#50A527"],
|
|
||||||
[0.3, "#719F21"],
|
|
||||||
[0.4, "#95991B"],
|
|
||||||
[0.5, "#CE8F11"],
|
|
||||||
[0.6, "#E27F0F"],
|
|
||||||
[0.7, "#E46612"],
|
|
||||||
[0.8, "#E64914"],
|
|
||||||
[0.9, "#B80016"],
|
|
||||||
[1.0, "#B80016"],
|
|
||||||
],
|
|
||||||
name: "",
|
|
||||||
hovertemplate: this._heatmapCaptions.tooltipText,
|
|
||||||
colorbar: {
|
|
||||||
thickness: 15,
|
|
||||||
outlinewidth: 0,
|
|
||||||
tickcolor: StyleConstants.BaseDark,
|
|
||||||
tickfont: this._getFontStyles(10, this._defaultFontColor),
|
|
||||||
},
|
|
||||||
y: this._chartData.yAxisPoints,
|
|
||||||
x: this._chartData.xAxisPoints,
|
|
||||||
},
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
// public for testing purposes
|
|
||||||
public _getLayoutSettings(): LayoutSettings {
|
|
||||||
return {
|
|
||||||
margin: {
|
|
||||||
l: 40,
|
|
||||||
r: 10,
|
|
||||||
b: 35,
|
|
||||||
t: 30,
|
|
||||||
pad: 0,
|
|
||||||
},
|
|
||||||
paper_bgcolor: "transparent",
|
|
||||||
plot_bgcolor: "transparent",
|
|
||||||
width: 462,
|
|
||||||
height: 240,
|
|
||||||
yaxis: {
|
|
||||||
title: this._heatmapCaptions.yAxisTitle,
|
|
||||||
titlefont: this._getFontStyles(11),
|
|
||||||
autorange: true,
|
|
||||||
showgrid: false,
|
|
||||||
zeroline: false,
|
|
||||||
showline: false,
|
|
||||||
autotick: true,
|
|
||||||
fixedrange: true,
|
|
||||||
ticks: "",
|
|
||||||
showticklabels: false,
|
|
||||||
},
|
|
||||||
xaxis: {
|
|
||||||
fixedrange: true,
|
|
||||||
title: "*White area in heatmap indicates there is no available data",
|
|
||||||
titlefont: this._getFontStyles(11),
|
|
||||||
autorange: true,
|
|
||||||
showgrid: false,
|
|
||||||
zeroline: false,
|
|
||||||
showline: false,
|
|
||||||
autotick: true,
|
|
||||||
tickformat: this._heatmapCaptions.timeWindow > 7 ? "%I:%M %p" : "%b %e",
|
|
||||||
showticklabels: true,
|
|
||||||
tickfont: this._getFontStyles(10),
|
|
||||||
},
|
|
||||||
title: {
|
|
||||||
text: this._heatmapCaptions.chartTitle,
|
|
||||||
x: 0.01,
|
|
||||||
font: this._getFontStyles(13, this._defaultFontColor),
|
|
||||||
},
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
// public for testing purposes
|
|
||||||
public _getChartDisplaySettings(): DisplaySettings {
|
|
||||||
return {
|
|
||||||
/* heatmap can be fully responsive however the min-height needed in that case is greater than the iframe portal height, hence explicit width + height have been set in _getLayoutSettings
|
|
||||||
responsive: true,*/
|
|
||||||
displayModeBar: false,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
public drawHeatmap(): void {
|
|
||||||
// todo - create random elementId generator so multiple heatmaps can be created - ticket # 431469
|
|
||||||
Plotly.plot(
|
|
||||||
Heatmap.elementId,
|
|
||||||
this._getChartSettings(),
|
|
||||||
this._getLayoutSettings(),
|
|
||||||
this._getChartDisplaySettings(),
|
|
||||||
);
|
|
||||||
const plotDiv: any = document.getElementById(Heatmap.elementId);
|
|
||||||
plotDiv.on("plotly_click", (data: any) => {
|
|
||||||
let timeSelected: string = data.points[0].x;
|
|
||||||
timeSelected = timeSelected.replace(" ", "T");
|
|
||||||
timeSelected = `${timeSelected}Z`;
|
|
||||||
let xAxisIndex = 0;
|
|
||||||
for (let i = 0; i < this._chartData.xAxisPoints.length; i++) {
|
|
||||||
if (this._chartData.xAxisPoints[i] === timeSelected) {
|
|
||||||
xAxisIndex = i;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
const output = [];
|
|
||||||
for (let i = 0; i < this._chartData.dataPoints.length; i++) {
|
|
||||||
output.push(this._chartData.dataPoints[i][xAxisIndex]);
|
|
||||||
}
|
|
||||||
sendCachedDataMessage(MessageTypes.LogInfo, output);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export function isDarkTheme(theme: PortalTheme) {
|
|
||||||
return theme === PortalTheme.dark;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function handleMessage(event: MessageEvent) {
|
|
||||||
if (isInvalidParentFrameOrigin(event)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (typeof event.data !== "object" || event.data["signature"] !== "pcIframe") {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (
|
|
||||||
typeof event.data.data !== "object" ||
|
|
||||||
!("chartData" in event.data.data) ||
|
|
||||||
!("chartSettings" in event.data.data)
|
|
||||||
) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
Plotly.purge(Heatmap.elementId);
|
|
||||||
|
|
||||||
document.getElementById(Heatmap.elementId)!.innerHTML = "";
|
|
||||||
const data = event.data.data;
|
|
||||||
const chartData: DataPayload = data.chartData;
|
|
||||||
const chartSettings: HeatmapCaptions = data.chartSettings;
|
|
||||||
const chartTheme: PortalTheme = data.theme;
|
|
||||||
if (Object.keys(chartData).length) {
|
|
||||||
new Heatmap(chartData, chartSettings, chartTheme).drawHeatmap();
|
|
||||||
} else {
|
|
||||||
const chartTitleElement = document.createElement("div");
|
|
||||||
chartTitleElement.innerHTML = data.chartSettings.chartTitle;
|
|
||||||
chartTitleElement.classList.add("chartTitle");
|
|
||||||
|
|
||||||
const noDataMessageElement = document.createElement("div");
|
|
||||||
noDataMessageElement.classList.add("noDataMessage");
|
|
||||||
const noDataMessageContent = document.createElement("div");
|
|
||||||
noDataMessageContent.innerHTML = data.errorMessage;
|
|
||||||
|
|
||||||
noDataMessageElement.appendChild(noDataMessageContent);
|
|
||||||
|
|
||||||
if (isDarkTheme(chartTheme)) {
|
|
||||||
chartTitleElement.classList.add("dark-theme");
|
|
||||||
noDataMessageElement.classList.add("dark-theme");
|
|
||||||
noDataMessageContent.classList.add("dark-theme");
|
|
||||||
}
|
|
||||||
|
|
||||||
document.getElementById(Heatmap.elementId)!.appendChild(chartTitleElement);
|
|
||||||
document.getElementById(Heatmap.elementId)!.appendChild(noDataMessageElement);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
window.addEventListener("message", handleMessage, false);
|
|
||||||
sendReadyMessage();
|
|
||||||
@@ -1,106 +0,0 @@
|
|||||||
type dataPoint = string | number;
|
|
||||||
|
|
||||||
export interface DataPayload {
|
|
||||||
[id: string]: PartitionTimeStampToData;
|
|
||||||
}
|
|
||||||
|
|
||||||
export enum PortalTheme {
|
|
||||||
blue = 1,
|
|
||||||
azure,
|
|
||||||
light,
|
|
||||||
dark,
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface HeatmapData {
|
|
||||||
yAxisPoints: string[];
|
|
||||||
xAxisPoints: string[];
|
|
||||||
dataPoints: dataPoint[][];
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface HeatmapCaptions {
|
|
||||||
chartTitle: string;
|
|
||||||
yAxisTitle: string;
|
|
||||||
tooltipText: string;
|
|
||||||
timeWindow: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface FontSettings {
|
|
||||||
family: string;
|
|
||||||
size: number;
|
|
||||||
color: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface LayoutSettings {
|
|
||||||
paper_bgcolor?: string;
|
|
||||||
plot_bgcolor?: string;
|
|
||||||
margin?: {
|
|
||||||
l: number;
|
|
||||||
r: number;
|
|
||||||
b: number;
|
|
||||||
t: number;
|
|
||||||
pad: number;
|
|
||||||
};
|
|
||||||
width?: number;
|
|
||||||
height?: number;
|
|
||||||
yaxis?: {
|
|
||||||
fixedrange: boolean;
|
|
||||||
title: HeatmapCaptions["yAxisTitle"];
|
|
||||||
titlefont: FontSettings;
|
|
||||||
autorange: boolean;
|
|
||||||
showgrid: boolean;
|
|
||||||
zeroline: boolean;
|
|
||||||
showline: boolean;
|
|
||||||
autotick: boolean;
|
|
||||||
ticks: "";
|
|
||||||
showticklabels: boolean;
|
|
||||||
};
|
|
||||||
xaxis?: {
|
|
||||||
fixedrange: boolean;
|
|
||||||
title: string;
|
|
||||||
titlefont: FontSettings;
|
|
||||||
autorange: boolean;
|
|
||||||
showgrid: boolean;
|
|
||||||
zeroline: boolean;
|
|
||||||
showline: boolean;
|
|
||||||
autotick: boolean;
|
|
||||||
showticklabels: boolean;
|
|
||||||
tickformat: string;
|
|
||||||
tickfont: FontSettings;
|
|
||||||
};
|
|
||||||
title?: {
|
|
||||||
text: HeatmapCaptions["chartTitle"];
|
|
||||||
x: number;
|
|
||||||
font?: FontSettings;
|
|
||||||
};
|
|
||||||
font?: FontSettings;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface ChartSettings {
|
|
||||||
z: HeatmapData["dataPoints"];
|
|
||||||
type: "heatmap";
|
|
||||||
zmin: number;
|
|
||||||
zmid: number;
|
|
||||||
zmax: number;
|
|
||||||
colorscale: [number, string][];
|
|
||||||
name: string;
|
|
||||||
hovertemplate: HeatmapCaptions["tooltipText"];
|
|
||||||
colorbar: {
|
|
||||||
thickness: number;
|
|
||||||
outlinewidth: number;
|
|
||||||
tickcolor: string;
|
|
||||||
tickfont: FontSettings;
|
|
||||||
};
|
|
||||||
y: HeatmapData["yAxisPoints"];
|
|
||||||
x: HeatmapData["xAxisPoints"];
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface DisplaySettings {
|
|
||||||
displayModeBar: boolean;
|
|
||||||
responsive?: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface PartitionTimeStampToData {
|
|
||||||
[timeSeriesDates: string]: {
|
|
||||||
[NormalizedThroughput: string]: number;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
186
src/Explorer/ContainerCopy/Actions/CopyJobActions.tsx
Normal file
186
src/Explorer/ContainerCopy/Actions/CopyJobActions.tsx
Normal file
@@ -0,0 +1,186 @@
|
|||||||
|
import React from "react";
|
||||||
|
import { userContext } from "UserContext";
|
||||||
|
import { useSidePanel } from "../../../hooks/useSidePanel";
|
||||||
|
import {
|
||||||
|
cancel,
|
||||||
|
complete,
|
||||||
|
create,
|
||||||
|
listByDatabaseAccount,
|
||||||
|
pause,
|
||||||
|
resume,
|
||||||
|
} from "../../../Utils/arm/generatedClients/dataTransferService/dataTransferJobs";
|
||||||
|
import {
|
||||||
|
CreateJobRequest,
|
||||||
|
DataTransferJobGetResults,
|
||||||
|
} from "../../../Utils/arm/generatedClients/dataTransferService/types";
|
||||||
|
import ContainerCopyMessages from "../ContainerCopyMessages";
|
||||||
|
import {
|
||||||
|
convertTime,
|
||||||
|
convertToCamelCase,
|
||||||
|
COSMOS_SQL_COMPONENT,
|
||||||
|
extractErrorMessage,
|
||||||
|
formatUTCDateTime,
|
||||||
|
getAccountDetailsFromResourceId,
|
||||||
|
} from "../CopyJobUtils";
|
||||||
|
import CreateCopyJobScreensProvider from "../CreateCopyJob/Screens/CreateCopyJobScreensProvider";
|
||||||
|
import { CopyJobActions, CopyJobStatusType } from "../Enums";
|
||||||
|
import { MonitorCopyJobsRefState } from "../MonitorCopyJobs/MonitorCopyJobRefState";
|
||||||
|
import { CopyJobContextState, CopyJobError, CopyJobErrorType, CopyJobType } from "../Types";
|
||||||
|
|
||||||
|
export const openCreateCopyJobPanel = () => {
|
||||||
|
const sidePanelState = useSidePanel.getState();
|
||||||
|
sidePanelState.setPanelHasConsole(false);
|
||||||
|
sidePanelState.openSidePanel(
|
||||||
|
ContainerCopyMessages.createCopyJobPanelTitle,
|
||||||
|
<CreateCopyJobScreensProvider />,
|
||||||
|
"650px",
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
let copyJobsAbortController: AbortController | null = null;
|
||||||
|
|
||||||
|
export const getCopyJobs = async (): Promise<CopyJobType[]> => {
|
||||||
|
// Abort previous request if still in-flight
|
||||||
|
if (copyJobsAbortController) {
|
||||||
|
copyJobsAbortController.abort();
|
||||||
|
}
|
||||||
|
copyJobsAbortController = new AbortController();
|
||||||
|
try {
|
||||||
|
const { subscriptionId, resourceGroup, accountName } = getAccountDetailsFromResourceId(
|
||||||
|
userContext.databaseAccount?.id || "",
|
||||||
|
);
|
||||||
|
const response = await listByDatabaseAccount(
|
||||||
|
subscriptionId,
|
||||||
|
resourceGroup,
|
||||||
|
accountName,
|
||||||
|
copyJobsAbortController.signal,
|
||||||
|
);
|
||||||
|
|
||||||
|
const jobs = response.value || [];
|
||||||
|
if (!Array.isArray(jobs)) {
|
||||||
|
throw new Error("Invalid migration job status response: Expected an array of jobs.");
|
||||||
|
}
|
||||||
|
copyJobsAbortController = null;
|
||||||
|
|
||||||
|
/* added a lower bound to "0" and upper bound to "100" */
|
||||||
|
const calculateCompletionPercentage = (processed: number, total: number): number => {
|
||||||
|
if (
|
||||||
|
typeof processed !== "number" ||
|
||||||
|
typeof total !== "number" ||
|
||||||
|
!isFinite(processed) ||
|
||||||
|
!isFinite(total) ||
|
||||||
|
total <= 0
|
||||||
|
) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
const percentage = Math.round((processed / total) * 100);
|
||||||
|
return Math.max(0, Math.min(100, percentage));
|
||||||
|
};
|
||||||
|
|
||||||
|
const formattedJobs: CopyJobType[] = jobs
|
||||||
|
.filter(
|
||||||
|
(job: DataTransferJobGetResults) =>
|
||||||
|
job.properties?.source?.component === COSMOS_SQL_COMPONENT &&
|
||||||
|
job.properties?.destination?.component === COSMOS_SQL_COMPONENT,
|
||||||
|
)
|
||||||
|
.sort(
|
||||||
|
(current: DataTransferJobGetResults, next: DataTransferJobGetResults) =>
|
||||||
|
new Date(next.properties.lastUpdatedUtcTime).getTime() -
|
||||||
|
new Date(current.properties.lastUpdatedUtcTime).getTime(),
|
||||||
|
)
|
||||||
|
.map((job: DataTransferJobGetResults, index: number) => {
|
||||||
|
const dateTimeObj = formatUTCDateTime(job.properties.lastUpdatedUtcTime);
|
||||||
|
|
||||||
|
return {
|
||||||
|
ID: (index + 1).toString(),
|
||||||
|
Mode: job.properties.mode,
|
||||||
|
Name: job.properties.jobName,
|
||||||
|
Status: convertToCamelCase(job.properties.status) as CopyJobType["Status"],
|
||||||
|
CompletionPercentage: calculateCompletionPercentage(job.properties.processedCount, job.properties.totalCount),
|
||||||
|
Duration: convertTime(job.properties.duration),
|
||||||
|
LastUpdatedTime: dateTimeObj.formattedDateTime,
|
||||||
|
timestamp: dateTimeObj.timestamp,
|
||||||
|
Error: job.properties.error ? extractErrorMessage(job.properties.error as unknown as CopyJobErrorType) : null,
|
||||||
|
} as CopyJobType;
|
||||||
|
});
|
||||||
|
return formattedJobs;
|
||||||
|
} catch (error) {
|
||||||
|
const errorContent = JSON.stringify(error.content || error);
|
||||||
|
console.error(`Error fetching copy jobs: ${errorContent}`);
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const submitCreateCopyJob = async (state: CopyJobContextState, onSuccess: () => void) => {
|
||||||
|
try {
|
||||||
|
const { source, target, migrationType, jobName } = state;
|
||||||
|
const { subscriptionId, resourceGroup, accountName } = getAccountDetailsFromResourceId(
|
||||||
|
userContext.databaseAccount?.id || "",
|
||||||
|
);
|
||||||
|
const body = {
|
||||||
|
properties: {
|
||||||
|
source: {
|
||||||
|
component: "CosmosDBSql",
|
||||||
|
remoteAccountName: source?.account?.name,
|
||||||
|
databaseName: source?.databaseId,
|
||||||
|
containerName: source?.containerId,
|
||||||
|
},
|
||||||
|
destination: {
|
||||||
|
component: "CosmosDBSql",
|
||||||
|
databaseName: target?.databaseId,
|
||||||
|
containerName: target?.containerId,
|
||||||
|
},
|
||||||
|
mode: migrationType,
|
||||||
|
},
|
||||||
|
} as unknown as CreateJobRequest;
|
||||||
|
|
||||||
|
const response = await create(subscriptionId, resourceGroup, accountName, jobName, body);
|
||||||
|
MonitorCopyJobsRefState.getState().ref?.refreshJobList();
|
||||||
|
onSuccess();
|
||||||
|
return response;
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error submitting create copy job:", error);
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const updateCopyJobStatus = async (job: CopyJobType, action: string): Promise<DataTransferJobGetResults> => {
|
||||||
|
try {
|
||||||
|
let updateFn = null;
|
||||||
|
switch (action.toLowerCase()) {
|
||||||
|
case CopyJobActions.pause:
|
||||||
|
updateFn = pause;
|
||||||
|
break;
|
||||||
|
case CopyJobActions.resume:
|
||||||
|
updateFn = resume;
|
||||||
|
break;
|
||||||
|
case CopyJobActions.cancel:
|
||||||
|
updateFn = cancel;
|
||||||
|
break;
|
||||||
|
case CopyJobActions.complete:
|
||||||
|
updateFn = complete;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw new Error(`Unsupported action: ${action}`);
|
||||||
|
}
|
||||||
|
const { subscriptionId, resourceGroup, accountName } = getAccountDetailsFromResourceId(
|
||||||
|
userContext.databaseAccount?.id || "",
|
||||||
|
);
|
||||||
|
const response = await updateFn?.(subscriptionId, resourceGroup, accountName, job.Name);
|
||||||
|
|
||||||
|
return response;
|
||||||
|
} catch (error) {
|
||||||
|
const errorMessage = JSON.stringify((error as CopyJobError).message || error.content || error);
|
||||||
|
|
||||||
|
const statusList = [CopyJobStatusType.Running, CopyJobStatusType.InProgress, CopyJobStatusType.Partitioning];
|
||||||
|
const pattern = new RegExp(`'(${statusList.join("|")})'`, "g");
|
||||||
|
const normalizedErrorMessage = errorMessage.replace(
|
||||||
|
pattern,
|
||||||
|
`'${ContainerCopyMessages.MonitorJobs.Status.InProgress}'`,
|
||||||
|
);
|
||||||
|
|
||||||
|
console.error(`Error updating copy job status: ${normalizedErrorMessage}`);
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
};
|
||||||
31
src/Explorer/ContainerCopy/CommandBar/CopyJobCommandBar.tsx
Normal file
31
src/Explorer/ContainerCopy/CommandBar/CopyJobCommandBar.tsx
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
import { CommandBar as FluentCommandBar, ICommandBarItemProps } from "@fluentui/react";
|
||||||
|
import React from "react";
|
||||||
|
import { StyleConstants } from "../../../Common/StyleConstants";
|
||||||
|
import { CommandButtonComponentProps } from "../../Controls/CommandButton/CommandButtonComponent";
|
||||||
|
import * as CommandBarUtil from "../../Menus/CommandBar/CommandBarUtil";
|
||||||
|
import { ContainerCopyProps } from "../Types";
|
||||||
|
import { getCommandBarButtons } from "./Utils";
|
||||||
|
|
||||||
|
const backgroundColor = StyleConstants.BaseLight;
|
||||||
|
const rootStyle = {
|
||||||
|
root: {
|
||||||
|
backgroundColor: backgroundColor,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
const CopyJobCommandBar: React.FC<ContainerCopyProps> = ({ container }) => {
|
||||||
|
const commandBarItems: CommandButtonComponentProps[] = getCommandBarButtons(container);
|
||||||
|
const controlButtons: ICommandBarItemProps[] = CommandBarUtil.convertButton(commandBarItems, backgroundColor);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="commandBarContainer">
|
||||||
|
<FluentCommandBar
|
||||||
|
ariaLabel="Use left and right arrow keys to navigate between commands"
|
||||||
|
styles={rootStyle}
|
||||||
|
items={controlButtons}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default CopyJobCommandBar;
|
||||||
58
src/Explorer/ContainerCopy/CommandBar/Utils.ts
Normal file
58
src/Explorer/ContainerCopy/CommandBar/Utils.ts
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
import AddIcon from "../../../../images/Add.svg";
|
||||||
|
import FeedbackIcon from "../../../../images/Feedback-Command.svg";
|
||||||
|
import RefreshIcon from "../../../../images/refresh-cosmos.svg";
|
||||||
|
import { configContext, Platform } from "../../../ConfigContext";
|
||||||
|
import { CommandButtonComponentProps } from "../../Controls/CommandButton/CommandButtonComponent";
|
||||||
|
import Explorer from "../../Explorer";
|
||||||
|
import * as Actions from "../Actions/CopyJobActions";
|
||||||
|
import ContainerCopyMessages from "../ContainerCopyMessages";
|
||||||
|
import { MonitorCopyJobsRefState } from "../MonitorCopyJobs/MonitorCopyJobRefState";
|
||||||
|
import { CopyJobCommandBarBtnType } from "../Types";
|
||||||
|
|
||||||
|
function getCopyJobBtns(): CopyJobCommandBarBtnType[] {
|
||||||
|
const monitorCopyJobsRef = MonitorCopyJobsRefState((state) => state.ref);
|
||||||
|
const buttons: CopyJobCommandBarBtnType[] = [
|
||||||
|
{
|
||||||
|
key: "createCopyJob",
|
||||||
|
iconSrc: AddIcon,
|
||||||
|
label: ContainerCopyMessages.createCopyJobButtonLabel,
|
||||||
|
ariaLabel: ContainerCopyMessages.createCopyJobButtonAriaLabel,
|
||||||
|
onClick: Actions.openCreateCopyJobPanel,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "refresh",
|
||||||
|
iconSrc: RefreshIcon,
|
||||||
|
label: ContainerCopyMessages.refreshButtonLabel,
|
||||||
|
ariaLabel: ContainerCopyMessages.refreshButtonAriaLabel,
|
||||||
|
onClick: () => monitorCopyJobsRef?.refreshJobList(),
|
||||||
|
},
|
||||||
|
];
|
||||||
|
if (configContext.platform === Platform.Portal) {
|
||||||
|
buttons.push({
|
||||||
|
key: "feedback",
|
||||||
|
iconSrc: FeedbackIcon,
|
||||||
|
label: ContainerCopyMessages.feedbackButtonLabel,
|
||||||
|
ariaLabel: ContainerCopyMessages.feedbackButtonAriaLabel,
|
||||||
|
onClick: () => {},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return buttons;
|
||||||
|
}
|
||||||
|
|
||||||
|
function btnMapper(config: CopyJobCommandBarBtnType): CommandButtonComponentProps {
|
||||||
|
return {
|
||||||
|
iconSrc: config.iconSrc,
|
||||||
|
iconAlt: config.label,
|
||||||
|
onCommandClick: config.onClick,
|
||||||
|
commandButtonLabel: undefined as string | undefined,
|
||||||
|
ariaLabel: config.ariaLabel,
|
||||||
|
tooltipText: config.label,
|
||||||
|
hasPopup: false,
|
||||||
|
disabled: config.disabled ?? false,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
||||||
|
export function getCommandBarButtons(_container: Explorer): CommandButtonComponentProps[] {
|
||||||
|
return getCopyJobBtns().map(btnMapper);
|
||||||
|
}
|
||||||
132
src/Explorer/ContainerCopy/ContainerCopyMessages.ts
Normal file
132
src/Explorer/ContainerCopy/ContainerCopyMessages.ts
Normal file
@@ -0,0 +1,132 @@
|
|||||||
|
export default {
|
||||||
|
// Copy Job Command Bar
|
||||||
|
feedbackButtonLabel: "Feedback",
|
||||||
|
feedbackButtonAriaLabel: "Provide feedback on copy jobs",
|
||||||
|
refreshButtonLabel: "Refresh",
|
||||||
|
refreshButtonAriaLabel: "Refresh copy jobs",
|
||||||
|
createCopyJobButtonLabel: "Create Copy Job",
|
||||||
|
createCopyJobButtonAriaLabel: "Create a new container copy job",
|
||||||
|
|
||||||
|
// No Copy Jobs Found
|
||||||
|
noCopyJobsTitle: "No copy jobs to show",
|
||||||
|
createCopyJobButtonText: "Create a container copy job",
|
||||||
|
|
||||||
|
// Create Copy Job Panel
|
||||||
|
createCopyJobPanelTitle: "Copy container",
|
||||||
|
|
||||||
|
// Select Account Screen
|
||||||
|
selectAccountDescription: "Please select a source account from which to copy.",
|
||||||
|
subscriptionDropdownLabel: "Subscription",
|
||||||
|
subscriptionDropdownPlaceholder: "Select a subscription",
|
||||||
|
sourceAccountDropdownLabel: "Account",
|
||||||
|
sourceAccountDropdownPlaceholder: "Select an account",
|
||||||
|
migrationTypeCheckboxLabel: "Copy container in offline mode",
|
||||||
|
|
||||||
|
// Select Source and Target Containers Screen
|
||||||
|
selectSourceAndTargetContainersDescription:
|
||||||
|
"Please select a source container and a destination container to copy to.",
|
||||||
|
sourceContainerSubHeading: "Source container",
|
||||||
|
targetContainerSubHeading: "Destination container",
|
||||||
|
databaseDropdownLabel: "Database",
|
||||||
|
databaseDropdownPlaceholder: "Select a database",
|
||||||
|
containerDropdownLabel: "Container",
|
||||||
|
containerDropdownPlaceholder: "Select a container",
|
||||||
|
|
||||||
|
// Preview and Create Screen
|
||||||
|
jobNameLabel: "Job name",
|
||||||
|
sourceSubscriptionLabel: "Source subscription",
|
||||||
|
sourceAccountLabel: "Source account",
|
||||||
|
sourceDatabaseLabel: "Source database",
|
||||||
|
sourceContainerLabel: "Source container",
|
||||||
|
targetDatabaseLabel: "Destination database",
|
||||||
|
targetContainerLabel: "Destination container",
|
||||||
|
|
||||||
|
// Assign Permissions Screen
|
||||||
|
assignPermissions: {
|
||||||
|
description:
|
||||||
|
"Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat.",
|
||||||
|
},
|
||||||
|
toggleBtn: {
|
||||||
|
onText: "On",
|
||||||
|
offText: "Off",
|
||||||
|
},
|
||||||
|
addManagedIdentity: {
|
||||||
|
title: "System assigned managed identity enabled",
|
||||||
|
description:
|
||||||
|
"Enable a system assigned managed identity for the destination account to allow the copy job to access it.",
|
||||||
|
toggleLabel: "System assigned managed identity",
|
||||||
|
managedIdentityTooltip:
|
||||||
|
"A system assigned managed identity is restricted to one per resource and is tied to the lifecycle of this resource. You can grant permissions to the managed identity by using Azure role-based access control (Azure RBAC). The managed identity is authenticated with Microsoft Entra ID, so you don't have to store any credentials in code.",
|
||||||
|
userAssignedIdentityTooltip: "You can select an existing user assigned identity or create a new one.",
|
||||||
|
userAssignedIdentityLabel: "You may also select a user assigned managed identity.",
|
||||||
|
createUserAssignedIdentityLink: "Create User Assigned Managed Identity",
|
||||||
|
enablementTitle: "Enable system assigned managed identity",
|
||||||
|
enablementDescription: (identityName: string) =>
|
||||||
|
identityName
|
||||||
|
? `'${identityName}' will be registered with Microsoft Entra ID. Once it is registered, '${identityName}' can be granted permissions to access resources protected by Microsoft Entra ID. Do you want to enable the system assigned managed identity for '${identityName}'?`
|
||||||
|
: "",
|
||||||
|
},
|
||||||
|
defaultManagedIdentity: {
|
||||||
|
title: "System assigned managed identity enabled as default",
|
||||||
|
description:
|
||||||
|
"Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.",
|
||||||
|
tooltip:
|
||||||
|
"A system assigned managed identity is restricted to one per resource and is tied to the lifecycle of this resource. You can grant permissions to the managed identity by using Azure role-based access control (Azure RBAC). The managed identity is authenticated with Microsoft Entra ID, so you don't have to store any credentials in code.",
|
||||||
|
popoverTitle: "System assigned managed identity set as default",
|
||||||
|
popoverDescription:
|
||||||
|
"Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco.",
|
||||||
|
},
|
||||||
|
readPermissionAssigned: {
|
||||||
|
title: "Read permission assigned to default identity",
|
||||||
|
description:
|
||||||
|
"Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.",
|
||||||
|
tooltip:
|
||||||
|
"A system assigned managed identity is restricted to one per resource and is tied to the lifecycle of this resource. You can grant permissions to the managed identity by using Azure role-based access control (Azure RBAC). The managed identity is authenticated with Microsoft Entra ID, so you don't have to store any credentials in code.",
|
||||||
|
popoverTitle: "Read permission assigned to default identity",
|
||||||
|
popoverDescription:
|
||||||
|
"Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco.",
|
||||||
|
},
|
||||||
|
pointInTimeRestore: {
|
||||||
|
title: "Point In Time Restore enabled",
|
||||||
|
description:
|
||||||
|
"Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.",
|
||||||
|
buttonText: "Enable Point In Time Restore",
|
||||||
|
},
|
||||||
|
onlineCopyEnabled: {
|
||||||
|
title: "Online copy enabled",
|
||||||
|
description:
|
||||||
|
"Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.",
|
||||||
|
buttonText: "Enable Online Copy",
|
||||||
|
},
|
||||||
|
MonitorJobs: {
|
||||||
|
Columns: {
|
||||||
|
lastUpdatedTime: "Date & time",
|
||||||
|
name: "Job name",
|
||||||
|
status: "Status",
|
||||||
|
completionPercentage: "Completion %",
|
||||||
|
duration: "Duration",
|
||||||
|
error: "Error message",
|
||||||
|
mode: "Mode",
|
||||||
|
actions: "Actions",
|
||||||
|
},
|
||||||
|
Actions: {
|
||||||
|
pause: "Pause",
|
||||||
|
resume: "Resume",
|
||||||
|
cancel: "Cancel",
|
||||||
|
complete: "Complete",
|
||||||
|
viewDetails: "View Details",
|
||||||
|
},
|
||||||
|
Status: {
|
||||||
|
Pending: "Pending",
|
||||||
|
InProgress: "In Progress",
|
||||||
|
Running: "In Progress",
|
||||||
|
Partitioning: "In Progress",
|
||||||
|
Paused: "Paused",
|
||||||
|
Completed: "Completed",
|
||||||
|
Failed: "Failed",
|
||||||
|
Faulted: "Failed",
|
||||||
|
Skipped: "Cancelled",
|
||||||
|
Cancelled: "Cancelled",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
54
src/Explorer/ContainerCopy/Context/CopyJobContext.tsx
Normal file
54
src/Explorer/ContainerCopy/Context/CopyJobContext.tsx
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
import React from "react";
|
||||||
|
import { userContext } from "UserContext";
|
||||||
|
import { CopyJobMigrationType } from "../Enums";
|
||||||
|
import { CopyJobContextProviderType, CopyJobContextState, CopyJobFlowType } from "../Types";
|
||||||
|
|
||||||
|
export const CopyJobContext = React.createContext<CopyJobContextProviderType>(null);
|
||||||
|
export const useCopyJobContext = (): CopyJobContextProviderType => {
|
||||||
|
const context = React.useContext(CopyJobContext);
|
||||||
|
if (!context) {
|
||||||
|
throw new Error("useCopyJobContext must be used within a CopyJobContextProvider");
|
||||||
|
}
|
||||||
|
return context;
|
||||||
|
};
|
||||||
|
|
||||||
|
interface CopyJobContextProviderProps {
|
||||||
|
children: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
const getInitialCopyJobState = (): CopyJobContextState => {
|
||||||
|
return {
|
||||||
|
jobName: "",
|
||||||
|
migrationType: CopyJobMigrationType.Offline,
|
||||||
|
source: {
|
||||||
|
subscription: null,
|
||||||
|
account: null,
|
||||||
|
databaseId: "",
|
||||||
|
containerId: "",
|
||||||
|
},
|
||||||
|
target: {
|
||||||
|
subscriptionId: userContext.subscriptionId || "",
|
||||||
|
account: userContext.databaseAccount || null,
|
||||||
|
databaseId: "",
|
||||||
|
containerId: "",
|
||||||
|
},
|
||||||
|
sourceReadAccessFromTarget: false,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const CopyJobContextProvider: React.FC<CopyJobContextProviderProps> = (props) => {
|
||||||
|
const [copyJobState, setCopyJobState] = React.useState<CopyJobContextState>(getInitialCopyJobState());
|
||||||
|
const [flow, setFlow] = React.useState<CopyJobFlowType | null>(null);
|
||||||
|
|
||||||
|
const resetCopyJobState = () => {
|
||||||
|
setCopyJobState(getInitialCopyJobState());
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<CopyJobContext.Provider value={{ copyJobState, setCopyJobState, flow, setFlow, resetCopyJobState }}>
|
||||||
|
{props.children}
|
||||||
|
</CopyJobContext.Provider>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default CopyJobContextProvider;
|
||||||
116
src/Explorer/ContainerCopy/CopyJobUtils.ts
Normal file
116
src/Explorer/ContainerCopy/CopyJobUtils.ts
Normal file
@@ -0,0 +1,116 @@
|
|||||||
|
import { DatabaseAccount } from "Contracts/DataModels";
|
||||||
|
import { CopyJobErrorType } from "./Types";
|
||||||
|
|
||||||
|
const azurePortalMpacEndpoint = "https://ms.portal.azure.com/";
|
||||||
|
|
||||||
|
export const buildResourceLink = (resource: DatabaseAccount): string => {
|
||||||
|
const resourceId = resource.id;
|
||||||
|
let parentOrigin = window.location.ancestorOrigins?.[0] ?? window.location.origin;
|
||||||
|
|
||||||
|
if (/\/\/localhost:/.test(parentOrigin)) {
|
||||||
|
parentOrigin = azurePortalMpacEndpoint;
|
||||||
|
} else if (/\/\/cosmos\.azure/.test(parentOrigin)) {
|
||||||
|
parentOrigin = parentOrigin.replace("cosmos.azure", "portal.azure");
|
||||||
|
}
|
||||||
|
|
||||||
|
parentOrigin = parentOrigin.replace(/\/$/, "");
|
||||||
|
|
||||||
|
return `${parentOrigin}/#resource${resourceId}`;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const COSMOS_SQL_COMPONENT = "CosmosDBSql";
|
||||||
|
|
||||||
|
export const COPY_JOB_API_VERSION = "2025-05-01-preview";
|
||||||
|
|
||||||
|
export function buildDataTransferJobPath({
|
||||||
|
subscriptionId,
|
||||||
|
resourceGroup,
|
||||||
|
accountName,
|
||||||
|
jobName,
|
||||||
|
action,
|
||||||
|
}: {
|
||||||
|
subscriptionId: string;
|
||||||
|
resourceGroup: string;
|
||||||
|
accountName: string;
|
||||||
|
jobName?: string;
|
||||||
|
action?: string;
|
||||||
|
}) {
|
||||||
|
let path = `/subscriptions/${subscriptionId}/resourceGroups/${resourceGroup}/providers/Microsoft.DocumentDB/databaseAccounts/${accountName}/dataTransferJobs`;
|
||||||
|
if (jobName) {
|
||||||
|
path += `/${jobName}`;
|
||||||
|
}
|
||||||
|
if (action) {
|
||||||
|
path += `/${action}`;
|
||||||
|
}
|
||||||
|
return path;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function convertTime(timeStr: string): string | null {
|
||||||
|
const timeParts = timeStr.split(":").map(Number);
|
||||||
|
|
||||||
|
if (timeParts.length !== 3 || timeParts.some(isNaN)) {
|
||||||
|
return null; // Return null for invalid format
|
||||||
|
}
|
||||||
|
const formatPart = (value: number, unit: string) => {
|
||||||
|
if (unit === "seconds") {
|
||||||
|
value = Math.round(value);
|
||||||
|
}
|
||||||
|
return value > 0 ? `${value.toString().padStart(2, "0")} ${unit}` : "";
|
||||||
|
};
|
||||||
|
|
||||||
|
const [hours, minutes, seconds] = timeParts;
|
||||||
|
const formattedTimeParts = [
|
||||||
|
formatPart(hours, "hours"),
|
||||||
|
formatPart(minutes, "minutes"),
|
||||||
|
formatPart(seconds, "seconds"),
|
||||||
|
]
|
||||||
|
.filter(Boolean)
|
||||||
|
.join(", ");
|
||||||
|
|
||||||
|
return formattedTimeParts || "0 seconds"; // Return "0 seconds" if all parts are zero
|
||||||
|
}
|
||||||
|
|
||||||
|
export function formatUTCDateTime(utcStr: string): { formattedDateTime: string; timestamp: number } | null {
|
||||||
|
const date = new Date(utcStr);
|
||||||
|
if (isNaN(date.getTime())) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
formattedDateTime: new Intl.DateTimeFormat("en-US", {
|
||||||
|
dateStyle: "short",
|
||||||
|
timeStyle: "medium",
|
||||||
|
timeZone: "UTC",
|
||||||
|
}).format(date),
|
||||||
|
timestamp: date.getTime(),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function convertToCamelCase(str: string): string {
|
||||||
|
const formattedStr = str
|
||||||
|
.split(/\s+/)
|
||||||
|
.map((word) => word.charAt(0).toUpperCase() + word.slice(1).toLowerCase())
|
||||||
|
.join("");
|
||||||
|
return formattedStr;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function extractErrorMessage(error: CopyJobErrorType): CopyJobErrorType {
|
||||||
|
return {
|
||||||
|
...error,
|
||||||
|
message: error.message.split("\r\n\r\n")[0],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getAccountDetailsFromResourceId(accountId: string | undefined) {
|
||||||
|
if (!accountId) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
const pattern = new RegExp(
|
||||||
|
"/subscriptions/([^/]+)/resourceGroups/([^/]+)/providers/Microsoft\\.DocumentDB/databaseAccounts/([^/]+)",
|
||||||
|
"i",
|
||||||
|
);
|
||||||
|
const matches = accountId.match(pattern);
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
||||||
|
const [_, subscriptionId, resourceGroup, accountName] = matches || [];
|
||||||
|
return { subscriptionId, resourceGroup, accountName };
|
||||||
|
}
|
||||||
@@ -0,0 +1,67 @@
|
|||||||
|
import { Link, Stack, Text, Toggle } from "@fluentui/react";
|
||||||
|
import React, { useMemo } from "react";
|
||||||
|
import { updateSystemIdentity } from "../../../../../Utils/arm/identityUtils";
|
||||||
|
import ContainerCopyMessages from "../../../ContainerCopyMessages";
|
||||||
|
import { useCopyJobContext } from "../../../Context/CopyJobContext";
|
||||||
|
import { buildResourceLink } from "../../../CopyJobUtils";
|
||||||
|
import InfoTooltip from "../Components/InfoTooltip";
|
||||||
|
import PopoverMessage from "../Components/PopoverContainer";
|
||||||
|
import useManagedIdentity from "./hooks/useManagedIdentity";
|
||||||
|
import { PermissionSectionConfig } from "./hooks/usePermissionsSection";
|
||||||
|
import useToggle from "./hooks/useToggle";
|
||||||
|
|
||||||
|
const managedIdentityTooltip = ContainerCopyMessages.addManagedIdentity.managedIdentityTooltip;
|
||||||
|
const userAssignedTooltip = ContainerCopyMessages.addManagedIdentity.userAssignedIdentityTooltip;
|
||||||
|
|
||||||
|
const textStyle = { display: "flex", alignItems: "center" };
|
||||||
|
|
||||||
|
type AddManagedIdentityProps = Partial<PermissionSectionConfig>;
|
||||||
|
|
||||||
|
const AddManagedIdentity: React.FC<AddManagedIdentityProps> = () => {
|
||||||
|
const { copyJobState } = useCopyJobContext();
|
||||||
|
const [systemAssigned, onToggle] = useToggle(false);
|
||||||
|
const { loading, handleAddSystemIdentity } = useManagedIdentity(updateSystemIdentity);
|
||||||
|
|
||||||
|
const manageIdentityLink = useMemo(() => {
|
||||||
|
const { target } = copyJobState;
|
||||||
|
const resourceUri = buildResourceLink(target.account);
|
||||||
|
return target?.account?.id ? `${resourceUri}/ManagedIdentitiesBlade` : "#";
|
||||||
|
}, [copyJobState]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Stack className="addManagedIdentityContainer" tokens={{ childrenGap: 15, padding: "0 0 0 20px" }}>
|
||||||
|
<Toggle
|
||||||
|
label={
|
||||||
|
<Text className="toggle-label" style={textStyle}>
|
||||||
|
{ContainerCopyMessages.addManagedIdentity.toggleLabel}
|
||||||
|
<InfoTooltip content={managedIdentityTooltip} />
|
||||||
|
</Text>
|
||||||
|
}
|
||||||
|
checked={systemAssigned}
|
||||||
|
onText={ContainerCopyMessages.toggleBtn.onText}
|
||||||
|
offText={ContainerCopyMessages.toggleBtn.offText}
|
||||||
|
onChange={onToggle}
|
||||||
|
/>
|
||||||
|
<Text className="user-assigned-label" style={textStyle}>
|
||||||
|
{ContainerCopyMessages.addManagedIdentity.userAssignedIdentityLabel}
|
||||||
|
<InfoTooltip content={userAssignedTooltip} />
|
||||||
|
</Text>
|
||||||
|
<div style={{ marginTop: 8 }}>
|
||||||
|
<Link href={manageIdentityLink} target="_blank" rel="noopener noreferrer">
|
||||||
|
{ContainerCopyMessages.addManagedIdentity.createUserAssignedIdentityLink}
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
<PopoverMessage
|
||||||
|
isLoading={loading}
|
||||||
|
visible={systemAssigned}
|
||||||
|
title={ContainerCopyMessages.addManagedIdentity.enablementTitle}
|
||||||
|
onCancel={() => onToggle(null, false)}
|
||||||
|
onPrimary={handleAddSystemIdentity}
|
||||||
|
>
|
||||||
|
{ContainerCopyMessages.addManagedIdentity.enablementDescription(copyJobState.target?.account?.name)}
|
||||||
|
</PopoverMessage>
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default AddManagedIdentity;
|
||||||
@@ -0,0 +1,80 @@
|
|||||||
|
import { Stack, Toggle } from "@fluentui/react";
|
||||||
|
import React, { useCallback } from "react";
|
||||||
|
import { assignRole } from "../../../../../Utils/arm/RbacUtils";
|
||||||
|
import ContainerCopyMessages from "../../../ContainerCopyMessages";
|
||||||
|
import { useCopyJobContext } from "../../../Context/CopyJobContext";
|
||||||
|
import { getAccountDetailsFromResourceId } from "../../../CopyJobUtils";
|
||||||
|
import InfoTooltip from "../Components/InfoTooltip";
|
||||||
|
import PopoverMessage from "../Components/PopoverContainer";
|
||||||
|
import { PermissionSectionConfig } from "./hooks/usePermissionsSection";
|
||||||
|
import useToggle from "./hooks/useToggle";
|
||||||
|
|
||||||
|
const TooltipContent = ContainerCopyMessages.readPermissionAssigned.tooltip;
|
||||||
|
type AddManagedIdentityProps = Partial<PermissionSectionConfig>;
|
||||||
|
|
||||||
|
const AddReadPermissionToDefaultIdentity: React.FC<AddManagedIdentityProps> = () => {
|
||||||
|
const [loading, setLoading] = React.useState(false);
|
||||||
|
const { copyJobState, setCopyJobState } = useCopyJobContext();
|
||||||
|
const [readPermissionAssigned, onToggle] = useToggle(false);
|
||||||
|
|
||||||
|
const handleAddReadPermission = useCallback(async () => {
|
||||||
|
const { source, target } = copyJobState;
|
||||||
|
const selectedSourceAccount = source?.account;
|
||||||
|
try {
|
||||||
|
const {
|
||||||
|
subscriptionId: sourceSubscriptionId,
|
||||||
|
resourceGroup: sourceResourceGroup,
|
||||||
|
accountName: sourceAccountName,
|
||||||
|
} = getAccountDetailsFromResourceId(selectedSourceAccount?.id);
|
||||||
|
|
||||||
|
setLoading(true);
|
||||||
|
const assignedRole = await assignRole(
|
||||||
|
sourceSubscriptionId,
|
||||||
|
sourceResourceGroup,
|
||||||
|
sourceAccountName,
|
||||||
|
target?.account?.identity?.principalId ?? "",
|
||||||
|
);
|
||||||
|
if (assignedRole) {
|
||||||
|
setCopyJobState((prevState) => ({
|
||||||
|
...prevState,
|
||||||
|
sourceReadAccessFromTarget: true,
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error assigning read permission to default identity:", error);
|
||||||
|
} finally {
|
||||||
|
setLoading(false);
|
||||||
|
}
|
||||||
|
}, [copyJobState, setCopyJobState]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Stack className="defaultManagedIdentityContainer" tokens={{ childrenGap: 15, padding: "0 0 0 20px" }}>
|
||||||
|
<div className="toggle-label">
|
||||||
|
{ContainerCopyMessages.readPermissionAssigned.description}
|
||||||
|
<InfoTooltip content={TooltipContent} />
|
||||||
|
</div>
|
||||||
|
<Toggle
|
||||||
|
checked={readPermissionAssigned}
|
||||||
|
onText={ContainerCopyMessages.toggleBtn.onText}
|
||||||
|
offText={ContainerCopyMessages.toggleBtn.offText}
|
||||||
|
onChange={onToggle}
|
||||||
|
inlineLabel
|
||||||
|
styles={{
|
||||||
|
root: { marginTop: 8, marginBottom: 12 },
|
||||||
|
label: { display: "none" },
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<PopoverMessage
|
||||||
|
isLoading={loading}
|
||||||
|
visible={readPermissionAssigned}
|
||||||
|
title={ContainerCopyMessages.readPermissionAssigned.popoverTitle}
|
||||||
|
onCancel={() => onToggle(null, false)}
|
||||||
|
onPrimary={handleAddReadPermission}
|
||||||
|
>
|
||||||
|
{ContainerCopyMessages.readPermissionAssigned.popoverDescription}
|
||||||
|
</PopoverMessage>
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default AddReadPermissionToDefaultIdentity;
|
||||||
@@ -0,0 +1,48 @@
|
|||||||
|
import { Stack, Toggle } from "@fluentui/react";
|
||||||
|
import React from "react";
|
||||||
|
import { updateDefaultIdentity } from "../../../../../Utils/arm/identityUtils";
|
||||||
|
import ContainerCopyMessages from "../../../ContainerCopyMessages";
|
||||||
|
import InfoTooltip from "../Components/InfoTooltip";
|
||||||
|
import PopoverMessage from "../Components/PopoverContainer";
|
||||||
|
import useManagedIdentity from "./hooks/useManagedIdentity";
|
||||||
|
import { PermissionSectionConfig } from "./hooks/usePermissionsSection";
|
||||||
|
import useToggle from "./hooks/useToggle";
|
||||||
|
|
||||||
|
const managedIdentityTooltip = ContainerCopyMessages.defaultManagedIdentity.tooltip;
|
||||||
|
type AddManagedIdentityProps = Partial<PermissionSectionConfig>;
|
||||||
|
|
||||||
|
const DefaultManagedIdentity: React.FC<AddManagedIdentityProps> = () => {
|
||||||
|
const [defaultSystemAssigned, onToggle] = useToggle(false);
|
||||||
|
const { loading, handleAddSystemIdentity } = useManagedIdentity(updateDefaultIdentity);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Stack className="defaultManagedIdentityContainer" tokens={{ childrenGap: 15, padding: "0 0 0 20px" }}>
|
||||||
|
<div className="toggle-label">
|
||||||
|
{ContainerCopyMessages.defaultManagedIdentity.description}
|
||||||
|
<InfoTooltip content={managedIdentityTooltip} />
|
||||||
|
</div>
|
||||||
|
<Toggle
|
||||||
|
checked={defaultSystemAssigned}
|
||||||
|
onText={ContainerCopyMessages.toggleBtn.onText}
|
||||||
|
offText={ContainerCopyMessages.toggleBtn.offText}
|
||||||
|
onChange={onToggle}
|
||||||
|
inlineLabel
|
||||||
|
styles={{
|
||||||
|
root: { marginTop: 8, marginBottom: 12 },
|
||||||
|
label: { display: "none" },
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<PopoverMessage
|
||||||
|
isLoading={loading}
|
||||||
|
visible={defaultSystemAssigned}
|
||||||
|
title={ContainerCopyMessages.defaultManagedIdentity.popoverTitle}
|
||||||
|
onCancel={() => onToggle(null, false)}
|
||||||
|
onPrimary={handleAddSystemIdentity}
|
||||||
|
>
|
||||||
|
{ContainerCopyMessages.defaultManagedIdentity.popoverDescription}
|
||||||
|
</PopoverMessage>
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default DefaultManagedIdentity;
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
import { PrimaryButton, Stack } from "@fluentui/react";
|
||||||
|
import React from "react";
|
||||||
|
import ContainerCopyMessages from "../../../ContainerCopyMessages";
|
||||||
|
import { useCopyJobContext } from "../../../Context/CopyJobContext";
|
||||||
|
import { buildResourceLink } from "../../../CopyJobUtils";
|
||||||
|
import { PermissionSectionConfig } from "./hooks/usePermissionsSection";
|
||||||
|
import useWindowOpenMonitor from "./hooks/useWindowOpenMonitor";
|
||||||
|
|
||||||
|
type AddManagedIdentityProps = Partial<PermissionSectionConfig>;
|
||||||
|
const OnlineCopyEnabled: React.FC<AddManagedIdentityProps> = () => {
|
||||||
|
const { copyJobState: { source } = {} } = useCopyJobContext();
|
||||||
|
const sourceAccountLink = buildResourceLink(source?.account);
|
||||||
|
const onlineCopyUrl = `${sourceAccountLink}/Features`;
|
||||||
|
const onWindowClosed = () => {
|
||||||
|
// eslint-disable-next-line no-console
|
||||||
|
console.log("Online copy window closed");
|
||||||
|
};
|
||||||
|
const openWindowAndMonitor = useWindowOpenMonitor(onlineCopyUrl, onWindowClosed);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Stack className="onlineCopyContainer" tokens={{ childrenGap: 15, padding: "0 0 0 20px" }}>
|
||||||
|
<div className="toggle-label">{ContainerCopyMessages.onlineCopyEnabled.description}</div>
|
||||||
|
<PrimaryButton text={ContainerCopyMessages.onlineCopyEnabled.buttonText} onClick={openWindowAndMonitor} />
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default OnlineCopyEnabled;
|
||||||
@@ -0,0 +1,55 @@
|
|||||||
|
import { PrimaryButton, Stack } from "@fluentui/react";
|
||||||
|
import React, { useCallback, useState } from "react";
|
||||||
|
import { fetchDatabaseAccount } from "Utils/arm/databaseAccountUtils";
|
||||||
|
import ContainerCopyMessages from "../../../ContainerCopyMessages";
|
||||||
|
import { useCopyJobContext } from "../../../Context/CopyJobContext";
|
||||||
|
import { buildResourceLink, getAccountDetailsFromResourceId } from "../../../CopyJobUtils";
|
||||||
|
import { PermissionSectionConfig } from "./hooks/usePermissionsSection";
|
||||||
|
import useWindowOpenMonitor from "./hooks/useWindowOpenMonitor";
|
||||||
|
|
||||||
|
type AddManagedIdentityProps = Partial<PermissionSectionConfig>;
|
||||||
|
const PointInTimeRestore: React.FC<AddManagedIdentityProps> = () => {
|
||||||
|
const [loading, setLoading] = useState<boolean>(false);
|
||||||
|
const { copyJobState: { source } = {}, setCopyJobState } = useCopyJobContext();
|
||||||
|
const sourceAccountLink = buildResourceLink(source?.account);
|
||||||
|
const pitrUrl = `${sourceAccountLink}/backupRestore`;
|
||||||
|
|
||||||
|
const onWindowClosed = useCallback(async () => {
|
||||||
|
try {
|
||||||
|
const selectedSourceAccount = source?.account;
|
||||||
|
const {
|
||||||
|
subscriptionId: sourceSubscriptionId,
|
||||||
|
resourceGroup: sourceResourceGroup,
|
||||||
|
accountName: sourceAccountName,
|
||||||
|
} = getAccountDetailsFromResourceId(selectedSourceAccount?.id);
|
||||||
|
|
||||||
|
setLoading(true);
|
||||||
|
const account = await fetchDatabaseAccount(sourceSubscriptionId, sourceResourceGroup, sourceAccountName);
|
||||||
|
if (account) {
|
||||||
|
setCopyJobState((prevState) => ({
|
||||||
|
...prevState,
|
||||||
|
source: { ...prevState.source, account: account },
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error fetching database account after PITR window closed:", error);
|
||||||
|
} finally {
|
||||||
|
setLoading(false);
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
const openWindowAndMonitor = useWindowOpenMonitor(pitrUrl, onWindowClosed);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Stack className="pointInTimeRestoreContainer" tokens={{ childrenGap: 15, padding: "0 0 0 20px" }}>
|
||||||
|
<div className="toggle-label">{ContainerCopyMessages.pointInTimeRestore.description}</div>
|
||||||
|
<PrimaryButton
|
||||||
|
text={loading ? "" : ContainerCopyMessages.pointInTimeRestore.buttonText}
|
||||||
|
{...(loading ? { iconProps: { iconName: "SyncStatusSolid" } } : {})}
|
||||||
|
disabled={loading}
|
||||||
|
onClick={openWindowAndMonitor}
|
||||||
|
/>
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default PointInTimeRestore;
|
||||||
@@ -0,0 +1,52 @@
|
|||||||
|
import { DatabaseAccount } from "Contracts/DataModels";
|
||||||
|
import { useCallback, useState } from "react";
|
||||||
|
import { useCopyJobContext } from "../../../../Context/CopyJobContext";
|
||||||
|
import { getAccountDetailsFromResourceId } from "../../../../CopyJobUtils";
|
||||||
|
|
||||||
|
interface UseManagedIdentityUpdaterParams {
|
||||||
|
updateIdentityFn: (
|
||||||
|
subscriptionId: string,
|
||||||
|
resourceGroup?: string,
|
||||||
|
accountName?: string,
|
||||||
|
) => Promise<DatabaseAccount | undefined>;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface UseManagedIdentityUpdaterReturn {
|
||||||
|
loading: boolean;
|
||||||
|
handleAddSystemIdentity: () => Promise<void>;
|
||||||
|
}
|
||||||
|
|
||||||
|
const useManagedIdentity = (
|
||||||
|
updateIdentityFn: UseManagedIdentityUpdaterParams["updateIdentityFn"],
|
||||||
|
): UseManagedIdentityUpdaterReturn => {
|
||||||
|
const { copyJobState, setCopyJobState } = useCopyJobContext();
|
||||||
|
const [loading, setLoading] = useState<boolean>(false);
|
||||||
|
|
||||||
|
const handleAddSystemIdentity = useCallback(async (): Promise<void> => {
|
||||||
|
try {
|
||||||
|
setLoading(true);
|
||||||
|
const selectedTargetAccount = copyJobState?.target?.account;
|
||||||
|
const {
|
||||||
|
subscriptionId: targetSubscriptionId,
|
||||||
|
resourceGroup: targetResourceGroup,
|
||||||
|
accountName: targetAccountName,
|
||||||
|
} = getAccountDetailsFromResourceId(selectedTargetAccount?.id);
|
||||||
|
|
||||||
|
const updatedAccount = await updateIdentityFn(targetSubscriptionId, targetResourceGroup, targetAccountName);
|
||||||
|
if (updatedAccount) {
|
||||||
|
setCopyJobState((prevState) => ({
|
||||||
|
...prevState,
|
||||||
|
target: { ...prevState.target, account: updatedAccount },
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error enabling system-assigned managed identity:", error);
|
||||||
|
} finally {
|
||||||
|
setLoading(false);
|
||||||
|
}
|
||||||
|
}, [copyJobState, updateIdentityFn, setCopyJobState]);
|
||||||
|
|
||||||
|
return { loading, handleAddSystemIdentity };
|
||||||
|
};
|
||||||
|
|
||||||
|
export default useManagedIdentity;
|
||||||
@@ -0,0 +1,198 @@
|
|||||||
|
import { useEffect, useMemo, useRef, useState } from "react";
|
||||||
|
import { fetchRoleAssignments, fetchRoleDefinitions, RoleDefinitionType } from "../../../../../../Utils/arm/RbacUtils";
|
||||||
|
import ContainerCopyMessages from "../../../../ContainerCopyMessages";
|
||||||
|
import { getAccountDetailsFromResourceId } from "../../../../CopyJobUtils";
|
||||||
|
import { BackupPolicyType, CopyJobMigrationType, DefaultIdentityType, IdentityType } from "../../../../Enums";
|
||||||
|
import { CopyJobContextState } from "../../../../Types";
|
||||||
|
import { useCopyJobPrerequisitesCache } from "../../../Utils/useCopyJobPrerequisitesCache";
|
||||||
|
import AddManagedIdentity from "../AddManagedIdentity";
|
||||||
|
import AddReadPermissionToDefaultIdentity from "../AddReadPermissionToDefaultIdentity";
|
||||||
|
import DefaultManagedIdentity from "../DefaultManagedIdentity";
|
||||||
|
import OnlineCopyEnabled from "../OnlineCopyEnabled";
|
||||||
|
import PointInTimeRestore from "../PointInTimeRestore";
|
||||||
|
|
||||||
|
export interface PermissionSectionConfig {
|
||||||
|
id: string;
|
||||||
|
title: string;
|
||||||
|
Component: React.ComponentType;
|
||||||
|
disabled: boolean;
|
||||||
|
completed?: boolean;
|
||||||
|
validate?: (state: CopyJobContextState) => boolean | Promise<boolean>;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Section IDs for maintainability
|
||||||
|
export const SECTION_IDS = {
|
||||||
|
addManagedIdentity: "addManagedIdentity",
|
||||||
|
defaultManagedIdentity: "defaultManagedIdentity",
|
||||||
|
readPermissionAssigned: "readPermissionAssigned",
|
||||||
|
pointInTimeRestore: "pointInTimeRestore",
|
||||||
|
onlineCopyEnabled: "onlineCopyEnabled",
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
const PERMISSION_SECTIONS_CONFIG: PermissionSectionConfig[] = [
|
||||||
|
{
|
||||||
|
id: SECTION_IDS.addManagedIdentity,
|
||||||
|
title: ContainerCopyMessages.addManagedIdentity.title,
|
||||||
|
Component: AddManagedIdentity,
|
||||||
|
disabled: true,
|
||||||
|
validate: (state: CopyJobContextState) => {
|
||||||
|
const targetAccountIdentityType = (state?.target?.account?.identity?.type ?? "").toLowerCase();
|
||||||
|
return (
|
||||||
|
targetAccountIdentityType === IdentityType.SystemAssigned ||
|
||||||
|
targetAccountIdentityType === IdentityType.UserAssigned
|
||||||
|
);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: SECTION_IDS.defaultManagedIdentity,
|
||||||
|
title: ContainerCopyMessages.defaultManagedIdentity.title,
|
||||||
|
Component: DefaultManagedIdentity,
|
||||||
|
disabled: true,
|
||||||
|
validate: (state: CopyJobContextState) => {
|
||||||
|
const targetAccountDefaultIdentity = (state?.target?.account?.properties?.defaultIdentity ?? "").toLowerCase();
|
||||||
|
return targetAccountDefaultIdentity === DefaultIdentityType.SystemAssignedIdentity;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: SECTION_IDS.readPermissionAssigned,
|
||||||
|
title: ContainerCopyMessages.readPermissionAssigned.title,
|
||||||
|
Component: AddReadPermissionToDefaultIdentity,
|
||||||
|
disabled: true,
|
||||||
|
validate: async (state: CopyJobContextState) => {
|
||||||
|
const principalId = state?.target?.account?.identity?.principalId;
|
||||||
|
const selectedSourceAccount = state?.source?.account;
|
||||||
|
const {
|
||||||
|
subscriptionId: sourceSubscriptionId,
|
||||||
|
resourceGroup: sourceResourceGroup,
|
||||||
|
accountName: sourceAccountName,
|
||||||
|
} = getAccountDetailsFromResourceId(selectedSourceAccount?.id);
|
||||||
|
|
||||||
|
const rolesAssigned = await fetchRoleAssignments(
|
||||||
|
sourceSubscriptionId,
|
||||||
|
sourceResourceGroup,
|
||||||
|
sourceAccountName,
|
||||||
|
principalId,
|
||||||
|
);
|
||||||
|
|
||||||
|
const roleDefinitions = await fetchRoleDefinitions(rolesAssigned ?? []);
|
||||||
|
return checkTargetHasReaderRoleOnSource(roleDefinitions ?? []);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const PERMISSION_SECTIONS_FOR_ONLINE_JOBS: PermissionSectionConfig[] = [
|
||||||
|
{
|
||||||
|
id: SECTION_IDS.pointInTimeRestore,
|
||||||
|
title: ContainerCopyMessages.pointInTimeRestore.title,
|
||||||
|
Component: PointInTimeRestore,
|
||||||
|
disabled: true,
|
||||||
|
validate: (state: CopyJobContextState) => {
|
||||||
|
const sourceAccountBackupPolicy = state?.source?.account?.properties?.backupPolicy?.type ?? "";
|
||||||
|
return sourceAccountBackupPolicy === BackupPolicyType.Continuous;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: SECTION_IDS.onlineCopyEnabled,
|
||||||
|
title: ContainerCopyMessages.onlineCopyEnabled.title,
|
||||||
|
Component: OnlineCopyEnabled,
|
||||||
|
disabled: true,
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
||||||
|
validate: (_state: CopyJobContextState) => {
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the user has the Reader role based on role definitions.
|
||||||
|
*/
|
||||||
|
export function checkTargetHasReaderRoleOnSource(roleDefinitions: RoleDefinitionType[]): boolean {
|
||||||
|
return roleDefinitions?.some(
|
||||||
|
(role) =>
|
||||||
|
role.name === "00000000-0000-0000-0000-000000000001" ||
|
||||||
|
role.permissions.some(
|
||||||
|
(permission) =>
|
||||||
|
permission.dataActions.includes("Microsoft.DocumentDB/databaseAccounts/readMetadata") &&
|
||||||
|
permission.dataActions.includes("Microsoft.DocumentDB/databaseAccounts/sqlDatabases/containers/items/read"),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the permission sections configuration for the Assign Permissions screen.
|
||||||
|
* Memoizes derived values for performance and decouples logic for testability.
|
||||||
|
*/
|
||||||
|
const usePermissionSections = (state: CopyJobContextState): PermissionSectionConfig[] => {
|
||||||
|
const { validationCache, setValidationCache } = useCopyJobPrerequisitesCache();
|
||||||
|
const [permissionSections, setPermissionSections] = useState<PermissionSectionConfig[] | null>(null);
|
||||||
|
const isValidatingRef = useRef(false);
|
||||||
|
|
||||||
|
const sectionToValidate = useMemo(() => {
|
||||||
|
const baseSections = [...PERMISSION_SECTIONS_CONFIG];
|
||||||
|
if (state.migrationType === CopyJobMigrationType.Online) {
|
||||||
|
return [...baseSections, ...PERMISSION_SECTIONS_FOR_ONLINE_JOBS];
|
||||||
|
}
|
||||||
|
return baseSections;
|
||||||
|
}, [state.migrationType]);
|
||||||
|
|
||||||
|
const memoizedValidationCache = useMemo(() => {
|
||||||
|
if (state.migrationType === CopyJobMigrationType.Offline) {
|
||||||
|
validationCache.delete(SECTION_IDS.pointInTimeRestore);
|
||||||
|
validationCache.delete(SECTION_IDS.onlineCopyEnabled);
|
||||||
|
}
|
||||||
|
return validationCache;
|
||||||
|
}, [state.migrationType]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const validateSections = async () => {
|
||||||
|
if (isValidatingRef.current) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
isValidatingRef.current = true;
|
||||||
|
const result: PermissionSectionConfig[] = [];
|
||||||
|
const newValidationCache = new Map(memoizedValidationCache);
|
||||||
|
|
||||||
|
for (let i = 0; i < sectionToValidate.length; i++) {
|
||||||
|
const section = sectionToValidate[i];
|
||||||
|
|
||||||
|
// Check if this section was already validated and passed
|
||||||
|
if (newValidationCache.has(section.id) && newValidationCache.get(section.id) === true) {
|
||||||
|
result.push({ ...section, completed: true });
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
// We've reached the first non-cached section - validate it
|
||||||
|
if (section.validate) {
|
||||||
|
const isValid = await section.validate(state);
|
||||||
|
newValidationCache.set(section.id, isValid);
|
||||||
|
result.push({ ...section, completed: isValid });
|
||||||
|
// Stop validation if current section failed
|
||||||
|
if (!isValid) {
|
||||||
|
for (let j = i + 1; j < sectionToValidate.length; j++) {
|
||||||
|
result.push({ ...sectionToValidate[j], completed: false });
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Section has no validate method
|
||||||
|
newValidationCache.set(section.id, false);
|
||||||
|
result.push({ ...section, completed: false });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setValidationCache(newValidationCache);
|
||||||
|
setPermissionSections(result);
|
||||||
|
isValidatingRef.current = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
validateSections();
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
isValidatingRef.current = false;
|
||||||
|
};
|
||||||
|
}, [state, sectionToValidate]);
|
||||||
|
|
||||||
|
return permissionSections ?? [];
|
||||||
|
};
|
||||||
|
|
||||||
|
export default usePermissionSections;
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
import { useCallback, useState } from "react";
|
||||||
|
|
||||||
|
const useToggle = (initialState = false) => {
|
||||||
|
const [state, setState] = useState<boolean>(initialState);
|
||||||
|
const onToggle = useCallback((_, checked?: boolean) => {
|
||||||
|
setState(!!checked);
|
||||||
|
}, []);
|
||||||
|
return [state, onToggle] as const;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default useToggle;
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
import { useEffect, useRef } from "react";
|
||||||
|
|
||||||
|
const useWindowOpenMonitor = (url: string, onClose?: () => void, intervalMs = 500) => {
|
||||||
|
const intervalRef = useRef<NodeJS.Timeout | null>(null);
|
||||||
|
|
||||||
|
const openWindowAndMonitor = () => {
|
||||||
|
const newWindow = window.open(url, "_blank");
|
||||||
|
intervalRef.current = setInterval(() => {
|
||||||
|
if (newWindow?.closed) {
|
||||||
|
clearInterval(intervalRef.current!);
|
||||||
|
intervalRef.current = null;
|
||||||
|
if (onClose) {
|
||||||
|
onClose();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, intervalMs);
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
return () => {
|
||||||
|
if (intervalRef.current) {
|
||||||
|
clearInterval(intervalRef.current);
|
||||||
|
intervalRef.current = null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return openWindowAndMonitor;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default useWindowOpenMonitor;
|
||||||
@@ -0,0 +1,66 @@
|
|||||||
|
import { Image, Stack, Text } from "@fluentui/react";
|
||||||
|
import { Accordion, AccordionHeader, AccordionItem, AccordionPanel } from "@fluentui/react-components";
|
||||||
|
import React, { useEffect } from "react";
|
||||||
|
import CheckmarkIcon from "../../../../../../images/successfulPopup.svg";
|
||||||
|
import WarningIcon from "../../../../../../images/warning.svg";
|
||||||
|
import ShimmerTree, { IndentLevel } from "../../../../../Common/ShimmerTree";
|
||||||
|
import ContainerCopyMessages from "../../../ContainerCopyMessages";
|
||||||
|
import { useCopyJobContext } from "../../../Context/CopyJobContext";
|
||||||
|
import { CopyJobMigrationType } from "../../../Enums";
|
||||||
|
import usePermissionSections, { PermissionSectionConfig } from "./hooks/usePermissionsSection";
|
||||||
|
|
||||||
|
const PermissionSection: React.FC<PermissionSectionConfig> = ({ id, title, Component, completed, disabled }) => (
|
||||||
|
<AccordionItem key={id} value={id} disabled={disabled}>
|
||||||
|
<AccordionHeader className="accordionHeader">
|
||||||
|
<Text className="accordionHeaderText" variant="medium">
|
||||||
|
{title}
|
||||||
|
</Text>
|
||||||
|
<Image
|
||||||
|
className="statusIcon"
|
||||||
|
src={completed ? CheckmarkIcon : WarningIcon}
|
||||||
|
alt={completed ? "Checkmark icon" : "Warning icon"}
|
||||||
|
width={completed ? 20 : 24}
|
||||||
|
height={completed ? 20 : 24}
|
||||||
|
/>
|
||||||
|
</AccordionHeader>
|
||||||
|
<AccordionPanel aria-disabled={disabled} className="accordionPanel">
|
||||||
|
<Component />
|
||||||
|
</AccordionPanel>
|
||||||
|
</AccordionItem>
|
||||||
|
);
|
||||||
|
|
||||||
|
const AssignPermissions = () => {
|
||||||
|
const { copyJobState } = useCopyJobContext();
|
||||||
|
const permissionSections = usePermissionSections(copyJobState);
|
||||||
|
const [openItems, setOpenItems] = React.useState<string[]>([]);
|
||||||
|
|
||||||
|
const indentLevels = React.useMemo<IndentLevel[]>(
|
||||||
|
() => Array(copyJobState.migrationType === CopyJobMigrationType.Online ? 5 : 3).fill({ level: 0, width: "100%" }),
|
||||||
|
[],
|
||||||
|
);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const firstIncompleteSection = permissionSections.find((section) => !section.completed);
|
||||||
|
const nextOpenItems = firstIncompleteSection ? [firstIncompleteSection.id] : [];
|
||||||
|
if (JSON.stringify(openItems) !== JSON.stringify(nextOpenItems)) {
|
||||||
|
setOpenItems(nextOpenItems);
|
||||||
|
}
|
||||||
|
}, [permissionSections]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Stack className="assignPermissionsContainer" tokens={{ childrenGap: 15 }}>
|
||||||
|
<span>{ContainerCopyMessages.assignPermissions.description}</span>
|
||||||
|
{permissionSections?.length === 0 ? (
|
||||||
|
<ShimmerTree indentLevels={indentLevels} style={{ width: "100%" }} />
|
||||||
|
) : (
|
||||||
|
<Accordion className="permissionsAccordion" collapsible openItems={openItems}>
|
||||||
|
{permissionSections.map((section) => (
|
||||||
|
<PermissionSection key={section.id} {...section} />
|
||||||
|
))}
|
||||||
|
</Accordion>
|
||||||
|
)}
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default AssignPermissions;
|
||||||
@@ -0,0 +1,25 @@
|
|||||||
|
import { Stack } from "@fluentui/react";
|
||||||
|
import React from "react";
|
||||||
|
|
||||||
|
interface FieldRowProps {
|
||||||
|
label?: string;
|
||||||
|
children: React.ReactNode;
|
||||||
|
labelClassName?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const FieldRow: React.FC<FieldRowProps> = ({ label = "", children, labelClassName = "" }) => {
|
||||||
|
return (
|
||||||
|
<Stack horizontal horizontalAlign="space-between" className="flex-row">
|
||||||
|
{label && (
|
||||||
|
<Stack.Item align="center" className="flex-fixed-width">
|
||||||
|
<label className={`field-label ${labelClassName}`}>{label}: </label>
|
||||||
|
</Stack.Item>
|
||||||
|
)}
|
||||||
|
<Stack.Item align="center" className="flex-grow-col">
|
||||||
|
{children}
|
||||||
|
</Stack.Item>
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default FieldRow;
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
import { Image, ITooltipHostStyles, TooltipHost } from "@fluentui/react";
|
||||||
|
import React from "react";
|
||||||
|
import InfoIcon from "../../../../../../images/Info.svg";
|
||||||
|
|
||||||
|
const InfoTooltip: React.FC<{ content?: string }> = ({ content }) => {
|
||||||
|
if (!content) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
const hostStyles: Partial<ITooltipHostStyles> = { root: { display: "inline-block" } };
|
||||||
|
return (
|
||||||
|
<TooltipHost content={content} calloutProps={{ gapSpace: 0 }} styles={hostStyles}>
|
||||||
|
<Image src={InfoIcon} alt="Information" width={14} height={14} />
|
||||||
|
</TooltipHost>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default React.memo(InfoTooltip);
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
import { DefaultButton, PrimaryButton, Stack } from "@fluentui/react";
|
||||||
|
import React from "react";
|
||||||
|
|
||||||
|
type NavigationControlsProps = {
|
||||||
|
primaryBtnText: string;
|
||||||
|
onPrimary: () => void;
|
||||||
|
onPrevious: () => void;
|
||||||
|
onCancel: () => void;
|
||||||
|
isPrimaryDisabled: boolean;
|
||||||
|
isPreviousDisabled: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
const NavigationControls: React.FC<NavigationControlsProps> = ({
|
||||||
|
primaryBtnText,
|
||||||
|
onPrimary,
|
||||||
|
onPrevious,
|
||||||
|
onCancel,
|
||||||
|
isPrimaryDisabled,
|
||||||
|
isPreviousDisabled,
|
||||||
|
}) => (
|
||||||
|
<Stack horizontal tokens={{ childrenGap: 20 }}>
|
||||||
|
<PrimaryButton text={primaryBtnText} onClick={onPrimary} allowDisabledFocus disabled={isPrimaryDisabled} />
|
||||||
|
<DefaultButton text="Previous" onClick={onPrevious} allowDisabledFocus disabled={isPreviousDisabled} />
|
||||||
|
<DefaultButton text="Cancel" onClick={onCancel} />
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
|
||||||
|
export default React.memo(NavigationControls);
|
||||||
@@ -0,0 +1,67 @@
|
|||||||
|
/* eslint-disable react/prop-types */
|
||||||
|
/* eslint-disable react/display-name */
|
||||||
|
import { DefaultButton, PrimaryButton, Stack, Text } from "@fluentui/react";
|
||||||
|
import React from "react";
|
||||||
|
|
||||||
|
interface PopoverContainerProps {
|
||||||
|
isLoading?: boolean;
|
||||||
|
title?: string;
|
||||||
|
children?: React.ReactNode;
|
||||||
|
onPrimary: () => void;
|
||||||
|
onCancel: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const PopoverContainer: React.FC<PopoverContainerProps> = React.memo(
|
||||||
|
({ isLoading = false, title, children, onPrimary, onCancel }) => {
|
||||||
|
return (
|
||||||
|
<Stack
|
||||||
|
className={`popover-container foreground ${isLoading ? "loading" : ""}`}
|
||||||
|
tokens={{ childrenGap: 20 }}
|
||||||
|
style={{ maxWidth: 450 }}
|
||||||
|
>
|
||||||
|
<Text variant="mediumPlus" style={{ fontWeight: 600 }}>
|
||||||
|
{title}
|
||||||
|
</Text>
|
||||||
|
<Text>{children}</Text>
|
||||||
|
<Stack horizontal tokens={{ childrenGap: 20 }}>
|
||||||
|
<PrimaryButton
|
||||||
|
text={isLoading ? "" : "Yes"}
|
||||||
|
{...(isLoading ? { iconProps: { iconName: "SyncStatusSolid" } } : {})}
|
||||||
|
onClick={onPrimary}
|
||||||
|
disabled={isLoading}
|
||||||
|
/>
|
||||||
|
<DefaultButton text="No" onClick={onCancel} disabled={isLoading} />
|
||||||
|
</Stack>
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
interface PopoverMessageProps {
|
||||||
|
isLoading?: boolean;
|
||||||
|
visible: boolean;
|
||||||
|
title: string;
|
||||||
|
onCancel: () => void;
|
||||||
|
onPrimary: () => void;
|
||||||
|
children: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
const PopoverMessage: React.FC<PopoverMessageProps> = ({
|
||||||
|
isLoading = false,
|
||||||
|
visible,
|
||||||
|
title,
|
||||||
|
onCancel,
|
||||||
|
onPrimary,
|
||||||
|
children,
|
||||||
|
}) => {
|
||||||
|
if (!visible) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<PopoverContainer title={title} onCancel={onCancel} onPrimary={onPrimary} isLoading={isLoading}>
|
||||||
|
{children}
|
||||||
|
</PopoverContainer>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default PopoverMessage;
|
||||||
@@ -0,0 +1,34 @@
|
|||||||
|
import { Stack } from "@fluentui/react";
|
||||||
|
import React from "react";
|
||||||
|
import { useCopyJobNavigation } from "../Utils/useCopyJobNavigation";
|
||||||
|
import NavigationControls from "./Components/NavigationControls";
|
||||||
|
|
||||||
|
const CreateCopyJobScreens: React.FC = () => {
|
||||||
|
const {
|
||||||
|
currentScreen,
|
||||||
|
isPrimaryDisabled,
|
||||||
|
isPreviousDisabled,
|
||||||
|
handlePrimary,
|
||||||
|
handlePrevious,
|
||||||
|
handleCancel,
|
||||||
|
primaryBtnText,
|
||||||
|
} = useCopyJobNavigation();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Stack verticalAlign="space-between" className="createCopyJobScreensContainer">
|
||||||
|
<Stack.Item className="createCopyJobScreensContent">{currentScreen?.component}</Stack.Item>
|
||||||
|
<Stack.Item className="createCopyJobScreensFooter">
|
||||||
|
<NavigationControls
|
||||||
|
primaryBtnText={primaryBtnText}
|
||||||
|
onPrimary={handlePrimary}
|
||||||
|
onPrevious={handlePrevious}
|
||||||
|
onCancel={handleCancel}
|
||||||
|
isPrimaryDisabled={isPrimaryDisabled}
|
||||||
|
isPreviousDisabled={isPreviousDisabled}
|
||||||
|
/>
|
||||||
|
</Stack.Item>
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default CreateCopyJobScreens;
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
import React from "react";
|
||||||
|
import CopyJobContextProvider from "../../Context/CopyJobContext";
|
||||||
|
import CreateCopyJobScreens from "./CreateCopyJobScreens";
|
||||||
|
|
||||||
|
const CreateCopyJobScreensProvider = () => {
|
||||||
|
return (
|
||||||
|
<CopyJobContextProvider>
|
||||||
|
<CreateCopyJobScreens />
|
||||||
|
</CopyJobContextProvider>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default CreateCopyJobScreensProvider;
|
||||||
@@ -0,0 +1,43 @@
|
|||||||
|
import { IColumn } from "@fluentui/react";
|
||||||
|
import ContainerCopyMessages from "../../../../ContainerCopyMessages";
|
||||||
|
|
||||||
|
const commonProps = {
|
||||||
|
minWidth: 130,
|
||||||
|
maxWidth: 140,
|
||||||
|
styles: {
|
||||||
|
root: {
|
||||||
|
whiteSpace: "normal",
|
||||||
|
lineHeight: "1.2",
|
||||||
|
wordBreak: "break-word",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getPreviewCopyJobDetailsListColumns = (): IColumn[] => {
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
key: "sourcedbname",
|
||||||
|
name: ContainerCopyMessages.sourceDatabaseLabel,
|
||||||
|
fieldName: "sourceDatabaseName",
|
||||||
|
...commonProps,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "sourcecolname",
|
||||||
|
name: ContainerCopyMessages.sourceContainerLabel,
|
||||||
|
fieldName: "sourceContainerName",
|
||||||
|
...commonProps,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "targetdbname",
|
||||||
|
name: ContainerCopyMessages.targetDatabaseLabel,
|
||||||
|
fieldName: "targetDatabaseName",
|
||||||
|
...commonProps,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "targetcolname",
|
||||||
|
name: ContainerCopyMessages.targetContainerLabel,
|
||||||
|
fieldName: "targetContainerName",
|
||||||
|
...commonProps,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
};
|
||||||
@@ -0,0 +1,52 @@
|
|||||||
|
import { DetailsList, DetailsListLayoutMode, Stack, Text, TextField } from "@fluentui/react";
|
||||||
|
import FieldRow from "Explorer/ContainerCopy/CreateCopyJob/Screens/Components/FieldRow";
|
||||||
|
import React from "react";
|
||||||
|
import ContainerCopyMessages from "../../../ContainerCopyMessages";
|
||||||
|
import { useCopyJobContext } from "../../../Context/CopyJobContext";
|
||||||
|
import { getPreviewCopyJobDetailsListColumns } from "./Utils/PreviewCopyJobUtils";
|
||||||
|
|
||||||
|
const PreviewCopyJob: React.FC = () => {
|
||||||
|
const { copyJobState, setCopyJobState } = useCopyJobContext();
|
||||||
|
|
||||||
|
const selectedDatabaseAndContainers = [
|
||||||
|
{
|
||||||
|
sourceDatabaseName: copyJobState.source?.databaseId,
|
||||||
|
sourceContainerName: copyJobState.source?.containerId,
|
||||||
|
targetDatabaseName: copyJobState.target?.databaseId,
|
||||||
|
targetContainerName: copyJobState.target?.containerId,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
const jobName = copyJobState.jobName;
|
||||||
|
|
||||||
|
const onJobNameChange = (_ev?: React.FormEvent, newValue?: string) => {
|
||||||
|
setCopyJobState((prevState) => ({
|
||||||
|
...prevState,
|
||||||
|
jobName: newValue || "",
|
||||||
|
}));
|
||||||
|
};
|
||||||
|
return (
|
||||||
|
<Stack tokens={{ childrenGap: 20 }} className="previewCopyJobContainer">
|
||||||
|
<FieldRow label={ContainerCopyMessages.jobNameLabel}>
|
||||||
|
<TextField value={jobName} onChange={onJobNameChange} />
|
||||||
|
</FieldRow>
|
||||||
|
<Stack>
|
||||||
|
<Text className="bold">{ContainerCopyMessages.sourceSubscriptionLabel}</Text>
|
||||||
|
<Text>{copyJobState.source?.subscription?.displayName}</Text>
|
||||||
|
</Stack>
|
||||||
|
<Stack>
|
||||||
|
<Text className="bold">{ContainerCopyMessages.sourceAccountLabel}</Text>
|
||||||
|
<Text>{copyJobState.source?.account?.name}</Text>
|
||||||
|
</Stack>
|
||||||
|
<Stack>
|
||||||
|
<DetailsList
|
||||||
|
items={selectedDatabaseAndContainers}
|
||||||
|
layoutMode={DetailsListLayoutMode.justified}
|
||||||
|
checkboxVisibility={2}
|
||||||
|
columns={getPreviewCopyJobDetailsListColumns()}
|
||||||
|
/>
|
||||||
|
</Stack>
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default PreviewCopyJob;
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
/* eslint-disable react/prop-types */
|
||||||
|
/* eslint-disable react/display-name */
|
||||||
|
import { Dropdown } from "@fluentui/react";
|
||||||
|
import React from "react";
|
||||||
|
import ContainerCopyMessages from "../../../../ContainerCopyMessages";
|
||||||
|
import { DropdownOptionType } from "../../../../Types";
|
||||||
|
import FieldRow from "../../Components/FieldRow";
|
||||||
|
|
||||||
|
interface AccountDropdownProps {
|
||||||
|
options: DropdownOptionType[];
|
||||||
|
selectedKey?: string;
|
||||||
|
disabled: boolean;
|
||||||
|
onChange: (_ev?: React.FormEvent, option?: DropdownOptionType) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const AccountDropdown: React.FC<AccountDropdownProps> = React.memo(
|
||||||
|
({ options, selectedKey, disabled, onChange }) => (
|
||||||
|
<FieldRow label={ContainerCopyMessages.sourceAccountDropdownLabel}>
|
||||||
|
<Dropdown
|
||||||
|
placeholder={ContainerCopyMessages.sourceAccountDropdownPlaceholder}
|
||||||
|
ariaLabel={ContainerCopyMessages.sourceAccountDropdownLabel}
|
||||||
|
options={options}
|
||||||
|
disabled={disabled}
|
||||||
|
required
|
||||||
|
selectedKey={selectedKey}
|
||||||
|
onChange={onChange}
|
||||||
|
/>
|
||||||
|
</FieldRow>
|
||||||
|
),
|
||||||
|
);
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
/* eslint-disable react/prop-types */
|
||||||
|
/* eslint-disable react/display-name */
|
||||||
|
import { Checkbox, Stack } from "@fluentui/react";
|
||||||
|
import React from "react";
|
||||||
|
import ContainerCopyMessages from "../../../../ContainerCopyMessages";
|
||||||
|
|
||||||
|
interface MigrationTypeCheckboxProps {
|
||||||
|
checked: boolean;
|
||||||
|
onChange: (_ev?: React.FormEvent, checked?: boolean) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const MigrationTypeCheckbox: React.FC<MigrationTypeCheckboxProps> = React.memo(({ checked, onChange }) => (
|
||||||
|
<Stack horizontal horizontalAlign="space-between" className="migrationTypeRow">
|
||||||
|
<Checkbox label={ContainerCopyMessages.migrationTypeCheckboxLabel} checked={checked} onChange={onChange} />
|
||||||
|
</Stack>
|
||||||
|
));
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
/* eslint-disable react/prop-types */
|
||||||
|
/* eslint-disable react/display-name */
|
||||||
|
import { Dropdown } from "@fluentui/react";
|
||||||
|
import React from "react";
|
||||||
|
import ContainerCopyMessages from "../../../../ContainerCopyMessages";
|
||||||
|
import { DropdownOptionType } from "../../../../Types";
|
||||||
|
import FieldRow from "../../Components/FieldRow";
|
||||||
|
|
||||||
|
interface SubscriptionDropdownProps {
|
||||||
|
options: DropdownOptionType[];
|
||||||
|
selectedKey?: string;
|
||||||
|
onChange: (_ev?: React.FormEvent, option?: DropdownOptionType) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SubscriptionDropdown: React.FC<SubscriptionDropdownProps> = React.memo(
|
||||||
|
({ options, selectedKey, onChange }) => (
|
||||||
|
<FieldRow label={ContainerCopyMessages.subscriptionDropdownLabel}>
|
||||||
|
<Dropdown
|
||||||
|
placeholder={ContainerCopyMessages.subscriptionDropdownPlaceholder}
|
||||||
|
ariaLabel={ContainerCopyMessages.subscriptionDropdownLabel}
|
||||||
|
options={options}
|
||||||
|
required
|
||||||
|
selectedKey={selectedKey}
|
||||||
|
onChange={onChange}
|
||||||
|
/>
|
||||||
|
</FieldRow>
|
||||||
|
),
|
||||||
|
);
|
||||||
@@ -0,0 +1,78 @@
|
|||||||
|
import React from "react";
|
||||||
|
import { DatabaseAccount, Subscription } from "../../../../../../Contracts/DataModels";
|
||||||
|
import { CopyJobMigrationType } from "../../../../Enums";
|
||||||
|
import { CopyJobContextProviderType, CopyJobContextState, DropdownOptionType } from "../../../../Types";
|
||||||
|
|
||||||
|
export function useDropdownOptions(
|
||||||
|
subscriptions: Subscription[],
|
||||||
|
accounts: DatabaseAccount[],
|
||||||
|
): {
|
||||||
|
subscriptionOptions: DropdownOptionType[];
|
||||||
|
accountOptions: DropdownOptionType[];
|
||||||
|
} {
|
||||||
|
const subscriptionOptions = React.useMemo(
|
||||||
|
() =>
|
||||||
|
subscriptions?.map((sub) => ({
|
||||||
|
key: sub.subscriptionId,
|
||||||
|
text: sub.displayName,
|
||||||
|
data: sub,
|
||||||
|
})) || [],
|
||||||
|
[subscriptions],
|
||||||
|
);
|
||||||
|
|
||||||
|
const accountOptions = React.useMemo(
|
||||||
|
() =>
|
||||||
|
accounts?.map((account) => ({
|
||||||
|
key: account.id,
|
||||||
|
text: account.name,
|
||||||
|
data: account,
|
||||||
|
})) || [],
|
||||||
|
[accounts],
|
||||||
|
);
|
||||||
|
|
||||||
|
return { subscriptionOptions, accountOptions };
|
||||||
|
}
|
||||||
|
|
||||||
|
type setCopyJobStateType = CopyJobContextProviderType["setCopyJobState"];
|
||||||
|
|
||||||
|
export function useEventHandlers(setCopyJobState: setCopyJobStateType) {
|
||||||
|
const handleSelectSourceAccount = React.useCallback(
|
||||||
|
(type: "subscription" | "account", data: (Subscription & DatabaseAccount) | undefined) => {
|
||||||
|
setCopyJobState((prevState: CopyJobContextState) => {
|
||||||
|
if (type === "subscription") {
|
||||||
|
return {
|
||||||
|
...prevState,
|
||||||
|
source: {
|
||||||
|
...prevState.source,
|
||||||
|
subscription: data || null,
|
||||||
|
account: null, // reset on subscription change
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
if (type === "account") {
|
||||||
|
return {
|
||||||
|
...prevState,
|
||||||
|
source: {
|
||||||
|
...prevState.source,
|
||||||
|
account: data || null,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return prevState;
|
||||||
|
});
|
||||||
|
},
|
||||||
|
[setCopyJobState],
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleMigrationTypeChange = React.useCallback(
|
||||||
|
(_ev?: React.FormEvent<HTMLElement>, checked?: boolean) => {
|
||||||
|
setCopyJobState((prevState: CopyJobContextState) => ({
|
||||||
|
...prevState,
|
||||||
|
migrationType: checked ? CopyJobMigrationType.Offline : CopyJobMigrationType.Online,
|
||||||
|
}));
|
||||||
|
},
|
||||||
|
[setCopyJobState],
|
||||||
|
);
|
||||||
|
|
||||||
|
return { handleSelectSourceAccount, handleMigrationTypeChange };
|
||||||
|
}
|
||||||
@@ -0,0 +1,52 @@
|
|||||||
|
/* eslint-disable react/display-name */
|
||||||
|
import { Stack } from "@fluentui/react";
|
||||||
|
import React from "react";
|
||||||
|
import { DatabaseAccount, Subscription } from "../../../../../Contracts/DataModels";
|
||||||
|
import { useDatabaseAccounts } from "../../../../../hooks/useDatabaseAccounts";
|
||||||
|
import { useSubscriptions } from "../../../../../hooks/useSubscriptions";
|
||||||
|
import ContainerCopyMessages from "../../../ContainerCopyMessages";
|
||||||
|
import { useCopyJobContext } from "../../../Context/CopyJobContext";
|
||||||
|
import { CopyJobMigrationType } from "../../../Enums";
|
||||||
|
import { AccountDropdown } from "./Components/AccountDropdown";
|
||||||
|
import { MigrationTypeCheckbox } from "./Components/MigrationTypeCheckbox";
|
||||||
|
import { SubscriptionDropdown } from "./Components/SubscriptionDropdown";
|
||||||
|
import { useDropdownOptions, useEventHandlers } from "./Utils/selectAccountUtils";
|
||||||
|
|
||||||
|
const SelectAccount = React.memo(() => {
|
||||||
|
const { copyJobState, setCopyJobState } = useCopyJobContext();
|
||||||
|
const selectedSubscriptionId = copyJobState?.source?.subscription?.subscriptionId;
|
||||||
|
|
||||||
|
const subscriptions: Subscription[] = useSubscriptions();
|
||||||
|
const allAccounts: DatabaseAccount[] = useDatabaseAccounts(selectedSubscriptionId);
|
||||||
|
const sqlApiOnlyAccounts: DatabaseAccount[] = allAccounts?.filter(
|
||||||
|
(account) => account.type === "SQL" || account.kind === "GlobalDocumentDB",
|
||||||
|
);
|
||||||
|
|
||||||
|
const { subscriptionOptions, accountOptions } = useDropdownOptions(subscriptions, sqlApiOnlyAccounts);
|
||||||
|
const { handleSelectSourceAccount, handleMigrationTypeChange } = useEventHandlers(setCopyJobState);
|
||||||
|
|
||||||
|
const migrationTypeChecked = copyJobState?.migrationType === CopyJobMigrationType.Offline;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Stack className="selectAccountContainer" tokens={{ childrenGap: 15 }}>
|
||||||
|
<span>{ContainerCopyMessages.selectAccountDescription}</span>
|
||||||
|
|
||||||
|
<SubscriptionDropdown
|
||||||
|
options={subscriptionOptions}
|
||||||
|
selectedKey={selectedSubscriptionId}
|
||||||
|
onChange={(_ev, option) => handleSelectSourceAccount("subscription", option?.data)}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<AccountDropdown
|
||||||
|
options={accountOptions}
|
||||||
|
selectedKey={copyJobState?.source?.account?.id}
|
||||||
|
disabled={!selectedSubscriptionId}
|
||||||
|
onChange={(_ev, option) => handleSelectSourceAccount("account", option?.data)}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<MigrationTypeCheckbox checked={migrationTypeChecked} onChange={handleMigrationTypeChange} />
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
export default SelectAccount;
|
||||||
@@ -0,0 +1,35 @@
|
|||||||
|
import React from "react";
|
||||||
|
import { CopyJobContextState, DropdownOptionType } from "../../../../Types";
|
||||||
|
|
||||||
|
export function dropDownChangeHandler(setCopyJobState: React.Dispatch<React.SetStateAction<CopyJobContextState>>) {
|
||||||
|
return (type: "sourceDatabase" | "sourceContainer" | "targetDatabase" | "targetContainer") =>
|
||||||
|
(_evnt: React.FormEvent, option: DropdownOptionType) => {
|
||||||
|
const value = option.key;
|
||||||
|
setCopyJobState((prevState) => {
|
||||||
|
switch (type) {
|
||||||
|
case "sourceDatabase":
|
||||||
|
return {
|
||||||
|
...prevState,
|
||||||
|
source: { ...prevState.source, databaseId: value, containerId: undefined },
|
||||||
|
};
|
||||||
|
case "sourceContainer":
|
||||||
|
return {
|
||||||
|
...prevState,
|
||||||
|
source: { ...prevState.source, containerId: value },
|
||||||
|
};
|
||||||
|
case "targetDatabase":
|
||||||
|
return {
|
||||||
|
...prevState,
|
||||||
|
target: { ...prevState.target, databaseId: value, containerId: undefined },
|
||||||
|
};
|
||||||
|
case "targetContainer":
|
||||||
|
return {
|
||||||
|
...prevState,
|
||||||
|
target: { ...prevState.target, containerId: value },
|
||||||
|
};
|
||||||
|
default:
|
||||||
|
return prevState;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,43 @@
|
|||||||
|
import { Dropdown, Stack } from "@fluentui/react";
|
||||||
|
import React from "react";
|
||||||
|
import ContainerCopyMessages from "../../../../ContainerCopyMessages";
|
||||||
|
import { DatabaseContainerSectionProps } from "../../../../Types";
|
||||||
|
import FieldRow from "../../Components/FieldRow";
|
||||||
|
|
||||||
|
export const DatabaseContainerSection = ({
|
||||||
|
heading,
|
||||||
|
databaseOptions,
|
||||||
|
selectedDatabase,
|
||||||
|
databaseDisabled,
|
||||||
|
databaseOnChange,
|
||||||
|
containerOptions,
|
||||||
|
selectedContainer,
|
||||||
|
containerDisabled,
|
||||||
|
containerOnChange,
|
||||||
|
}: DatabaseContainerSectionProps) => (
|
||||||
|
<Stack tokens={{ childrenGap: 15 }} className="databaseContainerSection">
|
||||||
|
<label className="subHeading">{heading}</label>
|
||||||
|
<FieldRow label={ContainerCopyMessages.databaseDropdownLabel}>
|
||||||
|
<Dropdown
|
||||||
|
placeholder={ContainerCopyMessages.databaseDropdownPlaceholder}
|
||||||
|
ariaLabel={ContainerCopyMessages.databaseDropdownLabel}
|
||||||
|
options={databaseOptions}
|
||||||
|
required
|
||||||
|
disabled={!!databaseDisabled}
|
||||||
|
selectedKey={selectedDatabase}
|
||||||
|
onChange={databaseOnChange}
|
||||||
|
/>
|
||||||
|
</FieldRow>
|
||||||
|
<FieldRow label={ContainerCopyMessages.containerDropdownLabel}>
|
||||||
|
<Dropdown
|
||||||
|
placeholder={ContainerCopyMessages.containerDropdownPlaceholder}
|
||||||
|
ariaLabel={ContainerCopyMessages.containerDropdownLabel}
|
||||||
|
options={containerOptions}
|
||||||
|
required
|
||||||
|
disabled={!!containerDisabled}
|
||||||
|
selectedKey={selectedContainer}
|
||||||
|
onChange={containerOnChange}
|
||||||
|
/>
|
||||||
|
</FieldRow>
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
@@ -0,0 +1,72 @@
|
|||||||
|
import { Stack } from "@fluentui/react";
|
||||||
|
import { DatabaseModel } from "Contracts/DataModels";
|
||||||
|
import React from "react";
|
||||||
|
import { useDatabases } from "../../../../../hooks/useDatabases";
|
||||||
|
import { useDataContainers } from "../../../../../hooks/useDataContainers";
|
||||||
|
import ContainerCopyMessages from "../../../ContainerCopyMessages";
|
||||||
|
import { useCopyJobContext } from "../../../Context/CopyJobContext";
|
||||||
|
import { DatabaseContainerSection } from "./components/DatabaseContainerSection";
|
||||||
|
import { dropDownChangeHandler } from "./Events/DropDownChangeHandler";
|
||||||
|
import { useMemoizedSourceAndTargetData } from "./memoizedData";
|
||||||
|
|
||||||
|
const SelectSourceAndTargetContainers = () => {
|
||||||
|
const { copyJobState, setCopyJobState } = useCopyJobContext();
|
||||||
|
const { source, target, sourceDbParams, sourceContainerParams, targetDbParams, targetContainerParams } =
|
||||||
|
useMemoizedSourceAndTargetData(copyJobState);
|
||||||
|
|
||||||
|
// Custom hooks
|
||||||
|
const sourceDatabases = useDatabases(...sourceDbParams) || [];
|
||||||
|
const sourceContainers = useDataContainers(...sourceContainerParams) || [];
|
||||||
|
const targetDatabases = useDatabases(...targetDbParams) || [];
|
||||||
|
const targetContainers = useDataContainers(...targetContainerParams) || [];
|
||||||
|
|
||||||
|
// Memoize option objects for dropdowns
|
||||||
|
const sourceDatabaseOptions = React.useMemo(
|
||||||
|
() => sourceDatabases.map((db: DatabaseModel) => ({ key: db.name, text: db.name, data: db })),
|
||||||
|
[sourceDatabases],
|
||||||
|
);
|
||||||
|
const sourceContainerOptions = React.useMemo(
|
||||||
|
() => sourceContainers.map((c: DatabaseModel) => ({ key: c.name, text: c.name, data: c })),
|
||||||
|
[sourceContainers],
|
||||||
|
);
|
||||||
|
const targetDatabaseOptions = React.useMemo(
|
||||||
|
() => targetDatabases.map((db: DatabaseModel) => ({ key: db.name, text: db.name, data: db })),
|
||||||
|
[targetDatabases],
|
||||||
|
);
|
||||||
|
const targetContainerOptions = React.useMemo(
|
||||||
|
() => targetContainers.map((c: DatabaseModel) => ({ key: c.name, text: c.name, data: c })),
|
||||||
|
[targetContainers],
|
||||||
|
);
|
||||||
|
|
||||||
|
const onDropdownChange = React.useCallback(dropDownChangeHandler(setCopyJobState), [setCopyJobState]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Stack className="selectSourceAndTargetContainers" tokens={{ childrenGap: 25 }}>
|
||||||
|
<span>{ContainerCopyMessages.selectSourceAndTargetContainersDescription}</span>
|
||||||
|
<DatabaseContainerSection
|
||||||
|
heading={ContainerCopyMessages.sourceContainerSubHeading}
|
||||||
|
databaseOptions={sourceDatabaseOptions}
|
||||||
|
selectedDatabase={source?.databaseId}
|
||||||
|
databaseDisabled={false}
|
||||||
|
databaseOnChange={onDropdownChange("sourceDatabase")}
|
||||||
|
containerOptions={sourceContainerOptions}
|
||||||
|
selectedContainer={source?.containerId}
|
||||||
|
containerDisabled={!source?.databaseId}
|
||||||
|
containerOnChange={onDropdownChange("sourceContainer")}
|
||||||
|
/>
|
||||||
|
<DatabaseContainerSection
|
||||||
|
heading={ContainerCopyMessages.targetContainerSubHeading}
|
||||||
|
databaseOptions={targetDatabaseOptions}
|
||||||
|
selectedDatabase={target?.databaseId}
|
||||||
|
databaseDisabled={false}
|
||||||
|
databaseOnChange={onDropdownChange("targetDatabase")}
|
||||||
|
containerOptions={targetContainerOptions}
|
||||||
|
selectedContainer={target?.containerId}
|
||||||
|
containerDisabled={!target?.databaseId}
|
||||||
|
containerOnChange={onDropdownChange("targetContainer")}
|
||||||
|
/>
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default SelectSourceAndTargetContainers;
|
||||||
@@ -0,0 +1,43 @@
|
|||||||
|
import React from "react";
|
||||||
|
import { getAccountDetailsFromResourceId } from "../../../CopyJobUtils";
|
||||||
|
import { CopyJobContextState, DatabaseParams, DataContainerParams } from "../../../Types";
|
||||||
|
|
||||||
|
export function useMemoizedSourceAndTargetData(copyJobState: CopyJobContextState) {
|
||||||
|
const { source, target } = copyJobState ?? {};
|
||||||
|
const selectedSourceAccount = source?.account;
|
||||||
|
const selectedTargetAccount = target?.account;
|
||||||
|
const {
|
||||||
|
subscriptionId: sourceSubscriptionId,
|
||||||
|
resourceGroup: sourceResourceGroup,
|
||||||
|
accountName: sourceAccountName,
|
||||||
|
} = getAccountDetailsFromResourceId(selectedSourceAccount?.id);
|
||||||
|
const {
|
||||||
|
subscriptionId: targetSubscriptionId,
|
||||||
|
resourceGroup: targetResourceGroup,
|
||||||
|
accountName: targetAccountName,
|
||||||
|
} = getAccountDetailsFromResourceId(selectedTargetAccount?.id);
|
||||||
|
|
||||||
|
const sourceDbParams = React.useMemo(
|
||||||
|
() => [sourceSubscriptionId, sourceResourceGroup, sourceAccountName, "SQL"] as DatabaseParams,
|
||||||
|
[sourceSubscriptionId, sourceResourceGroup, sourceAccountName],
|
||||||
|
);
|
||||||
|
|
||||||
|
const sourceContainerParams = React.useMemo(
|
||||||
|
() =>
|
||||||
|
[sourceSubscriptionId, sourceResourceGroup, sourceAccountName, source?.databaseId, "SQL"] as DataContainerParams,
|
||||||
|
[sourceSubscriptionId, sourceResourceGroup, sourceAccountName, source?.databaseId],
|
||||||
|
);
|
||||||
|
|
||||||
|
const targetDbParams = React.useMemo(
|
||||||
|
() => [targetSubscriptionId, targetResourceGroup, targetAccountName, "SQL"] as DatabaseParams,
|
||||||
|
[targetSubscriptionId, targetResourceGroup, targetAccountName],
|
||||||
|
);
|
||||||
|
|
||||||
|
const targetContainerParams = React.useMemo(
|
||||||
|
() =>
|
||||||
|
[targetSubscriptionId, targetResourceGroup, targetAccountName, target?.databaseId, "SQL"] as DataContainerParams,
|
||||||
|
[targetSubscriptionId, targetResourceGroup, targetAccountName, target?.databaseId],
|
||||||
|
);
|
||||||
|
|
||||||
|
return { source, target, sourceDbParams, sourceContainerParams, targetDbParams, targetContainerParams };
|
||||||
|
}
|
||||||
@@ -0,0 +1,89 @@
|
|||||||
|
import { useCallback, useMemo, useReducer } from "react";
|
||||||
|
import { useSidePanel } from "../../../../hooks/useSidePanel";
|
||||||
|
import { submitCreateCopyJob } from "../../Actions/CopyJobActions";
|
||||||
|
import { useCopyJobContext } from "../../Context/CopyJobContext";
|
||||||
|
import { useCopyJobPrerequisitesCache } from "./useCopyJobPrerequisitesCache";
|
||||||
|
import { SCREEN_KEYS, useCreateCopyJobScreensList } from "./useCreateCopyJobScreensList";
|
||||||
|
|
||||||
|
type NavigationState = {
|
||||||
|
screenHistory: string[];
|
||||||
|
};
|
||||||
|
|
||||||
|
type Action = { type: "NEXT"; nextScreen: string } | { type: "PREVIOUS" } | { type: "RESET" };
|
||||||
|
|
||||||
|
function navigationReducer(state: NavigationState, action: Action): NavigationState {
|
||||||
|
switch (action.type) {
|
||||||
|
case "NEXT":
|
||||||
|
return {
|
||||||
|
screenHistory: [...state.screenHistory, action.nextScreen],
|
||||||
|
};
|
||||||
|
case "PREVIOUS":
|
||||||
|
return {
|
||||||
|
screenHistory: state.screenHistory.length > 1 ? state.screenHistory.slice(0, -1) : state.screenHistory,
|
||||||
|
};
|
||||||
|
case "RESET":
|
||||||
|
return {
|
||||||
|
screenHistory: [SCREEN_KEYS.SelectAccount],
|
||||||
|
};
|
||||||
|
default:
|
||||||
|
return state;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useCopyJobNavigation() {
|
||||||
|
const { copyJobState, resetCopyJobState } = useCopyJobContext();
|
||||||
|
const screens = useCreateCopyJobScreensList();
|
||||||
|
const { validationCache: cache } = useCopyJobPrerequisitesCache();
|
||||||
|
const [state, dispatch] = useReducer(navigationReducer, { screenHistory: [SCREEN_KEYS.SelectAccount] });
|
||||||
|
|
||||||
|
const currentScreenKey = state.screenHistory[state.screenHistory.length - 1];
|
||||||
|
const currentScreen = screens.find((screen) => screen.key === currentScreenKey);
|
||||||
|
|
||||||
|
const isPrimaryDisabled = useMemo(() => {
|
||||||
|
const context = currentScreenKey === SCREEN_KEYS.AssignPermissions ? cache : copyJobState;
|
||||||
|
return !currentScreen?.validations.every((v) => v.validate(context));
|
||||||
|
}, [currentScreen.key, copyJobState, cache]);
|
||||||
|
const primaryBtnText = useMemo(() => {
|
||||||
|
if (currentScreenKey === SCREEN_KEYS.PreviewCopyJob) {
|
||||||
|
return "Copy";
|
||||||
|
}
|
||||||
|
return "Next";
|
||||||
|
}, [currentScreenKey]);
|
||||||
|
|
||||||
|
const isPreviousDisabled = state.screenHistory.length <= 1;
|
||||||
|
|
||||||
|
const handleCancel = useCallback(() => {
|
||||||
|
dispatch({ type: "RESET" });
|
||||||
|
resetCopyJobState();
|
||||||
|
useSidePanel.getState().closeSidePanel();
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handlePrimary = useCallback(() => {
|
||||||
|
const transitions = {
|
||||||
|
[SCREEN_KEYS.SelectAccount]: SCREEN_KEYS.AssignPermissions,
|
||||||
|
[SCREEN_KEYS.AssignPermissions]: SCREEN_KEYS.SelectSourceAndTargetContainers,
|
||||||
|
[SCREEN_KEYS.SelectSourceAndTargetContainers]: SCREEN_KEYS.PreviewCopyJob,
|
||||||
|
};
|
||||||
|
|
||||||
|
const nextScreen = transitions[currentScreenKey];
|
||||||
|
if (nextScreen) {
|
||||||
|
dispatch({ type: "NEXT", nextScreen });
|
||||||
|
} else if (currentScreenKey === SCREEN_KEYS.PreviewCopyJob) {
|
||||||
|
submitCreateCopyJob(copyJobState, handleCancel);
|
||||||
|
}
|
||||||
|
}, [currentScreenKey, copyJobState]);
|
||||||
|
|
||||||
|
const handlePrevious = useCallback(() => {
|
||||||
|
dispatch({ type: "PREVIOUS" });
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return {
|
||||||
|
currentScreen,
|
||||||
|
isPrimaryDisabled,
|
||||||
|
isPreviousDisabled,
|
||||||
|
handlePrimary,
|
||||||
|
handlePrevious,
|
||||||
|
handleCancel,
|
||||||
|
primaryBtnText,
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
import create from "zustand";
|
||||||
|
|
||||||
|
interface CopyJobPrerequisitesCacheState {
|
||||||
|
validationCache: Map<string, boolean>;
|
||||||
|
setValidationCache: (cache: Map<string, boolean>) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useCopyJobPrerequisitesCache = create<CopyJobPrerequisitesCacheState>((set) => ({
|
||||||
|
validationCache: new Map<string, boolean>(),
|
||||||
|
setValidationCache: (cache) => set({ validationCache: cache }),
|
||||||
|
}));
|
||||||
@@ -0,0 +1,87 @@
|
|||||||
|
import React from "react";
|
||||||
|
import { CopyJobContextState } from "../../Types";
|
||||||
|
import AssignPermissions from "../Screens/AssignPermissions";
|
||||||
|
import PreviewCopyJob from "../Screens/PreviewCopyJob";
|
||||||
|
import SelectAccount from "../Screens/SelectAccount";
|
||||||
|
import SelectSourceAndTargetContainers from "../Screens/SelectSourceAndTargetContainers";
|
||||||
|
|
||||||
|
const SCREEN_KEYS = {
|
||||||
|
SelectAccount: "SelectAccount",
|
||||||
|
SelectSourceAndTargetContainers: "SelectSourceAndTargetContainers",
|
||||||
|
PreviewCopyJob: "PreviewCopyJob",
|
||||||
|
AssignPermissions: "AssignPermissions",
|
||||||
|
};
|
||||||
|
|
||||||
|
type Validation = {
|
||||||
|
validate: (state: CopyJobContextState | Map<string, boolean>) => boolean;
|
||||||
|
message: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
type Screen = {
|
||||||
|
key: string;
|
||||||
|
component: React.ReactElement;
|
||||||
|
validations: Validation[];
|
||||||
|
};
|
||||||
|
|
||||||
|
function useCreateCopyJobScreensList() {
|
||||||
|
return React.useMemo<Screen[]>(
|
||||||
|
() => [
|
||||||
|
{
|
||||||
|
key: SCREEN_KEYS.SelectAccount,
|
||||||
|
component: <SelectAccount />,
|
||||||
|
validations: [
|
||||||
|
{
|
||||||
|
validate: (state: CopyJobContextState) => !!state?.source?.subscription && !!state?.source?.account,
|
||||||
|
message: "Please select a subscription and account to proceed",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: SCREEN_KEYS.SelectSourceAndTargetContainers,
|
||||||
|
component: <SelectSourceAndTargetContainers />,
|
||||||
|
validations: [
|
||||||
|
{
|
||||||
|
validate: (state: CopyJobContextState) =>
|
||||||
|
!!state?.source?.databaseId &&
|
||||||
|
!!state?.source?.containerId &&
|
||||||
|
!!state?.target?.databaseId &&
|
||||||
|
!!state?.target?.containerId,
|
||||||
|
message: "Please select source and target containers to proceed",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: SCREEN_KEYS.PreviewCopyJob,
|
||||||
|
component: <PreviewCopyJob />,
|
||||||
|
validations: [
|
||||||
|
{
|
||||||
|
validate: (state: CopyJobContextState) =>
|
||||||
|
!!(typeof state?.jobName === "string" && state?.jobName && /^[a-zA-Z0-9-.]+$/.test(state?.jobName)),
|
||||||
|
message: "Please enter a job name to proceed",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: SCREEN_KEYS.AssignPermissions,
|
||||||
|
component: <AssignPermissions />,
|
||||||
|
validations: [
|
||||||
|
{
|
||||||
|
validate: (cache: Map<string, boolean>) => {
|
||||||
|
const cacheValuesIterator = Array.from(cache.values());
|
||||||
|
if (cacheValuesIterator.length === 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const allValid = cacheValuesIterator.every((isValid: boolean) => isValid);
|
||||||
|
return allValid;
|
||||||
|
},
|
||||||
|
message: "Please ensure all previous steps are valid to proceed",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
[],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export { SCREEN_KEYS, useCreateCopyJobScreensList };
|
||||||
40
src/Explorer/ContainerCopy/Enums/index.ts
Normal file
40
src/Explorer/ContainerCopy/Enums/index.ts
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
export enum CopyJobMigrationType {
|
||||||
|
Offline = "offline",
|
||||||
|
Online = "online",
|
||||||
|
}
|
||||||
|
|
||||||
|
// all checks will happen
|
||||||
|
export enum IdentityType {
|
||||||
|
SystemAssigned = "systemassigned", // "SystemAssigned"
|
||||||
|
UserAssigned = "userassigned", // "UserAssigned"
|
||||||
|
None = "none", // "None"
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum DefaultIdentityType {
|
||||||
|
SystemAssignedIdentity = "systemassignedidentity", // "SystemAssignedIdentity"
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum BackupPolicyType {
|
||||||
|
Continuous = "Continuous",
|
||||||
|
Periodic = "Periodic",
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum CopyJobStatusType {
|
||||||
|
Pending = "Pending",
|
||||||
|
InProgress = "InProgress",
|
||||||
|
Running = "Running",
|
||||||
|
Partitioning = "Partitioning",
|
||||||
|
Paused = "Paused",
|
||||||
|
Skipped = "Skipped",
|
||||||
|
Completed = "Completed",
|
||||||
|
Cancelled = "Cancelled",
|
||||||
|
Failed = "Failed",
|
||||||
|
Faulted = "Faulted",
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum CopyJobActions {
|
||||||
|
pause = "pause",
|
||||||
|
resume = "resume",
|
||||||
|
cancel = "cancel",
|
||||||
|
complete = "complete",
|
||||||
|
}
|
||||||
@@ -0,0 +1,81 @@
|
|||||||
|
import { IconButton, IContextualMenuProps } from "@fluentui/react";
|
||||||
|
import React from "react";
|
||||||
|
import ContainerCopyMessages from "../../ContainerCopyMessages";
|
||||||
|
import { CopyJobActions, CopyJobMigrationType, CopyJobStatusType } from "../../Enums";
|
||||||
|
import { CopyJobType } from "../../Types";
|
||||||
|
|
||||||
|
interface CopyJobActionMenuProps {
|
||||||
|
job: CopyJobType;
|
||||||
|
handleClick: (job: CopyJobType, action: string) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const CopyJobActionMenu: React.FC<CopyJobActionMenuProps> = ({ job, handleClick }) => {
|
||||||
|
if ([CopyJobStatusType.Completed, CopyJobStatusType.Cancelled].includes(job.Status)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const getMenuItems = (): IContextualMenuProps["items"] => {
|
||||||
|
const baseItems = [
|
||||||
|
{
|
||||||
|
key: CopyJobActions.pause,
|
||||||
|
text: ContainerCopyMessages.MonitorJobs.Actions.pause,
|
||||||
|
iconProps: { iconName: "Pause" },
|
||||||
|
onClick: () => handleClick(job, CopyJobActions.pause),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: CopyJobActions.cancel,
|
||||||
|
text: ContainerCopyMessages.MonitorJobs.Actions.cancel,
|
||||||
|
iconProps: { iconName: "Cancel" },
|
||||||
|
onClick: () => handleClick(job, CopyJobActions.cancel),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: CopyJobActions.resume,
|
||||||
|
text: ContainerCopyMessages.MonitorJobs.Actions.resume,
|
||||||
|
iconProps: { iconName: "Play" },
|
||||||
|
onClick: () => handleClick(job, CopyJobActions.resume),
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
if (CopyJobStatusType.Paused === job.Status) {
|
||||||
|
return baseItems.filter((item) => item.key !== CopyJobActions.pause);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (CopyJobStatusType.Pending === job.Status) {
|
||||||
|
return baseItems.filter((item) => item.key !== CopyJobActions.resume);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
[CopyJobStatusType.InProgress, CopyJobStatusType.Running, CopyJobStatusType.Partitioning].includes(job.Status)
|
||||||
|
) {
|
||||||
|
const filteredItems = baseItems.filter((item) => item.key !== CopyJobActions.resume);
|
||||||
|
if (job.Mode === CopyJobMigrationType.Online) {
|
||||||
|
filteredItems.push({
|
||||||
|
key: CopyJobActions.complete,
|
||||||
|
text: ContainerCopyMessages.MonitorJobs.Actions.complete,
|
||||||
|
iconProps: { iconName: "CheckMark" },
|
||||||
|
onClick: () => handleClick(job, CopyJobActions.complete),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return filteredItems;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ([CopyJobStatusType.Failed, CopyJobStatusType.Faulted, CopyJobStatusType.Skipped].includes(job.Status)) {
|
||||||
|
return baseItems.filter((item) => item.key === CopyJobActions.resume);
|
||||||
|
}
|
||||||
|
|
||||||
|
return baseItems;
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<IconButton
|
||||||
|
role="button"
|
||||||
|
iconProps={{ iconName: "More", styles: { root: { fontSize: "20px", fontWeight: "bold" } } }}
|
||||||
|
menuProps={{ items: getMenuItems() }}
|
||||||
|
menuIconProps={{ iconName: "" }}
|
||||||
|
ariaLabel={ContainerCopyMessages.MonitorJobs.Columns.actions}
|
||||||
|
title={ContainerCopyMessages.MonitorJobs.Columns.actions}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default CopyJobActionMenu;
|
||||||
@@ -0,0 +1,79 @@
|
|||||||
|
import { IColumn } from "@fluentui/react";
|
||||||
|
import React from "react";
|
||||||
|
import ContainerCopyMessages from "../../ContainerCopyMessages";
|
||||||
|
import { CopyJobType } from "../../Types";
|
||||||
|
import CopyJobActionMenu from "./CopyJobActionMenu";
|
||||||
|
import CopyJobStatusWithIcon from "./CopyJobStatusWithIcon";
|
||||||
|
|
||||||
|
export const getColumns = (
|
||||||
|
handleSort: (columnKey: string) => void,
|
||||||
|
handleActionClick: (job: CopyJobType, action: string) => void,
|
||||||
|
sortedColumnKey: string | undefined,
|
||||||
|
isSortedDescending: boolean,
|
||||||
|
): IColumn[] => [
|
||||||
|
{
|
||||||
|
key: "LastUpdatedTime",
|
||||||
|
name: ContainerCopyMessages.MonitorJobs.Columns.lastUpdatedTime,
|
||||||
|
fieldName: "LastUpdatedTime",
|
||||||
|
minWidth: 140,
|
||||||
|
maxWidth: 300,
|
||||||
|
isResizable: true,
|
||||||
|
isSorted: sortedColumnKey === "timestamp",
|
||||||
|
isSortedDescending: isSortedDescending,
|
||||||
|
onColumnClick: () => handleSort("timestamp"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "Name",
|
||||||
|
name: ContainerCopyMessages.MonitorJobs.Columns.name,
|
||||||
|
fieldName: "Name",
|
||||||
|
minWidth: 140,
|
||||||
|
maxWidth: 300,
|
||||||
|
isResizable: true,
|
||||||
|
isSorted: sortedColumnKey === "Name",
|
||||||
|
isSortedDescending: isSortedDescending,
|
||||||
|
onColumnClick: () => handleSort("Name"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "Mode",
|
||||||
|
name: ContainerCopyMessages.MonitorJobs.Columns.mode,
|
||||||
|
fieldName: "Mode",
|
||||||
|
minWidth: 90,
|
||||||
|
maxWidth: 200,
|
||||||
|
isResizable: true,
|
||||||
|
isSorted: sortedColumnKey === "Mode",
|
||||||
|
isSortedDescending: isSortedDescending,
|
||||||
|
onColumnClick: () => handleSort("Mode"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "CompletionPercentage",
|
||||||
|
name: ContainerCopyMessages.MonitorJobs.Columns.completionPercentage,
|
||||||
|
fieldName: "CompletionPercentage",
|
||||||
|
minWidth: 110,
|
||||||
|
maxWidth: 200,
|
||||||
|
isResizable: true,
|
||||||
|
isSorted: sortedColumnKey === "CompletionPercentage",
|
||||||
|
isSortedDescending: isSortedDescending,
|
||||||
|
onRender: (job: CopyJobType) => `${job.CompletionPercentage}%`,
|
||||||
|
onColumnClick: () => handleSort("CompletionPercentage"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "CopyJobStatus",
|
||||||
|
name: ContainerCopyMessages.MonitorJobs.Columns.status,
|
||||||
|
fieldName: "Status",
|
||||||
|
minWidth: 130,
|
||||||
|
maxWidth: 200,
|
||||||
|
isResizable: true,
|
||||||
|
isSorted: sortedColumnKey === "Status",
|
||||||
|
isSortedDescending: isSortedDescending,
|
||||||
|
onRender: (job: CopyJobType) => <CopyJobStatusWithIcon status={job.Status} />,
|
||||||
|
onColumnClick: () => handleSort("Status"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "Actions",
|
||||||
|
name: "",
|
||||||
|
minWidth: 80,
|
||||||
|
maxWidth: 200,
|
||||||
|
isResizable: true,
|
||||||
|
onRender: (job: CopyJobType) => <CopyJobActionMenu job={job} handleClick={handleActionClick} />,
|
||||||
|
},
|
||||||
|
];
|
||||||
@@ -0,0 +1,54 @@
|
|||||||
|
import { FontIcon, getTheme, mergeStyles, mergeStyleSets, Stack, Text } from "@fluentui/react";
|
||||||
|
import React from "react";
|
||||||
|
import ContainerCopyMessages from "../../ContainerCopyMessages";
|
||||||
|
import { CopyJobStatusType } from "../../Enums";
|
||||||
|
|
||||||
|
const theme = getTheme();
|
||||||
|
|
||||||
|
const iconClass = mergeStyles({
|
||||||
|
fontSize: "16px",
|
||||||
|
marginRight: "8px",
|
||||||
|
});
|
||||||
|
|
||||||
|
const classNames = mergeStyleSets({
|
||||||
|
[CopyJobStatusType.Pending]: [{ color: theme.semanticColors.bodySubtext }, iconClass],
|
||||||
|
[CopyJobStatusType.InProgress]: [{ color: theme.palette.themePrimary }, iconClass],
|
||||||
|
[CopyJobStatusType.Running]: [{ color: theme.palette.themePrimary }, iconClass],
|
||||||
|
[CopyJobStatusType.Partitioning]: [{ color: theme.palette.themePrimary }, iconClass],
|
||||||
|
[CopyJobStatusType.Paused]: [{ color: theme.palette.themePrimary }, iconClass],
|
||||||
|
[CopyJobStatusType.Skipped]: [{ color: theme.semanticColors.bodySubtext }, iconClass],
|
||||||
|
[CopyJobStatusType.Cancelled]: [{ color: theme.semanticColors.bodySubtext }, iconClass],
|
||||||
|
[CopyJobStatusType.Failed]: [{ color: theme.semanticColors.errorIcon }, iconClass],
|
||||||
|
[CopyJobStatusType.Faulted]: [{ color: theme.semanticColors.errorIcon }, iconClass],
|
||||||
|
[CopyJobStatusType.Completed]: [{ color: theme.semanticColors.successIcon }, iconClass],
|
||||||
|
unknown: [{ color: theme.semanticColors.bodySubtext }, iconClass],
|
||||||
|
});
|
||||||
|
|
||||||
|
const iconMap: Record<CopyJobStatusType, string> = {
|
||||||
|
[CopyJobStatusType.Pending]: "StatusCircleRing",
|
||||||
|
[CopyJobStatusType.InProgress]: "ProgressRingDots",
|
||||||
|
[CopyJobStatusType.Running]: "ProgressRingDots",
|
||||||
|
[CopyJobStatusType.Partitioning]: "ProgressRingDots",
|
||||||
|
[CopyJobStatusType.Paused]: "CirclePause",
|
||||||
|
[CopyJobStatusType.Skipped]: "StatusCircleBlock2",
|
||||||
|
[CopyJobStatusType.Cancelled]: "StatusErrorFull",
|
||||||
|
[CopyJobStatusType.Failed]: "StatusErrorFull",
|
||||||
|
[CopyJobStatusType.Faulted]: "StatusErrorFull",
|
||||||
|
[CopyJobStatusType.Completed]: "CompletedSolid",
|
||||||
|
};
|
||||||
|
|
||||||
|
const CopyJobStatusWithIcon: React.FC<{ status: CopyJobStatusType }> = ({ status }) => {
|
||||||
|
const statusText = ContainerCopyMessages.MonitorJobs.Status[status] || "Unknown";
|
||||||
|
return (
|
||||||
|
<Stack horizontal verticalAlign="center">
|
||||||
|
<FontIcon
|
||||||
|
aria-label={status}
|
||||||
|
iconName={iconMap[status] || "UnknownSolid"}
|
||||||
|
className={classNames[status] || classNames.unknown}
|
||||||
|
/>
|
||||||
|
<Text>{statusText}</Text>
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default CopyJobStatusWithIcon;
|
||||||
@@ -0,0 +1,22 @@
|
|||||||
|
import { ActionButton, Image } from "@fluentui/react";
|
||||||
|
import React, { useCallback } from "react";
|
||||||
|
import CopyJobIcon from "../../../../../images/ContainerCopy/copy-jobs.svg";
|
||||||
|
import * as Actions from "../../Actions/CopyJobActions";
|
||||||
|
import ContainerCopyMessages from "../../ContainerCopyMessages";
|
||||||
|
|
||||||
|
interface CopyJobsNotFoundProps {}
|
||||||
|
|
||||||
|
const CopyJobsNotFound: React.FC<CopyJobsNotFoundProps> = () => {
|
||||||
|
const handleCreateCopyJob = useCallback(Actions.openCreateCopyJobPanel, []);
|
||||||
|
return (
|
||||||
|
<div className="notFoundContainer flexContainer centerContent">
|
||||||
|
<Image src={CopyJobIcon} alt={ContainerCopyMessages.noCopyJobsTitle} width={100} height={100} />
|
||||||
|
<h4 className="noCopyJobsMessage">{ContainerCopyMessages.noCopyJobsTitle}</h4>
|
||||||
|
<ActionButton allowDisabledFocus className="createCopyJobButton" onClick={handleCreateCopyJob}>
|
||||||
|
{ContainerCopyMessages.createCopyJobButtonText}
|
||||||
|
</ActionButton>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default CopyJobsNotFound;
|
||||||
@@ -0,0 +1,103 @@
|
|||||||
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
||||||
|
import {
|
||||||
|
ConstrainMode,
|
||||||
|
DetailsListLayoutMode,
|
||||||
|
DetailsRow,
|
||||||
|
IColumn,
|
||||||
|
ScrollablePane,
|
||||||
|
ScrollbarVisibility,
|
||||||
|
ShimmeredDetailsList,
|
||||||
|
Stack,
|
||||||
|
Sticky,
|
||||||
|
StickyPositionType,
|
||||||
|
} from "@fluentui/react";
|
||||||
|
import React, { useEffect } from "react";
|
||||||
|
import { CopyJobType } from "../../Types";
|
||||||
|
import { getColumns } from "./CopyJobColumns";
|
||||||
|
|
||||||
|
interface CopyJobsListProps {
|
||||||
|
jobs: CopyJobType[];
|
||||||
|
handleActionClick: (job: CopyJobType, action: string) => void;
|
||||||
|
pageSize?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
const styles = {
|
||||||
|
container: { height: "calc(100vh - 15em)" } as React.CSSProperties,
|
||||||
|
stackItem: { position: "relative", marginBottom: "20px" } as React.CSSProperties,
|
||||||
|
};
|
||||||
|
|
||||||
|
const PAGE_SIZE = 100; // Number of items per page
|
||||||
|
|
||||||
|
const CopyJobsList: React.FC<CopyJobsListProps> = ({ jobs, handleActionClick, pageSize = PAGE_SIZE }) => {
|
||||||
|
const [startIndex] = React.useState(0);
|
||||||
|
const [sortedJobs, setSortedJobs] = React.useState<CopyJobType[]>(jobs);
|
||||||
|
const [sortedColumnKey, setSortedColumnKey] = React.useState<string | undefined>(undefined);
|
||||||
|
const [isSortedDescending, setIsSortedDescending] = React.useState<boolean>(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
setSortedJobs(jobs);
|
||||||
|
}, [jobs]);
|
||||||
|
|
||||||
|
const handleSort = (columnKey: string) => {
|
||||||
|
const isDescending = sortedColumnKey === columnKey ? !isSortedDescending : false;
|
||||||
|
const sorted = [...sortedJobs].sort((current: any, next: any) => {
|
||||||
|
if (current[columnKey] < next[columnKey]) {
|
||||||
|
return isDescending ? 1 : -1;
|
||||||
|
}
|
||||||
|
if (current[columnKey] > next[columnKey]) {
|
||||||
|
return isDescending ? -1 : 1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
});
|
||||||
|
setSortedJobs(sorted);
|
||||||
|
setSortedColumnKey(columnKey);
|
||||||
|
setIsSortedDescending(isDescending);
|
||||||
|
};
|
||||||
|
|
||||||
|
const columns: IColumn[] = React.useMemo(
|
||||||
|
() => getColumns(handleSort, handleActionClick, sortedColumnKey, isSortedDescending),
|
||||||
|
[handleSort, handleActionClick, sortedColumnKey, isSortedDescending],
|
||||||
|
);
|
||||||
|
|
||||||
|
const _handleRowClick = React.useCallback((job: CopyJobType) => {
|
||||||
|
// eslint-disable-next-line no-console
|
||||||
|
console.log("Row clicked:", job);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const _onRenderRow = React.useCallback((props: any) => {
|
||||||
|
return (
|
||||||
|
<div onClick={_handleRowClick.bind(null, props.item)}>
|
||||||
|
<DetailsRow {...props} styles={{ root: { cursor: "pointer" } }} />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
// const totalCount = jobs.length;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div style={styles.container}>
|
||||||
|
<Stack verticalFill={true}>
|
||||||
|
<Stack.Item verticalFill={true} grow={1} shrink={1} style={styles.stackItem}>
|
||||||
|
<ScrollablePane scrollbarVisibility={ScrollbarVisibility.auto}>
|
||||||
|
<ShimmeredDetailsList
|
||||||
|
onRenderRow={_onRenderRow}
|
||||||
|
checkboxVisibility={2}
|
||||||
|
columns={columns}
|
||||||
|
items={sortedJobs.slice(startIndex, startIndex + pageSize)}
|
||||||
|
enableShimmer={false}
|
||||||
|
constrainMode={ConstrainMode.unconstrained}
|
||||||
|
layoutMode={DetailsListLayoutMode.justified}
|
||||||
|
onRenderDetailsHeader={(props, defaultRender) => (
|
||||||
|
<Sticky stickyPosition={StickyPositionType.Header} isScrollSynced>
|
||||||
|
{defaultRender({ ...props })}
|
||||||
|
</Sticky>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</ScrollablePane>
|
||||||
|
</Stack.Item>
|
||||||
|
</Stack>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default CopyJobsList;
|
||||||
@@ -0,0 +1,12 @@
|
|||||||
|
import create from "zustand";
|
||||||
|
import { MonitorCopyJobsRef } from "./MonitorCopyJobs";
|
||||||
|
|
||||||
|
type MonitorCopyJobsRefStateType = {
|
||||||
|
ref: MonitorCopyJobsRef;
|
||||||
|
setRef: (ref: MonitorCopyJobsRef) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const MonitorCopyJobsRefState = create<MonitorCopyJobsRefStateType>((set) => ({
|
||||||
|
ref: null,
|
||||||
|
setRef: (ref) => set({ ref: ref }),
|
||||||
|
}));
|
||||||
118
src/Explorer/ContainerCopy/MonitorCopyJobs/MonitorCopyJobs.tsx
Normal file
118
src/Explorer/ContainerCopy/MonitorCopyJobs/MonitorCopyJobs.tsx
Normal file
@@ -0,0 +1,118 @@
|
|||||||
|
/* eslint-disable react/display-name */
|
||||||
|
import { MessageBar, MessageBarType, Stack } from "@fluentui/react";
|
||||||
|
import ShimmerTree, { IndentLevel } from "Common/ShimmerTree";
|
||||||
|
import React, { forwardRef, useEffect, useImperativeHandle } from "react";
|
||||||
|
import { getCopyJobs, updateCopyJobStatus } from "../Actions/CopyJobActions";
|
||||||
|
import { convertToCamelCase } from "../CopyJobUtils";
|
||||||
|
import { CopyJobStatusType } from "../Enums";
|
||||||
|
import CopyJobsNotFound from "../MonitorCopyJobs/Components/CopyJobs.NotFound";
|
||||||
|
import { CopyJobType } from "../Types";
|
||||||
|
import CopyJobsList from "./Components/CopyJobsList";
|
||||||
|
|
||||||
|
const FETCH_INTERVAL_MS = 30 * 1000; // Interval time in milliseconds (30 seconds)
|
||||||
|
|
||||||
|
interface MonitorCopyJobsProps {}
|
||||||
|
|
||||||
|
export interface MonitorCopyJobsRef {
|
||||||
|
refreshJobList: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const MonitorCopyJobs = forwardRef<MonitorCopyJobsRef, MonitorCopyJobsProps>((_props, ref) => {
|
||||||
|
const [loading, setLoading] = React.useState(true); // Start with loading as true
|
||||||
|
const [error, setError] = React.useState<string | null>(null);
|
||||||
|
const [jobs, setJobs] = React.useState<CopyJobType[]>([]);
|
||||||
|
const isUpdatingRef = React.useRef(false); // Use ref to track updating state
|
||||||
|
const isFirstFetchRef = React.useRef(true); // Use ref to track the first fetch
|
||||||
|
|
||||||
|
const indentLevels = React.useMemo<IndentLevel[]>(() => Array(7).fill({ level: 0, width: "100%" }), []);
|
||||||
|
|
||||||
|
const fetchJobs = React.useCallback(async () => {
|
||||||
|
if (isUpdatingRef.current) {
|
||||||
|
return;
|
||||||
|
} // Skip if an update is in progress
|
||||||
|
try {
|
||||||
|
if (isFirstFetchRef.current) {
|
||||||
|
setLoading(true);
|
||||||
|
} // Show loading spinner only for the first fetch
|
||||||
|
setError(null);
|
||||||
|
|
||||||
|
const response = await getCopyJobs();
|
||||||
|
setJobs((prevJobs) => {
|
||||||
|
// Only update jobs if they are different
|
||||||
|
const isSame = JSON.stringify(prevJobs) === JSON.stringify(response);
|
||||||
|
return isSame ? prevJobs : response;
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
setError(error.message || "Failed to load copy jobs. Please try again later.");
|
||||||
|
} finally {
|
||||||
|
if (isFirstFetchRef.current) {
|
||||||
|
setLoading(false); // Hide loading spinner after the first fetch
|
||||||
|
isFirstFetchRef.current = false; // Mark the first fetch as complete
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
fetchJobs();
|
||||||
|
const intervalId = setInterval(fetchJobs, FETCH_INTERVAL_MS);
|
||||||
|
|
||||||
|
return () => clearInterval(intervalId);
|
||||||
|
}, [fetchJobs]);
|
||||||
|
|
||||||
|
useImperativeHandle(ref, () => ({
|
||||||
|
refreshJobList: () => {
|
||||||
|
if (isUpdatingRef.current) {
|
||||||
|
setError("Please wait for the current update to complete before refreshing.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
fetchJobs();
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
const handleActionClick = React.useCallback(async (job: CopyJobType, action: string) => {
|
||||||
|
try {
|
||||||
|
isUpdatingRef.current = true; // Mark as updating
|
||||||
|
const updatedCopyJob = await updateCopyJobStatus(job, action);
|
||||||
|
if (updatedCopyJob) {
|
||||||
|
setJobs((prevJobs) =>
|
||||||
|
prevJobs.map((prevJob) =>
|
||||||
|
prevJob.Name === updatedCopyJob.properties.jobName
|
||||||
|
? {
|
||||||
|
...prevJob,
|
||||||
|
Status: convertToCamelCase(updatedCopyJob.properties.status) as CopyJobStatusType,
|
||||||
|
}
|
||||||
|
: prevJob,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
setError(error.message || "Failed to update copy job status. Please try again later.");
|
||||||
|
} finally {
|
||||||
|
isUpdatingRef.current = false; // Mark as not updating
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const memoizedJobsList = React.useMemo(() => {
|
||||||
|
if (loading) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
if (jobs.length > 0) {
|
||||||
|
return <CopyJobsList jobs={jobs} handleActionClick={handleActionClick} />;
|
||||||
|
}
|
||||||
|
return <CopyJobsNotFound />;
|
||||||
|
}, [jobs, loading, handleActionClick]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Stack className="monitorCopyJobs flexContainer">
|
||||||
|
{loading && <ShimmerTree indentLevels={indentLevels} style={{ width: "100%", padding: "1rem 2.5rem" }} />}
|
||||||
|
{error && (
|
||||||
|
<MessageBar messageBarType={MessageBarType.error} isMultiline={false} onDismiss={() => setError(null)}>
|
||||||
|
{error}
|
||||||
|
</MessageBar>
|
||||||
|
)}
|
||||||
|
{memoizedJobsList}
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
export default MonitorCopyJobs;
|
||||||
132
src/Explorer/ContainerCopy/Types/index.ts
Normal file
132
src/Explorer/ContainerCopy/Types/index.ts
Normal file
@@ -0,0 +1,132 @@
|
|||||||
|
import { DatabaseAccount, Subscription } from "Contracts/DataModels";
|
||||||
|
import React from "react";
|
||||||
|
import { ApiType } from "UserContext";
|
||||||
|
import Explorer from "../../Explorer";
|
||||||
|
import { CopyJobMigrationType, CopyJobStatusType } from "../Enums";
|
||||||
|
|
||||||
|
export interface ContainerCopyProps {
|
||||||
|
container: Explorer;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type CopyJobCommandBarBtnType = {
|
||||||
|
key: string;
|
||||||
|
iconSrc: string;
|
||||||
|
label: string;
|
||||||
|
ariaLabel: string;
|
||||||
|
disabled?: boolean;
|
||||||
|
onClick: () => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type CopyJobTabForwardRefHandle = {
|
||||||
|
validate: (state: CopyJobContextState) => boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type DropdownOptionType = {
|
||||||
|
key: string;
|
||||||
|
text: string;
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
data: any;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type DatabaseParams = [string | undefined, string | undefined, string | undefined, ApiType];
|
||||||
|
export type DataContainerParams = [
|
||||||
|
string | undefined,
|
||||||
|
string | undefined,
|
||||||
|
string | undefined,
|
||||||
|
string | undefined,
|
||||||
|
ApiType,
|
||||||
|
];
|
||||||
|
|
||||||
|
export interface DatabaseContainerSectionProps {
|
||||||
|
heading: string;
|
||||||
|
databaseOptions: DropdownOptionType[];
|
||||||
|
selectedDatabase: string;
|
||||||
|
databaseDisabled?: boolean;
|
||||||
|
databaseOnChange: (ev: React.FormEvent<HTMLDivElement>, option: DropdownOptionType) => void;
|
||||||
|
containerOptions: DropdownOptionType[];
|
||||||
|
selectedContainer: string;
|
||||||
|
containerDisabled?: boolean;
|
||||||
|
containerOnChange: (ev: React.FormEvent<HTMLDivElement>, option: DropdownOptionType) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface CopyJobContextState {
|
||||||
|
jobName: string;
|
||||||
|
migrationType: CopyJobMigrationType;
|
||||||
|
sourceReadAccessFromTarget?: boolean;
|
||||||
|
// source details
|
||||||
|
source: {
|
||||||
|
subscription: Subscription;
|
||||||
|
account: DatabaseAccount;
|
||||||
|
databaseId: string;
|
||||||
|
containerId: string;
|
||||||
|
};
|
||||||
|
// target details
|
||||||
|
target: {
|
||||||
|
subscriptionId: string;
|
||||||
|
account: DatabaseAccount;
|
||||||
|
databaseId: string;
|
||||||
|
containerId: string;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface CopyJobFlowType {
|
||||||
|
currentScreen: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface CopyJobContextProviderType {
|
||||||
|
flow: CopyJobFlowType;
|
||||||
|
setFlow: React.Dispatch<React.SetStateAction<CopyJobFlowType>>;
|
||||||
|
copyJobState: CopyJobContextState | null;
|
||||||
|
setCopyJobState: React.Dispatch<React.SetStateAction<CopyJobContextState>>;
|
||||||
|
resetCopyJobState: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type CopyJobType = {
|
||||||
|
ID: string;
|
||||||
|
Mode: string;
|
||||||
|
Name: string;
|
||||||
|
Status: CopyJobStatusType;
|
||||||
|
CompletionPercentage: number;
|
||||||
|
Duration: string;
|
||||||
|
LastUpdatedTime: string;
|
||||||
|
timestamp: number;
|
||||||
|
Error?: CopyJobErrorType;
|
||||||
|
};
|
||||||
|
|
||||||
|
export interface CopyJobErrorType {
|
||||||
|
message: string;
|
||||||
|
code: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface CopyJobError {
|
||||||
|
message: string;
|
||||||
|
navigateToStep?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type DataTransferJobType = {
|
||||||
|
id: string;
|
||||||
|
type: string;
|
||||||
|
properties: {
|
||||||
|
jobName: string;
|
||||||
|
status: string;
|
||||||
|
lastUpdatedUtcTime: string;
|
||||||
|
processedCount: number;
|
||||||
|
totalCount: number;
|
||||||
|
mode: string;
|
||||||
|
duration: string;
|
||||||
|
source: {
|
||||||
|
databaseName: string;
|
||||||
|
collectionName: string;
|
||||||
|
component: string;
|
||||||
|
};
|
||||||
|
destination: {
|
||||||
|
databaseName: string;
|
||||||
|
collectionName: string;
|
||||||
|
component: string;
|
||||||
|
};
|
||||||
|
error: {
|
||||||
|
message: string;
|
||||||
|
code: string;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
128
src/Explorer/ContainerCopy/containerCopyStyles.less
Normal file
128
src/Explorer/ContainerCopy/containerCopyStyles.less
Normal file
@@ -0,0 +1,128 @@
|
|||||||
|
@import "../../../less/Common/Constants.less";
|
||||||
|
|
||||||
|
#containerCopyWrapper {
|
||||||
|
.centerContent {
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
.notFoundContainer {
|
||||||
|
.noCopyJobsMessage {
|
||||||
|
font-weight: 600;
|
||||||
|
margin: 0 auto;
|
||||||
|
color: @FocusColor;
|
||||||
|
}
|
||||||
|
button.createCopyJobButton {
|
||||||
|
color: @LinkColor;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.createCopyJobScreensContainer {
|
||||||
|
height: 100%;
|
||||||
|
padding: 1em 1.5em;
|
||||||
|
.bold {
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
label {
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
.flex-row {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
label.field-label {
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
.flex-fixed-width {
|
||||||
|
flex: 0 0 auto;
|
||||||
|
width: 150px;
|
||||||
|
}
|
||||||
|
.flex-grow-col {
|
||||||
|
flex: 1 1 auto;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.databaseContainerSection {
|
||||||
|
label.subHeading {
|
||||||
|
font: inherit;
|
||||||
|
padding: unset;
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.accordionHeader {
|
||||||
|
button {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
.accordionHeaderText {
|
||||||
|
margin-left: 5px;
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
.statusIcon {
|
||||||
|
margin-left: auto;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.popover-container {
|
||||||
|
button[disabled] {
|
||||||
|
cursor: not-allowed;
|
||||||
|
opacity: 0.8;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.foreground {
|
||||||
|
z-index: 10;
|
||||||
|
background-color: white;
|
||||||
|
padding: 20px;
|
||||||
|
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.2);
|
||||||
|
transform: translate(0%, -9%);
|
||||||
|
position: absolute;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.monitorCopyJobs {
|
||||||
|
padding: 0;
|
||||||
|
width: 100%;
|
||||||
|
max-width: 100%;
|
||||||
|
margin: 0 auto;
|
||||||
|
|
||||||
|
.ms-DetailsList {
|
||||||
|
width: 100%;
|
||||||
|
|
||||||
|
.ms-DetailsHeader {
|
||||||
|
.ms-DetailsHeader-cell {
|
||||||
|
padding: @DefaultSpace 20px;
|
||||||
|
font-weight: 600;
|
||||||
|
font-size: @DefaultFontSize;
|
||||||
|
color: @BaseHigh;
|
||||||
|
background-color: @BaseLow;
|
||||||
|
border-bottom: @ButtonBorderWidth solid @BaseMedium;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background-color: @BaseMediumLow;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.ms-DetailsRow {
|
||||||
|
border-bottom: @ButtonBorderWidth solid @BaseMedium;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background-color: @BaseMediumLow;
|
||||||
|
}
|
||||||
|
|
||||||
|
.ms-DetailsRow-cell {
|
||||||
|
padding: @MediumSpace 20px;
|
||||||
|
font-size: @DefaultFontSize;
|
||||||
|
color: @BaseHigh;
|
||||||
|
min-height: 48px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
button[role="button"] {
|
||||||
|
&.ms-Button--icon {
|
||||||
|
i.ms-Icon {
|
||||||
|
font-size: @LargeSpace;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
23
src/Explorer/ContainerCopy/index.tsx
Normal file
23
src/Explorer/ContainerCopy/index.tsx
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
import { MonitorCopyJobsRefState } from "Explorer/ContainerCopy/MonitorCopyJobs/MonitorCopyJobRefState";
|
||||||
|
import React, { useEffect } from "react";
|
||||||
|
import CopyJobCommandBar from "./CommandBar/CopyJobCommandBar";
|
||||||
|
import "./containerCopyStyles.less";
|
||||||
|
import MonitorCopyJobs, { MonitorCopyJobsRef } from "./MonitorCopyJobs/MonitorCopyJobs";
|
||||||
|
import { ContainerCopyProps } from "./Types";
|
||||||
|
|
||||||
|
const ContainerCopyPanel: React.FC<ContainerCopyProps> = ({ container }) => {
|
||||||
|
const monitorCopyJobsRef = React.useRef<MonitorCopyJobsRef>();
|
||||||
|
useEffect(() => {
|
||||||
|
if (monitorCopyJobsRef.current) {
|
||||||
|
MonitorCopyJobsRefState.getState().setRef(monitorCopyJobsRef.current);
|
||||||
|
}
|
||||||
|
}, [monitorCopyJobsRef.current]);
|
||||||
|
return (
|
||||||
|
<div id="containerCopyWrapper" className="flexContainer hideOverflows">
|
||||||
|
<CopyJobCommandBar container={container} />
|
||||||
|
<MonitorCopyJobs ref={monitorCopyJobsRef} />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default ContainerCopyPanel;
|
||||||
@@ -1,4 +1,6 @@
|
|||||||
import { IPivotItemProps, IPivotProps, Pivot, PivotItem } from "@fluentui/react";
|
import { IPivotItemProps, IPivotProps, Pivot, PivotItem } from "@fluentui/react";
|
||||||
|
import { sendMessage } from "Common/MessageHandler";
|
||||||
|
import { FabricMessageTypes } from "Contracts/FabricMessageTypes";
|
||||||
import {
|
import {
|
||||||
ComputedPropertiesComponent,
|
ComputedPropertiesComponent,
|
||||||
ComputedPropertiesComponentProps,
|
ComputedPropertiesComponentProps,
|
||||||
@@ -431,6 +433,15 @@ export class SettingsComponent extends React.Component<SettingsComponentProps, S
|
|||||||
);
|
);
|
||||||
} finally {
|
} finally {
|
||||||
this.props.settingsTab.isExecuting(false);
|
this.props.settingsTab.isExecuting(false);
|
||||||
|
|
||||||
|
// Send message to Fabric no matter success or failure.
|
||||||
|
// In case of failure, saveCollectionSettings might have partially succeeded and Fabric needs to refresh
|
||||||
|
if (isFabricNative() && this.isCollectionSettingsTab) {
|
||||||
|
sendMessage({
|
||||||
|
type: FabricMessageTypes.ContainerUpdated,
|
||||||
|
params: { updateType: "settings" },
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -559,26 +559,81 @@ export class ThroughputInputAutoPilotV3Component extends React.Component<
|
|||||||
private getThroughputTextField = (): JSX.Element => (
|
private getThroughputTextField = (): JSX.Element => (
|
||||||
<>
|
<>
|
||||||
{this.props.isAutoPilotSelected ? (
|
{this.props.isAutoPilotSelected ? (
|
||||||
<TextField
|
<Stack horizontal verticalAlign="end" tokens={{ childrenGap: 8 }}>
|
||||||
label="Maximum RU/s required by this resource"
|
{/* Column 1: Minimum RU/s */}
|
||||||
required
|
<Stack tokens={{ childrenGap: 4 }}>
|
||||||
type="number"
|
<Stack horizontal verticalAlign="center" tokens={{ childrenGap: 4 }}>
|
||||||
id="autopilotInput"
|
<Text variant="small" style={{ lineHeight: "20px", fontWeight: 600 }}>
|
||||||
key="auto pilot throughput input"
|
Minimum RU/s
|
||||||
styles={getTextFieldStyles(this.props.maxAutoPilotThroughput, this.props.maxAutoPilotThroughputBaseline)}
|
</Text>
|
||||||
disabled={this.overrideWithProvisionedThroughputSettings()}
|
<FontIcon iconName="Info" style={{ fontSize: 12, color: "#666" }} />
|
||||||
step={AutoPilotUtils.autoPilotIncrementStep}
|
</Stack>
|
||||||
value={this.overrideWithProvisionedThroughputSettings() ? "" : this.props.maxAutoPilotThroughput?.toString()}
|
<Text
|
||||||
onChange={this.onAutoPilotThroughputChange}
|
style={{
|
||||||
min={autoPilotThroughput1K}
|
fontFamily: "Segoe UI",
|
||||||
onGetErrorMessage={(value: string) => {
|
width: 70,
|
||||||
const sanitizedValue = getSanitizedInputValue(value);
|
height: 28,
|
||||||
return sanitizedValue % 1000
|
border: "none",
|
||||||
? "Throughput value must be in increments of 1000"
|
fontSize: 14,
|
||||||
: this.props.throughputError;
|
backgroundColor: "transparent",
|
||||||
}}
|
fontWeight: 400,
|
||||||
validateOnLoad={false}
|
display: "flex",
|
||||||
/>
|
alignItems: "center",
|
||||||
|
justifyContent: "center",
|
||||||
|
boxSizing: "border-box",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{AutoPilotUtils.getMinRUsBasedOnUserInput(this.props.maxAutoPilotThroughput)}
|
||||||
|
</Text>
|
||||||
|
</Stack>
|
||||||
|
|
||||||
|
{/* Column 2: "x 10 =" Text */}
|
||||||
|
<Text
|
||||||
|
style={{
|
||||||
|
fontFamily: "Segoe UI",
|
||||||
|
fontSize: 12,
|
||||||
|
fontWeight: 400,
|
||||||
|
paddingBottom: 6,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
x 10 =
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
{/* Column 3: Maximum RU/s */}
|
||||||
|
<Stack tokens={{ childrenGap: 4 }}>
|
||||||
|
<Stack horizontal verticalAlign="center" tokens={{ childrenGap: 4 }}>
|
||||||
|
<Text variant="small" style={{ lineHeight: "20px", fontWeight: 600 }}>
|
||||||
|
Maximum RU/s
|
||||||
|
</Text>
|
||||||
|
<FontIcon iconName="Info" style={{ fontSize: 12, color: "#666" }} />
|
||||||
|
</Stack>
|
||||||
|
<TextField
|
||||||
|
required
|
||||||
|
type="number"
|
||||||
|
id="autopilotInput"
|
||||||
|
key="auto pilot throughput input"
|
||||||
|
styles={{
|
||||||
|
...getTextFieldStyles(this.props.maxAutoPilotThroughput, this.props.maxAutoPilotThroughputBaseline),
|
||||||
|
fieldGroup: { width: 100, height: 28 },
|
||||||
|
field: { fontSize: 14, fontWeight: 400 },
|
||||||
|
}}
|
||||||
|
disabled={this.overrideWithProvisionedThroughputSettings()}
|
||||||
|
step={AutoPilotUtils.autoPilotIncrementStep}
|
||||||
|
value={
|
||||||
|
this.overrideWithProvisionedThroughputSettings() ? "" : this.props.maxAutoPilotThroughput?.toString()
|
||||||
|
}
|
||||||
|
onChange={this.onAutoPilotThroughputChange}
|
||||||
|
min={autoPilotThroughput1K}
|
||||||
|
onGetErrorMessage={(value: string) => {
|
||||||
|
const sanitizedValue = getSanitizedInputValue(value);
|
||||||
|
return sanitizedValue % 1000
|
||||||
|
? "Throughput value must be in increments of 1000"
|
||||||
|
: this.props.throughputError;
|
||||||
|
}}
|
||||||
|
validateOnLoad={false}
|
||||||
|
/>
|
||||||
|
</Stack>
|
||||||
|
</Stack>
|
||||||
) : (
|
) : (
|
||||||
<TextField
|
<TextField
|
||||||
required
|
required
|
||||||
|
|||||||
@@ -157,35 +157,148 @@ exports[`ThroughputInputAutoPilotV3Component autopilot input visible 1`] = `
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<StyledTextFieldBase
|
<Stack
|
||||||
disabled={true}
|
horizontal={true}
|
||||||
id="autopilotInput"
|
tokens={
|
||||||
key="auto pilot throughput input"
|
|
||||||
label="Maximum RU/s required by this resource"
|
|
||||||
min={1000}
|
|
||||||
onChange={[Function]}
|
|
||||||
onGetErrorMessage={[Function]}
|
|
||||||
required={true}
|
|
||||||
step={1000}
|
|
||||||
styles={
|
|
||||||
{
|
{
|
||||||
"fieldGroup": {
|
"childrenGap": 8,
|
||||||
"borderColor": "",
|
|
||||||
"height": 25,
|
|
||||||
"selectors": {
|
|
||||||
":disabled": {
|
|
||||||
"backgroundColor": undefined,
|
|
||||||
"borderColor": undefined,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"width": 300,
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
type="number"
|
verticalAlign="end"
|
||||||
validateOnLoad={false}
|
>
|
||||||
value=""
|
<Stack
|
||||||
/>
|
tokens={
|
||||||
|
{
|
||||||
|
"childrenGap": 4,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<Stack
|
||||||
|
horizontal={true}
|
||||||
|
tokens={
|
||||||
|
{
|
||||||
|
"childrenGap": 4,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
verticalAlign="center"
|
||||||
|
>
|
||||||
|
<Text
|
||||||
|
style={
|
||||||
|
{
|
||||||
|
"fontWeight": 600,
|
||||||
|
"lineHeight": "20px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
variant="small"
|
||||||
|
>
|
||||||
|
Minimum RU/s
|
||||||
|
</Text>
|
||||||
|
<FontIcon
|
||||||
|
iconName="Info"
|
||||||
|
style={
|
||||||
|
{
|
||||||
|
"color": "#666",
|
||||||
|
"fontSize": 12,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</Stack>
|
||||||
|
<Text
|
||||||
|
style={
|
||||||
|
{
|
||||||
|
"alignItems": "center",
|
||||||
|
"backgroundColor": "transparent",
|
||||||
|
"border": "none",
|
||||||
|
"boxSizing": "border-box",
|
||||||
|
"display": "flex",
|
||||||
|
"fontFamily": "Segoe UI",
|
||||||
|
"fontSize": 14,
|
||||||
|
"fontWeight": 400,
|
||||||
|
"height": 28,
|
||||||
|
"justifyContent": "center",
|
||||||
|
"width": 70,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
400
|
||||||
|
</Text>
|
||||||
|
</Stack>
|
||||||
|
<Text
|
||||||
|
style={
|
||||||
|
{
|
||||||
|
"fontFamily": "Segoe UI",
|
||||||
|
"fontSize": 12,
|
||||||
|
"fontWeight": 400,
|
||||||
|
"paddingBottom": 6,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
x 10 =
|
||||||
|
</Text>
|
||||||
|
<Stack
|
||||||
|
tokens={
|
||||||
|
{
|
||||||
|
"childrenGap": 4,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<Stack
|
||||||
|
horizontal={true}
|
||||||
|
tokens={
|
||||||
|
{
|
||||||
|
"childrenGap": 4,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
verticalAlign="center"
|
||||||
|
>
|
||||||
|
<Text
|
||||||
|
style={
|
||||||
|
{
|
||||||
|
"fontWeight": 600,
|
||||||
|
"lineHeight": "20px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
variant="small"
|
||||||
|
>
|
||||||
|
Maximum RU/s
|
||||||
|
</Text>
|
||||||
|
<FontIcon
|
||||||
|
iconName="Info"
|
||||||
|
style={
|
||||||
|
{
|
||||||
|
"color": "#666",
|
||||||
|
"fontSize": 12,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</Stack>
|
||||||
|
<StyledTextFieldBase
|
||||||
|
disabled={true}
|
||||||
|
id="autopilotInput"
|
||||||
|
key="auto pilot throughput input"
|
||||||
|
min={1000}
|
||||||
|
onChange={[Function]}
|
||||||
|
onGetErrorMessage={[Function]}
|
||||||
|
required={true}
|
||||||
|
step={1000}
|
||||||
|
styles={
|
||||||
|
{
|
||||||
|
"field": {
|
||||||
|
"fontSize": 14,
|
||||||
|
"fontWeight": 400,
|
||||||
|
},
|
||||||
|
"fieldGroup": {
|
||||||
|
"height": 28,
|
||||||
|
"width": 100,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
type="number"
|
||||||
|
validateOnLoad={false}
|
||||||
|
value=""
|
||||||
|
/>
|
||||||
|
</Stack>
|
||||||
|
</Stack>
|
||||||
<Stack>
|
<Stack>
|
||||||
<Stack>
|
<Stack>
|
||||||
<Stack
|
<Stack
|
||||||
|
|||||||
@@ -5,13 +5,13 @@ import { useDatabases } from "Explorer/useDatabases";
|
|||||||
import React, { FunctionComponent, useEffect, useState } from "react";
|
import React, { FunctionComponent, useEffect, useState } from "react";
|
||||||
import * as Constants from "../../../Common/Constants";
|
import * as Constants from "../../../Common/Constants";
|
||||||
import { InfoTooltip } from "../../../Common/Tooltip/InfoTooltip";
|
import { InfoTooltip } from "../../../Common/Tooltip/InfoTooltip";
|
||||||
|
import { isFabricNative } from "../../../Platform/Fabric/FabricUtil";
|
||||||
import * as SharedConstants from "../../../Shared/Constants";
|
import * as SharedConstants from "../../../Shared/Constants";
|
||||||
import { userContext } from "../../../UserContext";
|
import { userContext } from "../../../UserContext";
|
||||||
import { getCollectionName } from "../../../Utils/APITypeUtils";
|
import { getCollectionName } from "../../../Utils/APITypeUtils";
|
||||||
import * as AutoPilotUtils from "../../../Utils/AutoPilotUtils";
|
import * as AutoPilotUtils from "../../../Utils/AutoPilotUtils";
|
||||||
import * as PricingUtils from "../../../Utils/PricingUtils";
|
import * as PricingUtils from "../../../Utils/PricingUtils";
|
||||||
import "./ThroughputInput.less";
|
import "./ThroughputInput.less";
|
||||||
import { isFabricNative } from "../../../Platform/Fabric/FabricUtil";
|
|
||||||
|
|
||||||
export interface ThroughputInputProps {
|
export interface ThroughputInputProps {
|
||||||
isDatabase: boolean;
|
isDatabase: boolean;
|
||||||
@@ -41,11 +41,12 @@ export const ThroughputInput: FunctionComponent<ThroughputInputProps> = ({
|
|||||||
let defaultThroughput: number;
|
let defaultThroughput: number;
|
||||||
const workloadType: Constants.WorkloadType = getWorkloadType();
|
const workloadType: Constants.WorkloadType = getWorkloadType();
|
||||||
|
|
||||||
if (
|
if (isFabricNative()) {
|
||||||
|
defaultThroughput = AutoPilotUtils.autoPilotThroughput5K;
|
||||||
|
} else if (
|
||||||
isFreeTier ||
|
isFreeTier ||
|
||||||
isQuickstart ||
|
isQuickstart ||
|
||||||
[Constants.WorkloadType.Learning, Constants.WorkloadType.DevelopmentTesting].includes(workloadType) ||
|
[Constants.WorkloadType.Learning, Constants.WorkloadType.DevelopmentTesting].includes(workloadType)
|
||||||
isFabricNative()
|
|
||||||
) {
|
) {
|
||||||
defaultThroughput = AutoPilotUtils.autoPilotThroughput1K;
|
defaultThroughput = AutoPilotUtils.autoPilotThroughput1K;
|
||||||
} else if (workloadType === Constants.WorkloadType.Production) {
|
} else if (workloadType === Constants.WorkloadType.Production) {
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import { useDataplaneRbacAuthorization } from "Utils/AuthorizationUtils";
|
||||||
import { createCollection } from "../../Common/dataAccess/createCollection";
|
import { createCollection } from "../../Common/dataAccess/createCollection";
|
||||||
import { createDocument } from "../../Common/dataAccess/createDocument";
|
import { createDocument } from "../../Common/dataAccess/createDocument";
|
||||||
import { createDocument as createMongoDocument } from "../../Common/MongoProxyClient";
|
import { createDocument as createMongoDocument } from "../../Common/MongoProxyClient";
|
||||||
@@ -90,12 +91,13 @@ export class ContainerSampleGenerator {
|
|||||||
}
|
}
|
||||||
const { databaseAccount: account } = userContext;
|
const { databaseAccount: account } = userContext;
|
||||||
const databaseId = collection.databaseId;
|
const databaseId = collection.databaseId;
|
||||||
|
|
||||||
const gremlinClient = new GremlinClient();
|
const gremlinClient = new GremlinClient();
|
||||||
gremlinClient.initialize({
|
gremlinClient.initialize({
|
||||||
endpoint: `wss://${GraphTab.getGremlinEndpoint(account)}`,
|
endpoint: `wss://${GraphTab.getGremlinEndpoint(account)}`,
|
||||||
databaseId: databaseId,
|
databaseId: databaseId,
|
||||||
collectionId: collection.id(),
|
collectionId: collection.id(),
|
||||||
masterKey: userContext.masterKey || "",
|
password: useDataplaneRbacAuthorization(userContext) ? userContext.aadToken : userContext.masterKey || "",
|
||||||
maxResultSize: 100,
|
maxResultSize: 100,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -8,7 +8,12 @@ import { MessageTypes } from "Contracts/ExplorerContracts";
|
|||||||
import { useDataPlaneRbac } from "Explorer/Panes/SettingsPane/SettingsPane";
|
import { useDataPlaneRbac } from "Explorer/Panes/SettingsPane/SettingsPane";
|
||||||
import { getCopilotEnabled, isCopilotFeatureRegistered } from "Explorer/QueryCopilot/Shared/QueryCopilotClient";
|
import { getCopilotEnabled, isCopilotFeatureRegistered } from "Explorer/QueryCopilot/Shared/QueryCopilotClient";
|
||||||
import { IGalleryItem } from "Juno/JunoClient";
|
import { IGalleryItem } from "Juno/JunoClient";
|
||||||
import { isFabricMirrored, isFabricMirroredKey, scheduleRefreshFabricToken } from "Platform/Fabric/FabricUtil";
|
import {
|
||||||
|
isFabricMirrored,
|
||||||
|
isFabricMirroredKey,
|
||||||
|
isFabricNative,
|
||||||
|
scheduleRefreshFabricToken,
|
||||||
|
} from "Platform/Fabric/FabricUtil";
|
||||||
import { LocalStorageUtility, StorageKey } from "Shared/StorageUtility";
|
import { LocalStorageUtility, StorageKey } from "Shared/StorageUtility";
|
||||||
import { acquireMsalTokenForAccount } from "Utils/AuthorizationUtils";
|
import { acquireMsalTokenForAccount } from "Utils/AuthorizationUtils";
|
||||||
import { allowedNotebookServerUrls, validateEndpoint } from "Utils/EndpointUtils";
|
import { allowedNotebookServerUrls, validateEndpoint } from "Utils/EndpointUtils";
|
||||||
@@ -284,14 +289,40 @@ export default class Explorer {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public openInVsCode(): void {
|
/**
|
||||||
|
* Generates a VS Code DocumentDB connection URL using the current user's MongoDB connection parameters.
|
||||||
|
* Double-encodes the updated connection string for safe usage in VS Code URLs.
|
||||||
|
*
|
||||||
|
* The DocumentDB VS Code extension requires double encoding for connection strings.
|
||||||
|
* See: https://microsoft.github.io/vscode-documentdb/manual/how-to-construct-url.html#double-encoding
|
||||||
|
*
|
||||||
|
* @returns {string} The encoded VS Code DocumentDB connection URL.
|
||||||
|
*/
|
||||||
|
private getDocumentDbUrl() {
|
||||||
|
const { adminLogin: adminLoginuserName = "", connectionString = "" } = userContext.vcoreMongoConnectionParams;
|
||||||
|
const updatedConnectionString = connectionString.replace(/<(user|username)>:<password>/i, adminLoginuserName);
|
||||||
|
const encodedUpdatedConnectionString = encodeURIComponent(encodeURIComponent(updatedConnectionString));
|
||||||
|
const documentDbUrl = `vscode://ms-azuretools.vscode-documentdb?connectionString=${encodedUpdatedConnectionString}`;
|
||||||
|
return documentDbUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
private getCosmosDbUrl() {
|
||||||
const activeTab = useTabs.getState().activeTab;
|
const activeTab = useTabs.getState().activeTab;
|
||||||
const resourceId = encodeURIComponent(userContext.databaseAccount.id);
|
const resourceId = encodeURIComponent(userContext.databaseAccount.id);
|
||||||
const database = encodeURIComponent(activeTab?.collection?.databaseId);
|
const database = encodeURIComponent(activeTab?.collection?.databaseId);
|
||||||
const container = encodeURIComponent(activeTab?.collection?.id());
|
const container = encodeURIComponent(activeTab?.collection?.id());
|
||||||
const baseUrl = `vscode://ms-azuretools.vscode-cosmosdb?resourceId=${resourceId}`;
|
const baseUrl = `vscode://ms-azuretools.vscode-cosmosdb?resourceId=${resourceId}`;
|
||||||
const vscodeUrl = activeTab ? `${baseUrl}&database=${database}&container=${container}` : baseUrl;
|
const vscodeUrl = activeTab ? `${baseUrl}&database=${database}&container=${container}` : baseUrl;
|
||||||
|
return vscodeUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
private getVSCodeUrl(): string {
|
||||||
|
const isvCore = (userContext.apiType || userContext.databaseAccount.kind) === "VCoreMongo";
|
||||||
|
return isvCore ? this.getDocumentDbUrl() : this.getCosmosDbUrl();
|
||||||
|
}
|
||||||
|
|
||||||
|
public openInVsCode(): void {
|
||||||
|
const vscodeUrl = this.getVSCodeUrl();
|
||||||
const openVSCodeDialogProps: DialogProps = {
|
const openVSCodeDialogProps: DialogProps = {
|
||||||
linkProps: {
|
linkProps: {
|
||||||
linkText: "Download Visual Studio Code",
|
linkText: "Download Visual Studio Code",
|
||||||
@@ -1149,7 +1180,10 @@ export default class Explorer {
|
|||||||
? this.refreshDatabaseForResourceToken()
|
? this.refreshDatabaseForResourceToken()
|
||||||
: await this.refreshAllDatabases(); // await: we rely on the databases to be loaded before restoring the tabs further in the flow
|
: await this.refreshAllDatabases(); // await: we rely on the databases to be loaded before restoring the tabs further in the flow
|
||||||
}
|
}
|
||||||
await useNotebook.getState().refreshNotebooksEnabledStateForAccount();
|
|
||||||
|
if (!isFabricNative()) {
|
||||||
|
await useNotebook.getState().refreshNotebooksEnabledStateForAccount();
|
||||||
|
}
|
||||||
|
|
||||||
// TODO: remove reference to isNotebookEnabled and isNotebooksEnabledForAccount
|
// TODO: remove reference to isNotebookEnabled and isNotebooksEnabledForAccount
|
||||||
const isNotebookEnabled =
|
const isNotebookEnabled =
|
||||||
@@ -1171,7 +1205,7 @@ export default class Explorer {
|
|||||||
await this.initNotebooks(userContext.databaseAccount);
|
await this.initNotebooks(userContext.databaseAccount);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (userContext.authType === AuthType.AAD && userContext.apiType === "SQL") {
|
if (userContext.authType === AuthType.AAD && userContext.apiType === "SQL" && !isFabricNative()) {
|
||||||
const throughputBucketsEnabled = await featureRegistered(userContext.subscriptionId, "ThroughputBucketing");
|
const throughputBucketsEnabled = await featureRegistered(userContext.subscriptionId, "ThroughputBucketing");
|
||||||
updateUserContext({ throughputBucketsEnabled });
|
updateUserContext({ throughputBucketsEnabled });
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -163,8 +163,7 @@ describe("GraphExplorer", () => {
|
|||||||
graphBackendEndpoint: "graphBackendEndpoint",
|
graphBackendEndpoint: "graphBackendEndpoint",
|
||||||
databaseId: "databaseId",
|
databaseId: "databaseId",
|
||||||
collectionId: "collectionId",
|
collectionId: "collectionId",
|
||||||
masterKey: "masterKey",
|
password: "password",
|
||||||
|
|
||||||
onLoadStartKey: 0,
|
onLoadStartKey: 0,
|
||||||
onLoadStartKeyChange: (newKey: number): void => {},
|
onLoadStartKeyChange: (newKey: number): void => {},
|
||||||
resourceId: "resourceId",
|
resourceId: "resourceId",
|
||||||
|
|||||||
@@ -59,7 +59,7 @@ export interface GraphExplorerProps {
|
|||||||
graphBackendEndpoint: string;
|
graphBackendEndpoint: string;
|
||||||
databaseId: string;
|
databaseId: string;
|
||||||
collectionId: string;
|
collectionId: string;
|
||||||
masterKey: string;
|
password: string;
|
||||||
|
|
||||||
onLoadStartKey: number;
|
onLoadStartKey: number;
|
||||||
onLoadStartKeyChange: (newKey: number) => void;
|
onLoadStartKeyChange: (newKey: number) => void;
|
||||||
@@ -1300,7 +1300,7 @@ export class GraphExplorer extends React.Component<GraphExplorerProps, GraphExpl
|
|||||||
endpoint: `wss://${this.props.graphBackendEndpoint}`,
|
endpoint: `wss://${this.props.graphBackendEndpoint}`,
|
||||||
databaseId: this.props.databaseId,
|
databaseId: this.props.databaseId,
|
||||||
collectionId: this.props.collectionId,
|
collectionId: this.props.collectionId,
|
||||||
masterKey: this.props.masterKey,
|
password: this.props.password,
|
||||||
maxResultSize: GraphExplorer.MAX_RESULT_SIZE,
|
maxResultSize: GraphExplorer.MAX_RESULT_SIZE,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,28 +8,28 @@ describe("Gremlin Client", () => {
|
|||||||
endpoint: null,
|
endpoint: null,
|
||||||
collectionId: null,
|
collectionId: null,
|
||||||
databaseId: null,
|
databaseId: null,
|
||||||
masterKey: null,
|
|
||||||
maxResultSize: 10000,
|
maxResultSize: 10000,
|
||||||
|
password: null,
|
||||||
};
|
};
|
||||||
|
|
||||||
it("should use databaseId, collectionId and masterKey to authenticate", () => {
|
it("should use databaseId, collectionId and password to authenticate", () => {
|
||||||
const collectionId = "collectionId";
|
const collectionId = "collectionId";
|
||||||
const databaseId = "databaseId";
|
const databaseId = "databaseId";
|
||||||
const masterKey = "masterKey";
|
const testPassword = "password";
|
||||||
const gremlinClient = new GremlinClient();
|
const gremlinClient = new GremlinClient();
|
||||||
|
|
||||||
gremlinClient.initialize({
|
gremlinClient.initialize({
|
||||||
endpoint: null,
|
endpoint: null,
|
||||||
collectionId,
|
collectionId,
|
||||||
databaseId,
|
databaseId,
|
||||||
masterKey,
|
|
||||||
maxResultSize: 0,
|
maxResultSize: 0,
|
||||||
|
password: testPassword,
|
||||||
});
|
});
|
||||||
|
|
||||||
// User must includes these values
|
// User must includes these values
|
||||||
expect(gremlinClient.client.params.user.indexOf(collectionId)).not.toBe(-1);
|
expect(gremlinClient.client.params.user.indexOf(collectionId)).not.toBe(-1);
|
||||||
expect(gremlinClient.client.params.user.indexOf(databaseId)).not.toBe(-1);
|
expect(gremlinClient.client.params.user.indexOf(databaseId)).not.toBe(-1);
|
||||||
expect(gremlinClient.client.params.password).toEqual(masterKey);
|
expect(gremlinClient.client.params.password).toEqual(testPassword);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should aggregate RU charges across multiple responses", (done) => {
|
it("should aggregate RU charges across multiple responses", (done) => {
|
||||||
|
|||||||
@@ -11,8 +11,8 @@ export interface GremlinClientParameters {
|
|||||||
endpoint: string;
|
endpoint: string;
|
||||||
databaseId: string;
|
databaseId: string;
|
||||||
collectionId: string;
|
collectionId: string;
|
||||||
masterKey: string;
|
|
||||||
maxResultSize: number;
|
maxResultSize: number;
|
||||||
|
password: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface GremlinRequestResult {
|
export interface GremlinRequestResult {
|
||||||
@@ -43,7 +43,7 @@ export class GremlinClient {
|
|||||||
this.client = new GremlinSimpleClient({
|
this.client = new GremlinSimpleClient({
|
||||||
endpoint: params.endpoint,
|
endpoint: params.endpoint,
|
||||||
user: `/dbs/${params.databaseId}/colls/${params.collectionId}`,
|
user: `/dbs/${params.databaseId}/colls/${params.collectionId}`,
|
||||||
password: params.masterKey,
|
password: params.password,
|
||||||
successCallback: (result: Result) => {
|
successCallback: (result: Result) => {
|
||||||
this.storePendingResult(result);
|
this.storePendingResult(result);
|
||||||
this.flushResult(result.requestId);
|
this.flushResult(result.requestId);
|
||||||
|
|||||||
@@ -5,11 +5,11 @@
|
|||||||
|
|
||||||
import * as sinon from "sinon";
|
import * as sinon from "sinon";
|
||||||
import {
|
import {
|
||||||
|
GremlinRequestMessage,
|
||||||
|
GremlinResponseMessage,
|
||||||
GremlinSimpleClient,
|
GremlinSimpleClient,
|
||||||
GremlinSimpleClientParameters,
|
GremlinSimpleClientParameters,
|
||||||
Result,
|
Result,
|
||||||
GremlinRequestMessage,
|
|
||||||
GremlinResponseMessage,
|
|
||||||
} from "./GremlinSimpleClient";
|
} from "./GremlinSimpleClient";
|
||||||
|
|
||||||
describe("Gremlin Simple Client", () => {
|
describe("Gremlin Simple Client", () => {
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user