mirror of
https://github.com/Azure/cosmos-explorer.git
synced 2025-12-21 09:51:11 +00:00
resolve master merge conflict
This commit is contained in:
@@ -133,7 +133,6 @@ src/Explorer/Panes/Tables/AddTableEntityPane.ts
|
|||||||
src/Explorer/Panes/Tables/EditTableEntityPane.ts
|
src/Explorer/Panes/Tables/EditTableEntityPane.ts
|
||||||
src/Explorer/Panes/Tables/EntityPropertyViewModel.ts
|
src/Explorer/Panes/Tables/EntityPropertyViewModel.ts
|
||||||
src/Explorer/Panes/Tables/QuerySelectPane.ts
|
src/Explorer/Panes/Tables/QuerySelectPane.ts
|
||||||
src/Explorer/Panes/Tables/TableColumnOptionsPane.ts
|
|
||||||
src/Explorer/Panes/Tables/TableEntityPane.ts
|
src/Explorer/Panes/Tables/TableEntityPane.ts
|
||||||
src/Explorer/Panes/Tables/Validators/EntityPropertyNameValidator.ts
|
src/Explorer/Panes/Tables/Validators/EntityPropertyNameValidator.ts
|
||||||
src/Explorer/Panes/Tables/Validators/EntityPropertyValidationCommon.ts
|
src/Explorer/Panes/Tables/Validators/EntityPropertyValidationCommon.ts
|
||||||
|
|||||||
214
package-lock.json
generated
214
package-lock.json
generated
@@ -38078,73 +38078,67 @@
|
|||||||
"@testing-library/dom": "^7.28.1"
|
"@testing-library/dom": "^7.28.1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"@types/anymatch": {
|
"@microsoft/applicationinsights-analytics-js": {
|
||||||
"version": "1.3.1",
|
"version": "2.6.1",
|
||||||
"resolved": "https://registry.npmjs.org/@types/anymatch/-/anymatch-1.3.1.tgz",
|
"resolved": "https://registry.npmjs.org/@microsoft/applicationinsights-analytics-js/-/applicationinsights-analytics-js-2.6.1.tgz",
|
||||||
"integrity": "sha512-/+CRPXpBDpo2RK9C68N3b2cOvO0Cf5B9aPijHsoDQTHivnGSObdOF2BRQOYjojWTDy6nQvMjmqRXIxH55VjxxA==",
|
"integrity": "sha512-wRH67jZTPy6SP54ygAQsFXu5ZnfzGOoMkA6ll1pkhSC4hij7Cvzumr4PYkr2gIPBPxD354sLOjBL/lmOgBTc5g==",
|
||||||
"dev": true
|
|
||||||
},
|
|
||||||
"@types/applicationinsights-js": {
|
|
||||||
"version": "1.0.7",
|
|
||||||
"resolved": "https://registry.npmjs.org/@types/applicationinsights-js/-/applicationinsights-js-1.0.7.tgz",
|
|
||||||
"integrity": "sha512-oIqnhcvhz0NRJ+vSN56leWsHyUmz7bySe5f88ukwPhXlNzfTB3JyZ6/bw4+B8zjSktBoYxR9Pzf0t18XwRiwHw==",
|
|
||||||
"dev": true
|
|
||||||
},
|
|
||||||
"@types/aria-query": {
|
|
||||||
"version": "4.2.1",
|
|
||||||
"resolved": "https://registry.npmjs.org/@types/aria-query/-/aria-query-4.2.1.tgz",
|
|
||||||
"integrity": "sha512-S6oPal772qJZHoRZLFc/XoZW2gFvwXusYUmXPXkgxJLuEk2vOt7jc4Yo6z/vtI0EBkbPBVrJJ0B+prLIKiWqHg==",
|
|
||||||
"dev": true
|
|
||||||
},
|
|
||||||
"@types/asap": {
|
|
||||||
"version": "2.0.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/@types/asap/-/asap-2.0.0.tgz",
|
|
||||||
"integrity": "sha512-upIS0Gt9Mc8eEpCbYMZ1K8rhNosfKUtimNcINce+zLwJF5UpM3Vv7yz3S5l/1IX+DxTa8lTkUjqynvjRXyJzsg=="
|
|
||||||
},
|
|
||||||
"@types/babel__core": {
|
|
||||||
"version": "7.1.14",
|
|
||||||
"resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.1.14.tgz",
|
|
||||||
"integrity": "sha512-zGZJzzBUVDo/eV6KgbE0f0ZI7dInEYvo12Rb70uNQDshC3SkRMb67ja0GgRHZgAX3Za6rhaWlvbDO8rrGyAb1g==",
|
|
||||||
"requires": {
|
"requires": {
|
||||||
"@babel/parser": "^7.1.0",
|
"@microsoft/applicationinsights-common": "2.6.1",
|
||||||
"@babel/types": "^7.0.0",
|
"@microsoft/applicationinsights-core-js": "2.6.1",
|
||||||
"@types/babel__generator": "*",
|
"@microsoft/applicationinsights-shims": "1.0.3",
|
||||||
"@types/babel__template": "*",
|
"@microsoft/dynamicproto-js": "^1.1.1"
|
||||||
"@types/babel__traverse": "*"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"@types/babel__generator": {
|
"@microsoft/applicationinsights-channel-js": {
|
||||||
"version": "7.6.2",
|
"version": "2.6.1",
|
||||||
"resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.6.2.tgz",
|
"resolved": "https://registry.npmjs.org/@microsoft/applicationinsights-channel-js/-/applicationinsights-channel-js-2.6.1.tgz",
|
||||||
"integrity": "sha512-MdSJnBjl+bdwkLskZ3NGFp9YcXGx5ggLpQQPqtgakVhsWK0hTtNYhjpZLlWQTviGTvF8at+Bvli3jV7faPdgeQ==",
|
"integrity": "sha512-oatERKW3eAjVNm5ej2NpRvBCCtPtiINIKxLiWVHv2SC2rzGrUnxNWFlUJojRT+u5ZXXsB51Ktw9gx8iHexnDVw==",
|
||||||
"requires": {
|
"requires": {
|
||||||
"@babel/types": "^7.0.0"
|
"@microsoft/applicationinsights-common": "2.6.1",
|
||||||
|
"@microsoft/applicationinsights-core-js": "2.6.1",
|
||||||
|
"@microsoft/applicationinsights-shims": "1.0.3",
|
||||||
|
"@microsoft/dynamicproto-js": "^1.1.1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"@types/babel__template": {
|
"@microsoft/applicationinsights-common": {
|
||||||
"version": "7.4.0",
|
"version": "2.6.1",
|
||||||
"resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.0.tgz",
|
"resolved": "https://registry.npmjs.org/@microsoft/applicationinsights-common/-/applicationinsights-common-2.6.1.tgz",
|
||||||
"integrity": "sha512-NTPErx4/FiPCGScH7foPyr+/1Dkzkni+rHiYHHoTjvwou7AQzJkNeD60A9CXRy+ZEN2B1bggmkTMCDb+Mv5k+A==",
|
"integrity": "sha512-kV/dI9UwTew3mq+3F3Tkl2dBn2C4+FOOG3S5cS7/SssVsUlvLrWXBrOOxfIJ7+EjQQ6ijcqlDus4Nz7Ms2Kk2Q==",
|
||||||
"requires": {
|
"requires": {
|
||||||
"@babel/parser": "^7.1.0",
|
"@microsoft/applicationinsights-core-js": "2.6.1",
|
||||||
"@babel/types": "^7.0.0"
|
"@microsoft/applicationinsights-shims": "1.0.3",
|
||||||
|
"@microsoft/dynamicproto-js": "^1.1.1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"@types/babel__traverse": {
|
"@microsoft/applicationinsights-core-js": {
|
||||||
"version": "7.11.1",
|
"version": "2.6.1",
|
||||||
"resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.11.1.tgz",
|
"resolved": "https://registry.npmjs.org/@microsoft/applicationinsights-core-js/-/applicationinsights-core-js-2.6.1.tgz",
|
||||||
"integrity": "sha512-Vs0hm0vPahPMYi9tDjtP66llufgO3ST16WXaSTtDGEl9cewAl3AibmxWw6TINOqHPT9z0uABKAYjT9jNSg4npw==",
|
"integrity": "sha512-SFnTx48BGkmz6P9GvXFeIZ641vnfrKo0hB74Hp9GR7UE3hqIDuomIBQS17unnh05T5w3PWKlDCGNpiCbJV6kzQ==",
|
||||||
"requires": {
|
"requires": {
|
||||||
"@babel/types": "^7.3.0"
|
"@microsoft/applicationinsights-shims": "1.0.3",
|
||||||
|
"@microsoft/dynamicproto-js": "^1.1.1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"@types/cheerio": {
|
"@microsoft/applicationinsights-dependencies-js": {
|
||||||
"version": "0.22.28",
|
"version": "2.6.1",
|
||||||
"resolved": "https://registry.npmjs.org/@types/cheerio/-/cheerio-0.22.28.tgz",
|
"resolved": "https://registry.npmjs.org/@microsoft/applicationinsights-dependencies-js/-/applicationinsights-dependencies-js-2.6.1.tgz",
|
||||||
"integrity": "sha512-ehUMGSW5IeDxJjbru4awKYMlKGmo1wSSGUVqXtYwlgmUM8X1a0PZttEIm6yEY7vHsY/hh6iPnklF213G0UColw==",
|
"integrity": "sha512-Y7R6BjzyB6NrkIBTT6FJrr2jKr8MiLxy264b1xUQHjry3Bwu+fsPJ7EETV61dkOEHs6IlWxAtikNCv8f3zQydw==",
|
||||||
"dev": true,
|
|
||||||
"requires": {
|
"requires": {
|
||||||
"@types/node": "*"
|
"@microsoft/applicationinsights-common": "2.6.1",
|
||||||
|
"@microsoft/applicationinsights-core-js": "2.6.1",
|
||||||
|
"@microsoft/applicationinsights-shims": "1.0.3",
|
||||||
|
"@microsoft/dynamicproto-js": "^1.1.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"@microsoft/applicationinsights-properties-js": {
|
||||||
|
"version": "2.6.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/@microsoft/applicationinsights-properties-js/-/applicationinsights-properties-js-2.6.1.tgz",
|
||||||
|
"integrity": "sha512-TQmKp9/j0yMGjUHBatRQ41E4S/1yTkJImh8csZxc4waMQmV9ITkNrDaC3bIPcrw+ASepI3QfP7sry+n7nvYLFw==",
|
||||||
|
"requires": {
|
||||||
|
"@microsoft/applicationinsights-common": "2.6.1",
|
||||||
|
"@microsoft/applicationinsights-core-js": "2.6.1",
|
||||||
|
"@microsoft/applicationinsights-shims": "1.0.3",
|
||||||
|
"@microsoft/dynamicproto-js": "^1.1.1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"@types/codemirror": {
|
"@types/codemirror": {
|
||||||
@@ -38153,20 +38147,96 @@
|
|||||||
"integrity": "sha512-OMtPqg2wFOEcNeVga+m+UXpYJw8ugISPCQOtShdFUho/k91Ms1oWOozoDT1I87Phv6IdwLfMLtIOahh1tO1cJQ==",
|
"integrity": "sha512-OMtPqg2wFOEcNeVga+m+UXpYJw8ugISPCQOtShdFUho/k91Ms1oWOozoDT1I87Phv6IdwLfMLtIOahh1tO1cJQ==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
"@types/crossroads": {
|
"@microsoft/applicationinsights-web": {
|
||||||
"version": "0.0.30",
|
"version": "2.6.1",
|
||||||
"resolved": "https://registry.npmjs.org/@types/crossroads/-/crossroads-0.0.30.tgz",
|
"resolved": "https://registry.npmjs.org/@microsoft/applicationinsights-web/-/applicationinsights-web-2.6.1.tgz",
|
||||||
"integrity": "sha512-1Ze8YTD5tFWeM5CtqAiJ348tDDYBOGDdLid4rnnTLYDU5sJgEjMknLx4g5yxWDHCr51nSl4Z9M/ZAEqTtyEMVA==",
|
"integrity": "sha512-jKF6hpPq3pzLqLSVxTMh7HU9tr/SPOFJN12pFYpNMU/rAMs/fgZsmFG/oBAzYTZxNilDTFljoblB66J9X+K6RQ==",
|
||||||
"dev": true,
|
|
||||||
"requires": {
|
"requires": {
|
||||||
"@types/signals": "*"
|
"@microsoft/applicationinsights-analytics-js": "2.6.1",
|
||||||
|
"@microsoft/applicationinsights-channel-js": "2.6.1",
|
||||||
|
"@microsoft/applicationinsights-common": "2.6.1",
|
||||||
|
"@microsoft/applicationinsights-core-js": "2.6.1",
|
||||||
|
"@microsoft/applicationinsights-dependencies-js": "2.6.1",
|
||||||
|
"@microsoft/applicationinsights-properties-js": "2.6.1",
|
||||||
|
"@microsoft/applicationinsights-shims": "1.0.3",
|
||||||
|
"@microsoft/dynamicproto-js": "^1.1.1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"@types/d3": {
|
"@microsoft/dynamicproto-js": {
|
||||||
"version": "5.9.2",
|
"version": "1.1.1",
|
||||||
"resolved": "https://registry.npmjs.org/@types/d3/-/d3-5.9.2.tgz",
|
"resolved": "https://registry.npmjs.org/@microsoft/dynamicproto-js/-/dynamicproto-js-1.1.1.tgz",
|
||||||
"integrity": "sha512-eqiXmREP+cZhngaPalnJ1M88tXO74GyQB8TFd3h20e4zf5NOwLVGGlHU2ZoQ1e+rcdbfboSSwxffoQPIxEh55w==",
|
"integrity": "sha512-SPY1PlXmg4FbJVItVdhDV+zigajPtSI8oZPrsKBEIzAF4FROuwWIq5C+RAF8VJshBqphcvU8Eoh4DrUEZOB31g==",
|
||||||
"dev": true,
|
"requires": {
|
||||||
|
"findup-sync": "^4.0.0",
|
||||||
|
"nopt": "^5.0.0",
|
||||||
|
"pify": "^2.3.0"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"braces": {
|
||||||
|
"version": "3.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz",
|
||||||
|
"integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==",
|
||||||
|
"requires": {
|
||||||
|
"fill-range": "^7.0.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"fill-range": {
|
||||||
|
"version": "7.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz",
|
||||||
|
"integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==",
|
||||||
|
"requires": {
|
||||||
|
"to-regex-range": "^5.0.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"findup-sync": {
|
||||||
|
"version": "4.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/findup-sync/-/findup-sync-4.0.0.tgz",
|
||||||
|
"integrity": "sha512-6jvvn/12IC4quLBL1KNokxC7wWTvYncaVUYSoxWw7YykPLuRrnv4qdHcSOywOI5RpkOVGeQRtWM8/q+G6W6qfQ==",
|
||||||
|
"requires": {
|
||||||
|
"detect-file": "^1.0.0",
|
||||||
|
"is-glob": "^4.0.0",
|
||||||
|
"micromatch": "^4.0.2",
|
||||||
|
"resolve-dir": "^1.0.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"is-number": {
|
||||||
|
"version": "7.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
|
||||||
|
"integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng=="
|
||||||
|
},
|
||||||
|
"micromatch": {
|
||||||
|
"version": "4.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz",
|
||||||
|
"integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==",
|
||||||
|
"requires": {
|
||||||
|
"braces": "^3.0.1",
|
||||||
|
"picomatch": "^2.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"pify": {
|
||||||
|
"version": "2.3.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz",
|
||||||
|
"integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw="
|
||||||
|
},
|
||||||
|
"to-regex-range": {
|
||||||
|
"version": "5.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz",
|
||||||
|
"integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==",
|
||||||
|
"requires": {
|
||||||
|
"is-number": "^7.0.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"@microsoft/load-themed-styles": {
|
||||||
|
"version": "1.10.146",
|
||||||
|
"resolved": "https://registry.npmjs.org/@microsoft/load-themed-styles/-/load-themed-styles-1.10.146.tgz",
|
||||||
|
"integrity": "sha512-qQZ4J58J2VMe/XRpr2YRDusQB9uRBJ1SjJB76x7uH94t9hqxjVVxn2qL99Bl+ERbfrACZ9peGn2uamt4ponqZQ=="
|
||||||
|
},
|
||||||
|
"@nodelib/fs.scandir": {
|
||||||
|
"version": "2.1.4",
|
||||||
|
"resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.4.tgz",
|
||||||
|
"integrity": "sha512-33g3pMJk3bg5nXbL/+CY6I2eJDzZAni49PfJnL5fghPTggPvBd/pFNSgJsdAgWptuFu7qq/ERvOYFlhvsLTCKA==",
|
||||||
"requires": {
|
"requires": {
|
||||||
"@types/d3-array": "^1",
|
"@types/d3-array": "^1",
|
||||||
"@types/d3-axis": "*",
|
"@types/d3-axis": "*",
|
||||||
@@ -40732,10 +40802,18 @@
|
|||||||
"resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz",
|
||||||
"integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk="
|
"integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk="
|
||||||
},
|
},
|
||||||
"buffer-from": {
|
"abbrev": {
|
||||||
"version": "1.1.1",
|
"version": "1.1.1",
|
||||||
"resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz",
|
||||||
"integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A=="
|
"integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q=="
|
||||||
|
},
|
||||||
|
"abort-controller": {
|
||||||
|
"version": "3.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz",
|
||||||
|
"integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==",
|
||||||
|
"requires": {
|
||||||
|
"event-target-shim": "^5.0.0"
|
||||||
|
}
|
||||||
},
|
},
|
||||||
"buffer-indexof": {
|
"buffer-indexof": {
|
||||||
"version": "1.1.1",
|
"version": "1.1.1",
|
||||||
|
|||||||
@@ -13,7 +13,7 @@
|
|||||||
"@babel/plugin-proposal-decorators": "7.12.12",
|
"@babel/plugin-proposal-decorators": "7.12.12",
|
||||||
"@jupyterlab/services": "6.0.2",
|
"@jupyterlab/services": "6.0.2",
|
||||||
"@jupyterlab/terminal": "3.0.3",
|
"@jupyterlab/terminal": "3.0.3",
|
||||||
"@microsoft/applicationinsights-web": "2.5.9",
|
"@microsoft/applicationinsights-web": "2.6.1",
|
||||||
"@nteract/commutable": "7.4.2",
|
"@nteract/commutable": "7.4.2",
|
||||||
"@nteract/connected-components": "6.8.2",
|
"@nteract/connected-components": "6.8.2",
|
||||||
"@nteract/core": "15.1.0",
|
"@nteract/core": "15.1.0",
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { sendMessage } from "./MessageHandler";
|
|
||||||
import { Diagnostics, MessageTypes } from "../Contracts/ExplorerContracts";
|
|
||||||
import { appInsights } from "../Shared/appInsights";
|
|
||||||
import { SeverityLevel } from "@microsoft/applicationinsights-web";
|
import { SeverityLevel } from "@microsoft/applicationinsights-web";
|
||||||
|
import { Diagnostics, MessageTypes } from "../Contracts/ExplorerContracts";
|
||||||
|
import { trackTrace } from "../Shared/appInsights";
|
||||||
|
import { sendMessage } from "./MessageHandler";
|
||||||
|
|
||||||
// TODO: Move to a separate Diagnostics folder
|
// TODO: Move to a separate Diagnostics folder
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
@@ -46,7 +46,7 @@ function _logEntry(entry: Diagnostics.LogEntry): void {
|
|||||||
return SeverityLevel.Information;
|
return SeverityLevel.Information;
|
||||||
}
|
}
|
||||||
})(entry.level);
|
})(entry.level);
|
||||||
appInsights.trackTrace({ message: entry.message, severityLevel }, { area: entry.area });
|
trackTrace({ message: entry.message, severityLevel }, { area: entry.area });
|
||||||
}
|
}
|
||||||
|
|
||||||
function _generateLogEntry(
|
function _generateLogEntry(
|
||||||
|
|||||||
@@ -10,7 +10,6 @@ import { GraphStyleComponent } from "./Graph/GraphStyleComponent/GraphStyleCompo
|
|||||||
import { NewVertexComponent } from "./Graph/NewVertexComponent/NewVertexComponent";
|
import { NewVertexComponent } from "./Graph/NewVertexComponent/NewVertexComponent";
|
||||||
import * as PaneComponents from "./Panes/PaneComponents";
|
import * as PaneComponents from "./Panes/PaneComponents";
|
||||||
import ConflictsTab from "./Tabs/ConflictsTab";
|
import ConflictsTab from "./Tabs/ConflictsTab";
|
||||||
import DatabaseSettingsTab from "./Tabs/DatabaseSettingsTab";
|
|
||||||
import DocumentsTab from "./Tabs/DocumentsTab";
|
import DocumentsTab from "./Tabs/DocumentsTab";
|
||||||
import GalleryTab from "./Tabs/GalleryTab";
|
import GalleryTab from "./Tabs/GalleryTab";
|
||||||
import GraphTab from "./Tabs/GraphTab";
|
import GraphTab from "./Tabs/GraphTab";
|
||||||
@@ -53,7 +52,6 @@ ko.components.register("tabs-manager", { template: TabsManagerTemplate });
|
|||||||
TerminalTab,
|
TerminalTab,
|
||||||
GalleryTab,
|
GalleryTab,
|
||||||
NotebookViewerTab,
|
NotebookViewerTab,
|
||||||
DatabaseSettingsTab,
|
|
||||||
DatabaseSettingsTabV2,
|
DatabaseSettingsTabV2,
|
||||||
].forEach(({ component: { name, template } }) => ko.components.register(name, { template }));
|
].forEach(({ component: { name, template } }) => ko.components.register(name, { template }));
|
||||||
|
|
||||||
@@ -68,7 +66,6 @@ ko.components.register("graph-new-vertex-pane", new PaneComponents.GraphNewVerte
|
|||||||
ko.components.register("graph-styling-pane", new PaneComponents.GraphStylingPaneComponent());
|
ko.components.register("graph-styling-pane", new PaneComponents.GraphStylingPaneComponent());
|
||||||
ko.components.register("table-add-entity-pane", new PaneComponents.TableAddEntityPaneComponent());
|
ko.components.register("table-add-entity-pane", new PaneComponents.TableAddEntityPaneComponent());
|
||||||
ko.components.register("table-edit-entity-pane", new PaneComponents.TableEditEntityPaneComponent());
|
ko.components.register("table-edit-entity-pane", new PaneComponents.TableEditEntityPaneComponent());
|
||||||
ko.components.register("table-column-options-pane", new PaneComponents.TableColumnOptionsPaneComponent());
|
|
||||||
ko.components.register("table-query-select-pane", new PaneComponents.TableQuerySelectPaneComponent());
|
ko.components.register("table-query-select-pane", new PaneComponents.TableQuerySelectPaneComponent());
|
||||||
ko.components.register("cassandra-add-collection-pane", new PaneComponents.CassandraAddCollectionPaneComponent());
|
ko.components.register("cassandra-add-collection-pane", new PaneComponents.CassandraAddCollectionPaneComponent());
|
||||||
ko.components.register("string-input-pane", new PaneComponents.StringInputPaneComponent());
|
ko.components.register("string-input-pane", new PaneComponents.StringInputPaneComponent());
|
||||||
|
|||||||
@@ -220,31 +220,6 @@ exports[`SettingsComponent renders 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
TableColumnOptionsPane {
|
|
||||||
"allSelected": [Function],
|
|
||||||
"anyColumnSelected": [Function],
|
|
||||||
"availableColumnsLabel": "Available Columns",
|
|
||||||
"canMoveDown": [Function],
|
|
||||||
"canMoveUp": [Function],
|
|
||||||
"canSelectAll": [Function],
|
|
||||||
"columnOptions": [Function],
|
|
||||||
"container": [Circular],
|
|
||||||
"firstFieldHasFocus": [Function],
|
|
||||||
"formErrors": [Function],
|
|
||||||
"formErrorsDetails": [Function],
|
|
||||||
"handleClick": [Function],
|
|
||||||
"id": "tablecolumnoptionspane",
|
|
||||||
"instructionLabel": "Choose the columns and the order in which you want to display them in the table.",
|
|
||||||
"isExecuting": [Function],
|
|
||||||
"isTemplateReady": [Function],
|
|
||||||
"moveDownLabel": "Move Down",
|
|
||||||
"moveUpLabel": "Move Up",
|
|
||||||
"noColumnSelectedWarning": "At least one column should be selected.",
|
|
||||||
"selectedColumnOption": null,
|
|
||||||
"title": [Function],
|
|
||||||
"titleLabel": "Column Options",
|
|
||||||
"visible": [Function],
|
|
||||||
},
|
|
||||||
QuerySelectPane {
|
QuerySelectPane {
|
||||||
"allSelected": [Function],
|
"allSelected": [Function],
|
||||||
"anyColumnSelected": [Function],
|
"anyColumnSelected": [Function],
|
||||||
@@ -607,7 +582,6 @@ exports[`SettingsComponent renders 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"flight": [Function],
|
|
||||||
"graphStylingPane": GraphStylingPane {
|
"graphStylingPane": GraphStylingPane {
|
||||||
"container": [Circular],
|
"container": [Circular],
|
||||||
"firstFieldHasFocus": [Function],
|
"firstFieldHasFocus": [Function],
|
||||||
@@ -629,7 +603,6 @@ exports[`SettingsComponent renders 1`] = `
|
|||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"hasStorageAnalyticsAfecFeature": [Function],
|
"hasStorageAnalyticsAfecFeature": [Function],
|
||||||
"hasWriteAccess": [Function],
|
|
||||||
"isAccountReady": [Function],
|
"isAccountReady": [Function],
|
||||||
"isAutoscaleDefaultEnabled": [Function],
|
"isAutoscaleDefaultEnabled": [Function],
|
||||||
"isCopyNotebookPaneEnabled": [Function],
|
"isCopyNotebookPaneEnabled": [Function],
|
||||||
@@ -782,32 +755,6 @@ exports[`SettingsComponent renders 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"subscriptionType": [Function],
|
|
||||||
"tableColumnOptionsPane": TableColumnOptionsPane {
|
|
||||||
"allSelected": [Function],
|
|
||||||
"anyColumnSelected": [Function],
|
|
||||||
"availableColumnsLabel": "Available Columns",
|
|
||||||
"canMoveDown": [Function],
|
|
||||||
"canMoveUp": [Function],
|
|
||||||
"canSelectAll": [Function],
|
|
||||||
"columnOptions": [Function],
|
|
||||||
"container": [Circular],
|
|
||||||
"firstFieldHasFocus": [Function],
|
|
||||||
"formErrors": [Function],
|
|
||||||
"formErrorsDetails": [Function],
|
|
||||||
"handleClick": [Function],
|
|
||||||
"id": "tablecolumnoptionspane",
|
|
||||||
"instructionLabel": "Choose the columns and the order in which you want to display them in the table.",
|
|
||||||
"isExecuting": [Function],
|
|
||||||
"isTemplateReady": [Function],
|
|
||||||
"moveDownLabel": "Move Down",
|
|
||||||
"moveUpLabel": "Move Up",
|
|
||||||
"noColumnSelectedWarning": "At least one column should be selected.",
|
|
||||||
"selectedColumnOption": null,
|
|
||||||
"title": [Function],
|
|
||||||
"titleLabel": "Column Options",
|
|
||||||
"visible": [Function],
|
|
||||||
},
|
|
||||||
"tabsManager": TabsManager {
|
"tabsManager": TabsManager {
|
||||||
"activeTab": [Function],
|
"activeTab": [Function],
|
||||||
"openedTabs": [Function],
|
"openedTabs": [Function],
|
||||||
@@ -1026,31 +973,6 @@ exports[`SettingsComponent renders 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
TableColumnOptionsPane {
|
|
||||||
"allSelected": [Function],
|
|
||||||
"anyColumnSelected": [Function],
|
|
||||||
"availableColumnsLabel": "Available Columns",
|
|
||||||
"canMoveDown": [Function],
|
|
||||||
"canMoveUp": [Function],
|
|
||||||
"canSelectAll": [Function],
|
|
||||||
"columnOptions": [Function],
|
|
||||||
"container": [Circular],
|
|
||||||
"firstFieldHasFocus": [Function],
|
|
||||||
"formErrors": [Function],
|
|
||||||
"formErrorsDetails": [Function],
|
|
||||||
"handleClick": [Function],
|
|
||||||
"id": "tablecolumnoptionspane",
|
|
||||||
"instructionLabel": "Choose the columns and the order in which you want to display them in the table.",
|
|
||||||
"isExecuting": [Function],
|
|
||||||
"isTemplateReady": [Function],
|
|
||||||
"moveDownLabel": "Move Down",
|
|
||||||
"moveUpLabel": "Move Up",
|
|
||||||
"noColumnSelectedWarning": "At least one column should be selected.",
|
|
||||||
"selectedColumnOption": null,
|
|
||||||
"title": [Function],
|
|
||||||
"titleLabel": "Column Options",
|
|
||||||
"visible": [Function],
|
|
||||||
},
|
|
||||||
QuerySelectPane {
|
QuerySelectPane {
|
||||||
"allSelected": [Function],
|
"allSelected": [Function],
|
||||||
"anyColumnSelected": [Function],
|
"anyColumnSelected": [Function],
|
||||||
@@ -1413,7 +1335,6 @@ exports[`SettingsComponent renders 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"flight": [Function],
|
|
||||||
"graphStylingPane": GraphStylingPane {
|
"graphStylingPane": GraphStylingPane {
|
||||||
"container": [Circular],
|
"container": [Circular],
|
||||||
"firstFieldHasFocus": [Function],
|
"firstFieldHasFocus": [Function],
|
||||||
@@ -1435,7 +1356,6 @@ exports[`SettingsComponent renders 1`] = `
|
|||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"hasStorageAnalyticsAfecFeature": [Function],
|
"hasStorageAnalyticsAfecFeature": [Function],
|
||||||
"hasWriteAccess": [Function],
|
|
||||||
"isAccountReady": [Function],
|
"isAccountReady": [Function],
|
||||||
"isAutoscaleDefaultEnabled": [Function],
|
"isAutoscaleDefaultEnabled": [Function],
|
||||||
"isCopyNotebookPaneEnabled": [Function],
|
"isCopyNotebookPaneEnabled": [Function],
|
||||||
@@ -1588,32 +1508,6 @@ exports[`SettingsComponent renders 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"subscriptionType": [Function],
|
|
||||||
"tableColumnOptionsPane": TableColumnOptionsPane {
|
|
||||||
"allSelected": [Function],
|
|
||||||
"anyColumnSelected": [Function],
|
|
||||||
"availableColumnsLabel": "Available Columns",
|
|
||||||
"canMoveDown": [Function],
|
|
||||||
"canMoveUp": [Function],
|
|
||||||
"canSelectAll": [Function],
|
|
||||||
"columnOptions": [Function],
|
|
||||||
"container": [Circular],
|
|
||||||
"firstFieldHasFocus": [Function],
|
|
||||||
"formErrors": [Function],
|
|
||||||
"formErrorsDetails": [Function],
|
|
||||||
"handleClick": [Function],
|
|
||||||
"id": "tablecolumnoptionspane",
|
|
||||||
"instructionLabel": "Choose the columns and the order in which you want to display them in the table.",
|
|
||||||
"isExecuting": [Function],
|
|
||||||
"isTemplateReady": [Function],
|
|
||||||
"moveDownLabel": "Move Down",
|
|
||||||
"moveUpLabel": "Move Up",
|
|
||||||
"noColumnSelectedWarning": "At least one column should be selected.",
|
|
||||||
"selectedColumnOption": null,
|
|
||||||
"title": [Function],
|
|
||||||
"titleLabel": "Column Options",
|
|
||||||
"visible": [Function],
|
|
||||||
},
|
|
||||||
"tabsManager": TabsManager {
|
"tabsManager": TabsManager {
|
||||||
"activeTab": [Function],
|
"activeTab": [Function],
|
||||||
"openedTabs": [Function],
|
"openedTabs": [Function],
|
||||||
@@ -1845,31 +1739,6 @@ exports[`SettingsComponent renders 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
TableColumnOptionsPane {
|
|
||||||
"allSelected": [Function],
|
|
||||||
"anyColumnSelected": [Function],
|
|
||||||
"availableColumnsLabel": "Available Columns",
|
|
||||||
"canMoveDown": [Function],
|
|
||||||
"canMoveUp": [Function],
|
|
||||||
"canSelectAll": [Function],
|
|
||||||
"columnOptions": [Function],
|
|
||||||
"container": [Circular],
|
|
||||||
"firstFieldHasFocus": [Function],
|
|
||||||
"formErrors": [Function],
|
|
||||||
"formErrorsDetails": [Function],
|
|
||||||
"handleClick": [Function],
|
|
||||||
"id": "tablecolumnoptionspane",
|
|
||||||
"instructionLabel": "Choose the columns and the order in which you want to display them in the table.",
|
|
||||||
"isExecuting": [Function],
|
|
||||||
"isTemplateReady": [Function],
|
|
||||||
"moveDownLabel": "Move Down",
|
|
||||||
"moveUpLabel": "Move Up",
|
|
||||||
"noColumnSelectedWarning": "At least one column should be selected.",
|
|
||||||
"selectedColumnOption": null,
|
|
||||||
"title": [Function],
|
|
||||||
"titleLabel": "Column Options",
|
|
||||||
"visible": [Function],
|
|
||||||
},
|
|
||||||
QuerySelectPane {
|
QuerySelectPane {
|
||||||
"allSelected": [Function],
|
"allSelected": [Function],
|
||||||
"anyColumnSelected": [Function],
|
"anyColumnSelected": [Function],
|
||||||
@@ -2232,7 +2101,6 @@ exports[`SettingsComponent renders 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"flight": [Function],
|
|
||||||
"graphStylingPane": GraphStylingPane {
|
"graphStylingPane": GraphStylingPane {
|
||||||
"container": [Circular],
|
"container": [Circular],
|
||||||
"firstFieldHasFocus": [Function],
|
"firstFieldHasFocus": [Function],
|
||||||
@@ -2254,7 +2122,6 @@ exports[`SettingsComponent renders 1`] = `
|
|||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"hasStorageAnalyticsAfecFeature": [Function],
|
"hasStorageAnalyticsAfecFeature": [Function],
|
||||||
"hasWriteAccess": [Function],
|
|
||||||
"isAccountReady": [Function],
|
"isAccountReady": [Function],
|
||||||
"isAutoscaleDefaultEnabled": [Function],
|
"isAutoscaleDefaultEnabled": [Function],
|
||||||
"isCopyNotebookPaneEnabled": [Function],
|
"isCopyNotebookPaneEnabled": [Function],
|
||||||
@@ -2407,32 +2274,6 @@ exports[`SettingsComponent renders 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"subscriptionType": [Function],
|
|
||||||
"tableColumnOptionsPane": TableColumnOptionsPane {
|
|
||||||
"allSelected": [Function],
|
|
||||||
"anyColumnSelected": [Function],
|
|
||||||
"availableColumnsLabel": "Available Columns",
|
|
||||||
"canMoveDown": [Function],
|
|
||||||
"canMoveUp": [Function],
|
|
||||||
"canSelectAll": [Function],
|
|
||||||
"columnOptions": [Function],
|
|
||||||
"container": [Circular],
|
|
||||||
"firstFieldHasFocus": [Function],
|
|
||||||
"formErrors": [Function],
|
|
||||||
"formErrorsDetails": [Function],
|
|
||||||
"handleClick": [Function],
|
|
||||||
"id": "tablecolumnoptionspane",
|
|
||||||
"instructionLabel": "Choose the columns and the order in which you want to display them in the table.",
|
|
||||||
"isExecuting": [Function],
|
|
||||||
"isTemplateReady": [Function],
|
|
||||||
"moveDownLabel": "Move Down",
|
|
||||||
"moveUpLabel": "Move Up",
|
|
||||||
"noColumnSelectedWarning": "At least one column should be selected.",
|
|
||||||
"selectedColumnOption": null,
|
|
||||||
"title": [Function],
|
|
||||||
"titleLabel": "Column Options",
|
|
||||||
"visible": [Function],
|
|
||||||
},
|
|
||||||
"tabsManager": TabsManager {
|
"tabsManager": TabsManager {
|
||||||
"activeTab": [Function],
|
"activeTab": [Function],
|
||||||
"openedTabs": [Function],
|
"openedTabs": [Function],
|
||||||
@@ -2651,31 +2492,6 @@ exports[`SettingsComponent renders 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
TableColumnOptionsPane {
|
|
||||||
"allSelected": [Function],
|
|
||||||
"anyColumnSelected": [Function],
|
|
||||||
"availableColumnsLabel": "Available Columns",
|
|
||||||
"canMoveDown": [Function],
|
|
||||||
"canMoveUp": [Function],
|
|
||||||
"canSelectAll": [Function],
|
|
||||||
"columnOptions": [Function],
|
|
||||||
"container": [Circular],
|
|
||||||
"firstFieldHasFocus": [Function],
|
|
||||||
"formErrors": [Function],
|
|
||||||
"formErrorsDetails": [Function],
|
|
||||||
"handleClick": [Function],
|
|
||||||
"id": "tablecolumnoptionspane",
|
|
||||||
"instructionLabel": "Choose the columns and the order in which you want to display them in the table.",
|
|
||||||
"isExecuting": [Function],
|
|
||||||
"isTemplateReady": [Function],
|
|
||||||
"moveDownLabel": "Move Down",
|
|
||||||
"moveUpLabel": "Move Up",
|
|
||||||
"noColumnSelectedWarning": "At least one column should be selected.",
|
|
||||||
"selectedColumnOption": null,
|
|
||||||
"title": [Function],
|
|
||||||
"titleLabel": "Column Options",
|
|
||||||
"visible": [Function],
|
|
||||||
},
|
|
||||||
QuerySelectPane {
|
QuerySelectPane {
|
||||||
"allSelected": [Function],
|
"allSelected": [Function],
|
||||||
"anyColumnSelected": [Function],
|
"anyColumnSelected": [Function],
|
||||||
@@ -3038,7 +2854,6 @@ exports[`SettingsComponent renders 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"flight": [Function],
|
|
||||||
"graphStylingPane": GraphStylingPane {
|
"graphStylingPane": GraphStylingPane {
|
||||||
"container": [Circular],
|
"container": [Circular],
|
||||||
"firstFieldHasFocus": [Function],
|
"firstFieldHasFocus": [Function],
|
||||||
@@ -3060,7 +2875,6 @@ exports[`SettingsComponent renders 1`] = `
|
|||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"hasStorageAnalyticsAfecFeature": [Function],
|
"hasStorageAnalyticsAfecFeature": [Function],
|
||||||
"hasWriteAccess": [Function],
|
|
||||||
"isAccountReady": [Function],
|
"isAccountReady": [Function],
|
||||||
"isAutoscaleDefaultEnabled": [Function],
|
"isAutoscaleDefaultEnabled": [Function],
|
||||||
"isCopyNotebookPaneEnabled": [Function],
|
"isCopyNotebookPaneEnabled": [Function],
|
||||||
@@ -3213,32 +3027,6 @@ exports[`SettingsComponent renders 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"subscriptionType": [Function],
|
|
||||||
"tableColumnOptionsPane": TableColumnOptionsPane {
|
|
||||||
"allSelected": [Function],
|
|
||||||
"anyColumnSelected": [Function],
|
|
||||||
"availableColumnsLabel": "Available Columns",
|
|
||||||
"canMoveDown": [Function],
|
|
||||||
"canMoveUp": [Function],
|
|
||||||
"canSelectAll": [Function],
|
|
||||||
"columnOptions": [Function],
|
|
||||||
"container": [Circular],
|
|
||||||
"firstFieldHasFocus": [Function],
|
|
||||||
"formErrors": [Function],
|
|
||||||
"formErrorsDetails": [Function],
|
|
||||||
"handleClick": [Function],
|
|
||||||
"id": "tablecolumnoptionspane",
|
|
||||||
"instructionLabel": "Choose the columns and the order in which you want to display them in the table.",
|
|
||||||
"isExecuting": [Function],
|
|
||||||
"isTemplateReady": [Function],
|
|
||||||
"moveDownLabel": "Move Down",
|
|
||||||
"moveUpLabel": "Move Up",
|
|
||||||
"noColumnSelectedWarning": "At least one column should be selected.",
|
|
||||||
"selectedColumnOption": null,
|
|
||||||
"title": [Function],
|
|
||||||
"titleLabel": "Column Options",
|
|
||||||
"visible": [Function],
|
|
||||||
},
|
|
||||||
"tabsManager": TabsManager {
|
"tabsManager": TabsManager {
|
||||||
"activeTab": [Function],
|
"activeTab": [Function],
|
||||||
"openedTabs": [Function],
|
"openedTabs": [Function],
|
||||||
|
|||||||
@@ -19,13 +19,12 @@ import { Splitter, SplitterBounds, SplitterDirection } from "../Common/Splitter"
|
|||||||
import { configContext, Platform } from "../ConfigContext";
|
import { configContext, Platform } from "../ConfigContext";
|
||||||
import * as DataModels from "../Contracts/DataModels";
|
import * as DataModels from "../Contracts/DataModels";
|
||||||
import { MessageTypes } from "../Contracts/ExplorerContracts";
|
import { MessageTypes } from "../Contracts/ExplorerContracts";
|
||||||
import { SubscriptionType } from "../Contracts/SubscriptionType";
|
|
||||||
import * as ViewModels from "../Contracts/ViewModels";
|
import * as ViewModels from "../Contracts/ViewModels";
|
||||||
import { IGalleryItem } from "../Juno/JunoClient";
|
import { IGalleryItem } from "../Juno/JunoClient";
|
||||||
import { NotebookWorkspaceManager } from "../NotebookWorkspaceManager/NotebookWorkspaceManager";
|
import { NotebookWorkspaceManager } from "../NotebookWorkspaceManager/NotebookWorkspaceManager";
|
||||||
import { ResourceProviderClientFactory } from "../ResourceProvider/ResourceProviderClientFactory";
|
import { ResourceProviderClientFactory } from "../ResourceProvider/ResourceProviderClientFactory";
|
||||||
import { RouteHandler } from "../RouteHandlers/RouteHandler";
|
import { RouteHandler } from "../RouteHandlers/RouteHandler";
|
||||||
import { appInsights } from "../Shared/appInsights";
|
import { trackEvent } from "../Shared/appInsights";
|
||||||
import * as SharedConstants from "../Shared/Constants";
|
import * as SharedConstants from "../Shared/Constants";
|
||||||
import { DefaultExperienceUtility } from "../Shared/DefaultExperienceUtility";
|
import { DefaultExperienceUtility } from "../Shared/DefaultExperienceUtility";
|
||||||
import { ExplorerSettings } from "../Shared/ExplorerSettings";
|
import { ExplorerSettings } from "../Shared/ExplorerSettings";
|
||||||
@@ -67,7 +66,6 @@ import { StringInputPane } from "./Panes/StringInputPane";
|
|||||||
import AddTableEntityPane from "./Panes/Tables/AddTableEntityPane";
|
import AddTableEntityPane from "./Panes/Tables/AddTableEntityPane";
|
||||||
import EditTableEntityPane from "./Panes/Tables/EditTableEntityPane";
|
import EditTableEntityPane from "./Panes/Tables/EditTableEntityPane";
|
||||||
import { QuerySelectPane } from "./Panes/Tables/QuerySelectPane";
|
import { QuerySelectPane } from "./Panes/Tables/QuerySelectPane";
|
||||||
import { TableColumnOptionsPane } from "./Panes/Tables/TableColumnOptionsPane";
|
|
||||||
import { UploadFilePane } from "./Panes/UploadFilePane";
|
import { UploadFilePane } from "./Panes/UploadFilePane";
|
||||||
import { UploadItemsPane } from "./Panes/UploadItemsPane";
|
import { UploadItemsPane } from "./Panes/UploadItemsPane";
|
||||||
import { CassandraAPIDataClient, TableDataClient, TablesAPIDataClient } from "./Tables/TableDataClient";
|
import { CassandraAPIDataClient, TableDataClient, TablesAPIDataClient } from "./Tables/TableDataClient";
|
||||||
@@ -98,10 +96,6 @@ export interface ExplorerParams {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default class Explorer {
|
export default class Explorer {
|
||||||
public flight: ko.Observable<string> = ko.observable<string>(
|
|
||||||
SharedConstants.CollectionCreation.DefaultAddCollectionDefaultFlight
|
|
||||||
);
|
|
||||||
|
|
||||||
public addCollectionText: ko.Observable<string>;
|
public addCollectionText: ko.Observable<string>;
|
||||||
public addDatabaseText: ko.Observable<string>;
|
public addDatabaseText: ko.Observable<string>;
|
||||||
public collectionTitle: ko.Observable<string>;
|
public collectionTitle: ko.Observable<string>;
|
||||||
@@ -109,7 +103,6 @@ export default class Explorer {
|
|||||||
public deleteDatabaseText: ko.Observable<string>;
|
public deleteDatabaseText: ko.Observable<string>;
|
||||||
public collectionTreeNodeAltText: ko.Observable<string>;
|
public collectionTreeNodeAltText: ko.Observable<string>;
|
||||||
public refreshTreeTitle: ko.Observable<string>;
|
public refreshTreeTitle: ko.Observable<string>;
|
||||||
public hasWriteAccess: ko.Observable<boolean>;
|
|
||||||
public collapsedResourceTreeWidth: number = ExplorerMetrics.CollapsedResourceTreeWidth;
|
public collapsedResourceTreeWidth: number = ExplorerMetrics.CollapsedResourceTreeWidth;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -118,11 +111,6 @@ export default class Explorer {
|
|||||||
* */
|
* */
|
||||||
public databaseAccount: ko.Observable<DataModels.DatabaseAccount>;
|
public databaseAccount: ko.Observable<DataModels.DatabaseAccount>;
|
||||||
public collectionCreationDefaults: ViewModels.CollectionCreationDefaults = SharedConstants.CollectionCreationDefaults;
|
public collectionCreationDefaults: ViewModels.CollectionCreationDefaults = SharedConstants.CollectionCreationDefaults;
|
||||||
/**
|
|
||||||
* @deprecated
|
|
||||||
* Use userContext.subscriptionType instead
|
|
||||||
* */
|
|
||||||
public subscriptionType: ko.Observable<SubscriptionType>;
|
|
||||||
/**
|
/**
|
||||||
* @deprecated
|
* @deprecated
|
||||||
* Use userContext.apiType instead
|
* Use userContext.apiType instead
|
||||||
@@ -205,7 +193,6 @@ export default class Explorer {
|
|||||||
public graphStylingPane: GraphStylingPane;
|
public graphStylingPane: GraphStylingPane;
|
||||||
public addTableEntityPane: AddTableEntityPane;
|
public addTableEntityPane: AddTableEntityPane;
|
||||||
public editTableEntityPane: EditTableEntityPane;
|
public editTableEntityPane: EditTableEntityPane;
|
||||||
public tableColumnOptionsPane: TableColumnOptionsPane;
|
|
||||||
public querySelectPane: QuerySelectPane;
|
public querySelectPane: QuerySelectPane;
|
||||||
public newVertexPane: NewVertexPane;
|
public newVertexPane: NewVertexPane;
|
||||||
public cassandraAddCollectionPane: CassandraAddCollectionPane;
|
public cassandraAddCollectionPane: CassandraAddCollectionPane;
|
||||||
@@ -272,7 +259,6 @@ export default class Explorer {
|
|||||||
});
|
});
|
||||||
this.addCollectionText = ko.observable<string>("New Collection");
|
this.addCollectionText = ko.observable<string>("New Collection");
|
||||||
this.addDatabaseText = ko.observable<string>("New Database");
|
this.addDatabaseText = ko.observable<string>("New Database");
|
||||||
this.hasWriteAccess = ko.observable<boolean>(true);
|
|
||||||
this.collectionTitle = ko.observable<string>("Collections");
|
this.collectionTitle = ko.observable<string>("Collections");
|
||||||
this.collectionTreeNodeAltText = ko.observable<string>("Collection");
|
this.collectionTreeNodeAltText = ko.observable<string>("Collection");
|
||||||
this.deleteCollectionText = ko.observable<string>("Delete Collection");
|
this.deleteCollectionText = ko.observable<string>("Delete Collection");
|
||||||
@@ -280,7 +266,6 @@ export default class Explorer {
|
|||||||
this.refreshTreeTitle = ko.observable<string>("Refresh collections");
|
this.refreshTreeTitle = ko.observable<string>("Refresh collections");
|
||||||
|
|
||||||
this.databaseAccount = ko.observable<DataModels.DatabaseAccount>();
|
this.databaseAccount = ko.observable<DataModels.DatabaseAccount>();
|
||||||
this.subscriptionType = ko.observable<SubscriptionType>(SharedConstants.CollectionCreation.DefaultSubscriptionType);
|
|
||||||
this.isAccountReady = ko.observable<boolean>(false);
|
this.isAccountReady = ko.observable<boolean>(false);
|
||||||
this._isInitializingNotebooks = false;
|
this._isInitializingNotebooks = false;
|
||||||
this.arcadiaToken = ko.observable<string>();
|
this.arcadiaToken = ko.observable<string>();
|
||||||
@@ -341,7 +326,7 @@ export default class Explorer {
|
|||||||
userContext.features.enableSpark
|
userContext.features.enableSpark
|
||||||
);
|
);
|
||||||
if (this.isSparkEnabled()) {
|
if (this.isSparkEnabled()) {
|
||||||
appInsights.trackEvent(
|
trackEvent(
|
||||||
{ name: "LoadedWithSparkEnabled" },
|
{ name: "LoadedWithSparkEnabled" },
|
||||||
{
|
{
|
||||||
subscriptionId: userContext.subscriptionId,
|
subscriptionId: userContext.subscriptionId,
|
||||||
@@ -572,13 +557,6 @@ export default class Explorer {
|
|||||||
container: this,
|
container: this,
|
||||||
});
|
});
|
||||||
|
|
||||||
this.tableColumnOptionsPane = new TableColumnOptionsPane({
|
|
||||||
id: "tablecolumnoptionspane",
|
|
||||||
visible: ko.observable<boolean>(false),
|
|
||||||
|
|
||||||
container: this,
|
|
||||||
});
|
|
||||||
|
|
||||||
this.querySelectPane = new QuerySelectPane({
|
this.querySelectPane = new QuerySelectPane({
|
||||||
id: "queryselectpane",
|
id: "queryselectpane",
|
||||||
visible: ko.observable<boolean>(false),
|
visible: ko.observable<boolean>(false),
|
||||||
@@ -622,7 +600,6 @@ export default class Explorer {
|
|||||||
this.graphStylingPane,
|
this.graphStylingPane,
|
||||||
this.addTableEntityPane,
|
this.addTableEntityPane,
|
||||||
this.editTableEntityPane,
|
this.editTableEntityPane,
|
||||||
this.tableColumnOptionsPane,
|
|
||||||
this.querySelectPane,
|
this.querySelectPane,
|
||||||
this.newVertexPane,
|
this.newVertexPane,
|
||||||
this.cassandraAddCollectionPane,
|
this.cassandraAddCollectionPane,
|
||||||
@@ -1283,11 +1260,6 @@ export default class Explorer {
|
|||||||
this.collectionCreationDefaults = inputs.defaultCollectionThroughput;
|
this.collectionCreationDefaults = inputs.defaultCollectionThroughput;
|
||||||
}
|
}
|
||||||
this.databaseAccount(databaseAccount);
|
this.databaseAccount(databaseAccount);
|
||||||
this.subscriptionType(inputs.subscriptionType ?? SharedConstants.CollectionCreation.DefaultSubscriptionType);
|
|
||||||
this.hasWriteAccess(inputs.hasWriteAccess ?? true);
|
|
||||||
if (inputs.addCollectionDefaultFlight) {
|
|
||||||
this.flight(inputs.addCollectionDefaultFlight);
|
|
||||||
}
|
|
||||||
this.setFeatureFlagsFromFlights(inputs.flights);
|
this.setFeatureFlagsFromFlights(inputs.flights);
|
||||||
TelemetryProcessor.traceSuccess(
|
TelemetryProcessor.traceSuccess(
|
||||||
Action.LoadDatabaseAccount,
|
Action.LoadDatabaseAccount,
|
||||||
|
|||||||
@@ -105,10 +105,6 @@ export default class AddCollectionPane extends ContextualPaneBase {
|
|||||||
this.databaseId = ko.observable<string>();
|
this.databaseId = ko.observable<string>();
|
||||||
this.databaseCreateNew = ko.observable<boolean>(true);
|
this.databaseCreateNew = ko.observable<boolean>(true);
|
||||||
this.databaseCreateNewShared = ko.observable<boolean>(this.getSharedThroughputDefault());
|
this.databaseCreateNewShared = ko.observable<boolean>(this.getSharedThroughputDefault());
|
||||||
this.container.subscriptionType &&
|
|
||||||
this.container.subscriptionType.subscribe((subscriptionType) => {
|
|
||||||
this.databaseCreateNewShared(this.getSharedThroughputDefault());
|
|
||||||
});
|
|
||||||
this.collectionWithThroughputInShared = ko.observable<boolean>(false);
|
this.collectionWithThroughputInShared = ko.observable<boolean>(false);
|
||||||
this.databaseIds = ko.observableArray<string>();
|
this.databaseIds = ko.observableArray<string>();
|
||||||
this.uniqueKeys = ko.observableArray<DynamicListItem>();
|
this.uniqueKeys = ko.observableArray<DynamicListItem>();
|
||||||
@@ -478,9 +474,6 @@ export default class AddCollectionPane extends ContextualPaneBase {
|
|||||||
});
|
});
|
||||||
|
|
||||||
this.resetData();
|
this.resetData();
|
||||||
this.container.flight.subscribe(() => {
|
|
||||||
this.resetData();
|
|
||||||
});
|
|
||||||
|
|
||||||
this.freeTierExceedThroughputTooltip = ko.pureComputed<string>(() =>
|
this.freeTierExceedThroughputTooltip = ko.pureComputed<string>(() =>
|
||||||
this.isFreeTierAccount() && !this.container.isFirstResourceCreated()
|
this.isFreeTierAccount() && !this.container.isFirstResourceCreated()
|
||||||
@@ -659,7 +652,7 @@ export default class AddCollectionPane extends ContextualPaneBase {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public getSharedThroughputDefault(): boolean {
|
public getSharedThroughputDefault(): boolean {
|
||||||
const subscriptionType = this.container.subscriptionType && this.container.subscriptionType();
|
const subscriptionType = userContext.subscriptionType;
|
||||||
if (subscriptionType === SubscriptionType.EA || this.container.isServerlessEnabled()) {
|
if (subscriptionType === SubscriptionType.EA || this.container.isServerlessEnabled()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -701,12 +694,12 @@ export default class AddCollectionPane extends ContextualPaneBase {
|
|||||||
partitionKey: this.partitionKey(),
|
partitionKey: this.partitionKey(),
|
||||||
databaseId: this.databaseId(),
|
databaseId: this.databaseId(),
|
||||||
}),
|
}),
|
||||||
subscriptionType: SubscriptionType[this.container.subscriptionType()],
|
subscriptionType: userContext.subscriptionType,
|
||||||
subscriptionQuotaId: userContext.quotaId,
|
subscriptionQuotaId: userContext.quotaId,
|
||||||
defaultsCheck: {
|
defaultsCheck: {
|
||||||
storage: this.storage() === Constants.BackendDefaults.singlePartitionStorageInGb ? "f" : "u",
|
storage: this.storage() === Constants.BackendDefaults.singlePartitionStorageInGb ? "f" : "u",
|
||||||
throughput: this._getThroughput(),
|
throughput: this._getThroughput(),
|
||||||
flight: this.container.flight(),
|
flight: userContext.addCollectionFlight,
|
||||||
},
|
},
|
||||||
dataExplorerArea: Constants.Areas.ContextualPane,
|
dataExplorerArea: Constants.Areas.ContextualPane,
|
||||||
};
|
};
|
||||||
@@ -805,12 +798,12 @@ export default class AddCollectionPane extends ContextualPaneBase {
|
|||||||
uniqueKeyPolicy,
|
uniqueKeyPolicy,
|
||||||
collectionWithThroughputInShared: this.collectionWithThroughputInShared(),
|
collectionWithThroughputInShared: this.collectionWithThroughputInShared(),
|
||||||
}),
|
}),
|
||||||
subscriptionType: SubscriptionType[this.container.subscriptionType()],
|
subscriptionType: userContext.subscriptionType,
|
||||||
subscriptionQuotaId: userContext.quotaId,
|
subscriptionQuotaId: userContext.quotaId,
|
||||||
defaultsCheck: {
|
defaultsCheck: {
|
||||||
storage: this.storage() === Constants.BackendDefaults.singlePartitionStorageInGb ? "f" : "u",
|
storage: this.storage() === Constants.BackendDefaults.singlePartitionStorageInGb ? "f" : "u",
|
||||||
throughput: offerThroughput,
|
throughput: offerThroughput,
|
||||||
flight: this.container.flight(),
|
flight: userContext.addCollectionFlight,
|
||||||
},
|
},
|
||||||
dataExplorerArea: Constants.Areas.ContextualPane,
|
dataExplorerArea: Constants.Areas.ContextualPane,
|
||||||
useIndexingForSharedThroughput: this.useIndexingForSharedThroughput(),
|
useIndexingForSharedThroughput: this.useIndexingForSharedThroughput(),
|
||||||
@@ -877,12 +870,12 @@ export default class AddCollectionPane extends ContextualPaneBase {
|
|||||||
uniqueKeyPolicy,
|
uniqueKeyPolicy,
|
||||||
collectionWithThroughputInShared: this.collectionWithThroughputInShared(),
|
collectionWithThroughputInShared: this.collectionWithThroughputInShared(),
|
||||||
}),
|
}),
|
||||||
subscriptionType: SubscriptionType[this.container.subscriptionType()],
|
subscriptionType: userContext.subscriptionType,
|
||||||
subscriptionQuotaId: userContext.quotaId,
|
subscriptionQuotaId: userContext.quotaId,
|
||||||
defaultsCheck: {
|
defaultsCheck: {
|
||||||
storage: this.storage() === Constants.BackendDefaults.singlePartitionStorageInGb ? "f" : "u",
|
storage: this.storage() === Constants.BackendDefaults.singlePartitionStorageInGb ? "f" : "u",
|
||||||
throughput: offerThroughput,
|
throughput: offerThroughput,
|
||||||
flight: this.container.flight(),
|
flight: userContext.addCollectionFlight,
|
||||||
},
|
},
|
||||||
dataExplorerArea: Constants.Areas.ContextualPane,
|
dataExplorerArea: Constants.Areas.ContextualPane,
|
||||||
};
|
};
|
||||||
@@ -909,12 +902,12 @@ export default class AddCollectionPane extends ContextualPaneBase {
|
|||||||
uniqueKeyPolicy,
|
uniqueKeyPolicy,
|
||||||
collectionWithThroughputInShared: this.collectionWithThroughputInShared(),
|
collectionWithThroughputInShared: this.collectionWithThroughputInShared(),
|
||||||
},
|
},
|
||||||
subscriptionType: SubscriptionType[this.container.subscriptionType()],
|
subscriptionType: userContext.subscriptionType,
|
||||||
subscriptionQuotaId: userContext.quotaId,
|
subscriptionQuotaId: userContext.quotaId,
|
||||||
defaultsCheck: {
|
defaultsCheck: {
|
||||||
storage: this.storage() === Constants.BackendDefaults.singlePartitionStorageInGb ? "f" : "u",
|
storage: this.storage() === Constants.BackendDefaults.singlePartitionStorageInGb ? "f" : "u",
|
||||||
throughput: offerThroughput,
|
throughput: offerThroughput,
|
||||||
flight: this.container.flight(),
|
flight: userContext.addCollectionFlight,
|
||||||
},
|
},
|
||||||
dataExplorerArea: Constants.Areas.ContextualPane,
|
dataExplorerArea: Constants.Areas.ContextualPane,
|
||||||
error: errorMessage,
|
error: errorMessage,
|
||||||
|
|||||||
105
src/Explorer/Panes/AddDatabasePane.test.ts
Normal file
105
src/Explorer/Panes/AddDatabasePane.test.ts
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
import * as Constants from "../../Common/Constants";
|
||||||
|
import { DatabaseAccount } from "../../Contracts/DataModels";
|
||||||
|
import { SubscriptionType } from "../../Contracts/SubscriptionType";
|
||||||
|
import { updateUserContext } from "../../UserContext";
|
||||||
|
import Explorer from "../Explorer";
|
||||||
|
import AddDatabasePane from "./AddDatabasePane";
|
||||||
|
|
||||||
|
describe("Add Database Pane", () => {
|
||||||
|
describe("getSharedThroughputDefault()", () => {
|
||||||
|
let explorer: Explorer;
|
||||||
|
const mockDatabaseAccount: DatabaseAccount = {
|
||||||
|
id: "mock",
|
||||||
|
kind: "DocumentDB",
|
||||||
|
location: "",
|
||||||
|
name: "mock",
|
||||||
|
properties: {
|
||||||
|
documentEndpoint: "",
|
||||||
|
cassandraEndpoint: "",
|
||||||
|
gremlinEndpoint: "",
|
||||||
|
tableEndpoint: "",
|
||||||
|
enableFreeTier: false,
|
||||||
|
},
|
||||||
|
type: undefined,
|
||||||
|
tags: [],
|
||||||
|
};
|
||||||
|
|
||||||
|
const mockFreeTierDatabaseAccount: DatabaseAccount = {
|
||||||
|
id: "mock",
|
||||||
|
kind: "DocumentDB",
|
||||||
|
location: "",
|
||||||
|
name: "mock",
|
||||||
|
properties: {
|
||||||
|
documentEndpoint: "",
|
||||||
|
cassandraEndpoint: "",
|
||||||
|
gremlinEndpoint: "",
|
||||||
|
tableEndpoint: "",
|
||||||
|
enableFreeTier: true,
|
||||||
|
},
|
||||||
|
type: undefined,
|
||||||
|
tags: [],
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
explorer = new Explorer();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be true if subscription type is Benefits", () => {
|
||||||
|
updateUserContext({
|
||||||
|
subscriptionType: SubscriptionType.Benefits,
|
||||||
|
});
|
||||||
|
const addDatabasePane = explorer.addDatabasePane as AddDatabasePane;
|
||||||
|
expect(addDatabasePane.getSharedThroughputDefault()).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be false if subscription type is EA", () => {
|
||||||
|
updateUserContext({
|
||||||
|
subscriptionType: SubscriptionType.EA,
|
||||||
|
});
|
||||||
|
const addDatabasePane = explorer.addDatabasePane as AddDatabasePane;
|
||||||
|
expect(addDatabasePane.getSharedThroughputDefault()).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be true if subscription type is Free", () => {
|
||||||
|
updateUserContext({
|
||||||
|
subscriptionType: SubscriptionType.Free,
|
||||||
|
});
|
||||||
|
const addDatabasePane = explorer.addDatabasePane as AddDatabasePane;
|
||||||
|
expect(addDatabasePane.getSharedThroughputDefault()).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be true if subscription type is Internal", () => {
|
||||||
|
updateUserContext({
|
||||||
|
subscriptionType: SubscriptionType.Internal,
|
||||||
|
});
|
||||||
|
const addDatabasePane = explorer.addDatabasePane as AddDatabasePane;
|
||||||
|
expect(addDatabasePane.getSharedThroughputDefault()).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be true if subscription type is PAYG", () => {
|
||||||
|
updateUserContext({
|
||||||
|
subscriptionType: SubscriptionType.PAYG,
|
||||||
|
});
|
||||||
|
const addDatabasePane = explorer.addDatabasePane as AddDatabasePane;
|
||||||
|
expect(addDatabasePane.getSharedThroughputDefault()).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should display free tier text in upsell messaging", () => {
|
||||||
|
explorer.databaseAccount(mockFreeTierDatabaseAccount);
|
||||||
|
const addDatabasePane = explorer.addDatabasePane as AddDatabasePane;
|
||||||
|
expect(addDatabasePane.isFreeTierAccount()).toBe(true);
|
||||||
|
expect(addDatabasePane.upsellMessage()).toContain("With free tier");
|
||||||
|
expect(addDatabasePane.upsellAnchorUrl()).toBe(Constants.Urls.freeTierInformation);
|
||||||
|
expect(addDatabasePane.upsellAnchorText()).toBe("Learn more");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should display standard texr in upsell messaging", () => {
|
||||||
|
explorer.databaseAccount(mockDatabaseAccount);
|
||||||
|
const addDatabasePane = explorer.addDatabasePane as AddDatabasePane;
|
||||||
|
expect(addDatabasePane.isFreeTierAccount()).toBe(false);
|
||||||
|
expect(addDatabasePane.upsellMessage()).toContain("Start at");
|
||||||
|
expect(addDatabasePane.upsellAnchorUrl()).toBe(Constants.Urls.cosmosPricing);
|
||||||
|
expect(addDatabasePane.upsellAnchorText()).toBe("More details");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
461
src/Explorer/Panes/AddDatabasePane.ts
Normal file
461
src/Explorer/Panes/AddDatabasePane.ts
Normal file
@@ -0,0 +1,461 @@
|
|||||||
|
import * as ko from "knockout";
|
||||||
|
import * as Constants from "../../Common/Constants";
|
||||||
|
import { createDatabase } from "../../Common/dataAccess/createDatabase";
|
||||||
|
import editable from "../../Common/EditableUtility";
|
||||||
|
import { getErrorMessage, getErrorStack } from "../../Common/ErrorHandlingUtils";
|
||||||
|
import { configContext, Platform } from "../../ConfigContext";
|
||||||
|
import * as DataModels from "../../Contracts/DataModels";
|
||||||
|
import { SubscriptionType } from "../../Contracts/SubscriptionType";
|
||||||
|
import * as ViewModels from "../../Contracts/ViewModels";
|
||||||
|
import * as SharedConstants from "../../Shared/Constants";
|
||||||
|
import { Action, ActionModifiers } from "../../Shared/Telemetry/TelemetryConstants";
|
||||||
|
import * as TelemetryProcessor from "../../Shared/Telemetry/TelemetryProcessor";
|
||||||
|
import { userContext } from "../../UserContext";
|
||||||
|
import * as AutoPilotUtils from "../../Utils/AutoPilotUtils";
|
||||||
|
import * as PricingUtils from "../../Utils/PricingUtils";
|
||||||
|
import { ContextualPaneBase } from "./ContextualPaneBase";
|
||||||
|
|
||||||
|
export default class AddDatabasePane extends ContextualPaneBase {
|
||||||
|
public defaultExperience: ko.Computed<string>;
|
||||||
|
public databaseIdLabel: ko.Computed<string>;
|
||||||
|
public databaseIdPlaceHolder: ko.Computed<string>;
|
||||||
|
public databaseId: ko.Observable<string>;
|
||||||
|
public databaseIdTooltipText: ko.Computed<string>;
|
||||||
|
public databaseLevelThroughputTooltipText: ko.Computed<string>;
|
||||||
|
public databaseCreateNewShared: ko.Observable<boolean>;
|
||||||
|
public formErrorsDetails: ko.Observable<string>;
|
||||||
|
public throughput: ViewModels.Editable<number>;
|
||||||
|
public maxThroughputRU: ko.Observable<number>;
|
||||||
|
public minThroughputRU: ko.Observable<number>;
|
||||||
|
public maxThroughputRUText: ko.PureComputed<string>;
|
||||||
|
public throughputRangeText: ko.Computed<string>;
|
||||||
|
public throughputSpendAckText: ko.Observable<string>;
|
||||||
|
public throughputSpendAck: ko.Observable<boolean>;
|
||||||
|
public throughputSpendAckVisible: ko.Computed<boolean>;
|
||||||
|
public requestUnitsUsageCost: ko.Computed<string>;
|
||||||
|
public canRequestSupport: ko.PureComputed<boolean>;
|
||||||
|
public costsVisible: ko.PureComputed<boolean>;
|
||||||
|
public upsellMessage: ko.PureComputed<string>;
|
||||||
|
public upsellMessageAriaLabel: ko.PureComputed<string>;
|
||||||
|
public upsellAnchorUrl: ko.PureComputed<string>;
|
||||||
|
public upsellAnchorText: ko.PureComputed<string>;
|
||||||
|
public isAutoPilotSelected: ko.Observable<boolean>;
|
||||||
|
public maxAutoPilotThroughputSet: ko.Observable<number>;
|
||||||
|
public autoPilotUsageCost: ko.Computed<string>;
|
||||||
|
public canExceedMaximumValue: ko.PureComputed<boolean>;
|
||||||
|
public ruToolTipText: ko.Computed<string>;
|
||||||
|
public freeTierExceedThroughputTooltip: ko.Computed<string>;
|
||||||
|
public isFreeTierAccount: ko.Computed<boolean>;
|
||||||
|
public canConfigureThroughput: ko.PureComputed<boolean>;
|
||||||
|
public showUpsellMessage: ko.PureComputed<boolean>;
|
||||||
|
|
||||||
|
constructor(options: ViewModels.PaneOptions) {
|
||||||
|
super(options);
|
||||||
|
this.title((this.container && this.container.addDatabaseText()) || "New Database");
|
||||||
|
this.databaseId = ko.observable<string>();
|
||||||
|
this.ruToolTipText = ko.pureComputed(() => PricingUtils.getRuToolTipText());
|
||||||
|
this.canConfigureThroughput = ko.pureComputed(() => !this.container.isServerlessEnabled());
|
||||||
|
|
||||||
|
this.canExceedMaximumValue = ko.pureComputed(() => this.container.canExceedMaximumValue());
|
||||||
|
|
||||||
|
// TODO 388844: get defaults from parent frame
|
||||||
|
this.databaseCreateNewShared = ko.observable<boolean>(this.getSharedThroughputDefault());
|
||||||
|
|
||||||
|
this.databaseIdLabel = ko.computed<string>(() =>
|
||||||
|
this.container.isPreferredApiCassandra() ? "Keyspace id" : "Database id"
|
||||||
|
);
|
||||||
|
|
||||||
|
this.databaseIdPlaceHolder = ko.computed<string>(() =>
|
||||||
|
this.container.isPreferredApiCassandra() ? "Type a new keyspace id" : "Type a new database id"
|
||||||
|
);
|
||||||
|
|
||||||
|
this.databaseIdTooltipText = ko.computed<string>(() => {
|
||||||
|
const isCassandraAccount: boolean = this.container.isPreferredApiCassandra();
|
||||||
|
return `A ${isCassandraAccount ? "keyspace" : "database"} is a logical container of one or more ${
|
||||||
|
isCassandraAccount ? "tables" : "collections"
|
||||||
|
}`;
|
||||||
|
});
|
||||||
|
this.databaseLevelThroughputTooltipText = ko.computed<string>(() => {
|
||||||
|
const isCassandraAccount: boolean = this.container.isPreferredApiCassandra();
|
||||||
|
const databaseLabel: string = isCassandraAccount ? "keyspace" : "database";
|
||||||
|
const collectionsLabel: string = isCassandraAccount ? "tables" : "collections";
|
||||||
|
return `Provisioned throughput at the ${databaseLabel} level will be shared across all ${collectionsLabel} within the ${databaseLabel}.`;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.throughput = editable.observable<number>();
|
||||||
|
this.maxThroughputRU = ko.observable<number>();
|
||||||
|
this.minThroughputRU = ko.observable<number>();
|
||||||
|
this.throughputSpendAckText = ko.observable<string>();
|
||||||
|
this.throughputSpendAck = ko.observable<boolean>(false);
|
||||||
|
this.isAutoPilotSelected = ko.observable<boolean>(false);
|
||||||
|
this.maxAutoPilotThroughputSet = ko.observable<number>(AutoPilotUtils.minAutoPilotThroughput);
|
||||||
|
this.autoPilotUsageCost = ko.pureComputed<string>(() => {
|
||||||
|
const autoPilot = this._isAutoPilotSelectedAndWhatTier();
|
||||||
|
if (!autoPilot) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
return PricingUtils.getAutoPilotV3SpendHtml(autoPilot.maxThroughput, true /* isDatabaseThroughput */);
|
||||||
|
});
|
||||||
|
this.throughputRangeText = ko.pureComputed<string>(() => {
|
||||||
|
if (this.isAutoPilotSelected()) {
|
||||||
|
return AutoPilotUtils.getAutoPilotHeaderText();
|
||||||
|
}
|
||||||
|
return `Throughput (${this.minThroughputRU().toLocaleString()} - ${this.maxThroughputRU().toLocaleString()} RU/s)`;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.requestUnitsUsageCost = ko.computed(() => {
|
||||||
|
const offerThroughput: number = this.throughput();
|
||||||
|
if (
|
||||||
|
offerThroughput < this.minThroughputRU() ||
|
||||||
|
(offerThroughput > this.maxThroughputRU() && !this.canExceedMaximumValue())
|
||||||
|
) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
const account = this.container.databaseAccount();
|
||||||
|
if (!account) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
const regions =
|
||||||
|
(account &&
|
||||||
|
account.properties &&
|
||||||
|
account.properties.readLocations &&
|
||||||
|
account.properties.readLocations.length) ||
|
||||||
|
1;
|
||||||
|
const multimaster = (account && account.properties && account.properties.enableMultipleWriteLocations) || false;
|
||||||
|
|
||||||
|
let estimatedSpendAcknowledge: string;
|
||||||
|
let estimatedSpend: string;
|
||||||
|
if (!this.isAutoPilotSelected()) {
|
||||||
|
estimatedSpend = PricingUtils.getEstimatedSpendHtml(
|
||||||
|
offerThroughput,
|
||||||
|
userContext.portalEnv,
|
||||||
|
regions,
|
||||||
|
multimaster
|
||||||
|
);
|
||||||
|
estimatedSpendAcknowledge = PricingUtils.getEstimatedSpendAcknowledgeString(
|
||||||
|
offerThroughput,
|
||||||
|
userContext.portalEnv,
|
||||||
|
regions,
|
||||||
|
multimaster,
|
||||||
|
this.isAutoPilotSelected()
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
estimatedSpend = PricingUtils.getEstimatedAutoscaleSpendHtml(
|
||||||
|
this.maxAutoPilotThroughputSet(),
|
||||||
|
userContext.portalEnv,
|
||||||
|
regions,
|
||||||
|
multimaster
|
||||||
|
);
|
||||||
|
estimatedSpendAcknowledge = PricingUtils.getEstimatedSpendAcknowledgeString(
|
||||||
|
this.maxAutoPilotThroughputSet(),
|
||||||
|
userContext.portalEnv,
|
||||||
|
regions,
|
||||||
|
multimaster,
|
||||||
|
this.isAutoPilotSelected()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
// TODO: change throughputSpendAckText to be a computed value, instead of having this side effect
|
||||||
|
this.throughputSpendAckText(estimatedSpendAcknowledge);
|
||||||
|
return estimatedSpend;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.canRequestSupport = ko.pureComputed(() => {
|
||||||
|
if (
|
||||||
|
configContext.platform !== Platform.Emulator &&
|
||||||
|
!userContext.isTryCosmosDBSubscription &&
|
||||||
|
configContext.platform !== Platform.Portal
|
||||||
|
) {
|
||||||
|
const offerThroughput: number = this.throughput();
|
||||||
|
return offerThroughput <= 100000;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.isFreeTierAccount = ko.computed<boolean>(() => {
|
||||||
|
const databaseAccount = this.container && this.container.databaseAccount && this.container.databaseAccount();
|
||||||
|
const isFreeTierAccount =
|
||||||
|
databaseAccount && databaseAccount.properties && databaseAccount.properties.enableFreeTier;
|
||||||
|
return isFreeTierAccount;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.showUpsellMessage = ko.pureComputed(() => {
|
||||||
|
if (this.container.isServerlessEnabled()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.isFreeTierAccount()) {
|
||||||
|
return this.databaseCreateNewShared();
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.maxThroughputRUText = ko.pureComputed(() => {
|
||||||
|
return this.maxThroughputRU().toLocaleString();
|
||||||
|
});
|
||||||
|
|
||||||
|
this.costsVisible = ko.pureComputed(() => {
|
||||||
|
return configContext.platform !== Platform.Emulator;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.throughputSpendAckVisible = ko.pureComputed<boolean>(() => {
|
||||||
|
const autoscaleThroughput = this.maxAutoPilotThroughputSet() * 1;
|
||||||
|
if (this.isAutoPilotSelected()) {
|
||||||
|
return autoscaleThroughput > SharedConstants.CollectionCreation.DefaultCollectionRUs100K;
|
||||||
|
}
|
||||||
|
|
||||||
|
const selectedThroughput: number = this.throughput();
|
||||||
|
const maxRU: number = this.maxThroughputRU && this.maxThroughputRU();
|
||||||
|
|
||||||
|
const isMaxRUGreaterThanDefault: boolean = maxRU > SharedConstants.CollectionCreation.DefaultCollectionRUs100K;
|
||||||
|
const isThroughputSetGreaterThanDefault: boolean =
|
||||||
|
selectedThroughput > SharedConstants.CollectionCreation.DefaultCollectionRUs100K;
|
||||||
|
|
||||||
|
if (this.canExceedMaximumValue()) {
|
||||||
|
return isThroughputSetGreaterThanDefault;
|
||||||
|
}
|
||||||
|
|
||||||
|
return isThroughputSetGreaterThanDefault && isMaxRUGreaterThanDefault;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.databaseCreateNewShared.subscribe((useShared: boolean) => {
|
||||||
|
this._updateThroughputLimitByDatabase();
|
||||||
|
});
|
||||||
|
|
||||||
|
this.resetData();
|
||||||
|
|
||||||
|
this.freeTierExceedThroughputTooltip = ko.pureComputed<string>(() =>
|
||||||
|
this.isFreeTierAccount() && !this.container.isFirstResourceCreated()
|
||||||
|
? "The first 400 RU/s in this account are free. Billing will apply to any throughput beyond 400 RU/s."
|
||||||
|
: ""
|
||||||
|
);
|
||||||
|
|
||||||
|
this.upsellMessage = ko.pureComputed<string>(() => {
|
||||||
|
return PricingUtils.getUpsellMessage(
|
||||||
|
userContext.portalEnv,
|
||||||
|
this.isFreeTierAccount(),
|
||||||
|
this.container.isFirstResourceCreated(),
|
||||||
|
this.container.defaultExperience(),
|
||||||
|
false
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
this.upsellMessageAriaLabel = ko.pureComputed<string>(() => {
|
||||||
|
return `${this.upsellMessage()}. Click ${this.isFreeTierAccount() ? "to learn more" : "for more details"}`;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.upsellAnchorUrl = ko.pureComputed<string>(() => {
|
||||||
|
return this.isFreeTierAccount() ? Constants.Urls.freeTierInformation : Constants.Urls.cosmosPricing;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.upsellAnchorText = ko.pureComputed<string>(() => {
|
||||||
|
return this.isFreeTierAccount() ? "Learn more" : "More details";
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public onMoreDetailsKeyPress = (source: any, event: KeyboardEvent): boolean => {
|
||||||
|
if (event.keyCode === Constants.KeyCodes.Space || event.keyCode === Constants.KeyCodes.Enter) {
|
||||||
|
this.showErrorDetails();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|
||||||
|
public open() {
|
||||||
|
super.open();
|
||||||
|
this.resetData();
|
||||||
|
const addDatabasePaneOpenMessage = {
|
||||||
|
subscriptionType: userContext.subscriptionType,
|
||||||
|
subscriptionQuotaId: userContext.quotaId,
|
||||||
|
defaultsCheck: {
|
||||||
|
throughput: this.throughput(),
|
||||||
|
flight: userContext.addCollectionFlight,
|
||||||
|
},
|
||||||
|
dataExplorerArea: Constants.Areas.ContextualPane,
|
||||||
|
};
|
||||||
|
const focusElement = document.getElementById("database-id");
|
||||||
|
focusElement && focusElement.focus();
|
||||||
|
TelemetryProcessor.trace(Action.CreateDatabase, ActionModifiers.Open, addDatabasePaneOpenMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
public submit() {
|
||||||
|
if (!this._isValid()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const offerThroughput: number = this._computeOfferThroughput();
|
||||||
|
|
||||||
|
const addDatabasePaneStartMessage = {
|
||||||
|
database: ko.toJS({
|
||||||
|
id: this.databaseId(),
|
||||||
|
shared: this.databaseCreateNewShared(),
|
||||||
|
}),
|
||||||
|
offerThroughput,
|
||||||
|
subscriptionType: userContext.subscriptionType,
|
||||||
|
subscriptionQuotaId: userContext.quotaId,
|
||||||
|
defaultsCheck: {
|
||||||
|
flight: userContext.addCollectionFlight,
|
||||||
|
},
|
||||||
|
dataExplorerArea: Constants.Areas.ContextualPane,
|
||||||
|
};
|
||||||
|
const startKey: number = TelemetryProcessor.traceStart(Action.CreateDatabase, addDatabasePaneStartMessage);
|
||||||
|
this.formErrors("");
|
||||||
|
this.isExecuting(true);
|
||||||
|
|
||||||
|
const createDatabaseParams: DataModels.CreateDatabaseParams = {
|
||||||
|
databaseId: addDatabasePaneStartMessage.database.id,
|
||||||
|
databaseLevelThroughput: addDatabasePaneStartMessage.database.shared,
|
||||||
|
};
|
||||||
|
|
||||||
|
if (this.isAutoPilotSelected()) {
|
||||||
|
createDatabaseParams.autoPilotMaxThroughput = this.maxAutoPilotThroughputSet();
|
||||||
|
} else {
|
||||||
|
createDatabaseParams.offerThroughput = addDatabasePaneStartMessage.offerThroughput;
|
||||||
|
}
|
||||||
|
|
||||||
|
createDatabase(createDatabaseParams).then(
|
||||||
|
(database: DataModels.Database) => {
|
||||||
|
this._onCreateDatabaseSuccess(offerThroughput, startKey);
|
||||||
|
},
|
||||||
|
(error: any) => {
|
||||||
|
this._onCreateDatabaseFailure(error, offerThroughput, startKey);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public resetData() {
|
||||||
|
this.databaseId("");
|
||||||
|
this.databaseCreateNewShared(this.getSharedThroughputDefault());
|
||||||
|
this.isAutoPilotSelected(this.container.isAutoscaleDefaultEnabled());
|
||||||
|
this.maxAutoPilotThroughputSet(AutoPilotUtils.minAutoPilotThroughput);
|
||||||
|
this._updateThroughputLimitByDatabase();
|
||||||
|
this.throughputSpendAck(false);
|
||||||
|
super.resetData();
|
||||||
|
}
|
||||||
|
|
||||||
|
public getSharedThroughputDefault(): boolean {
|
||||||
|
const subscriptionType = userContext.subscriptionType;
|
||||||
|
|
||||||
|
if (subscriptionType === SubscriptionType.EA || this.container.isServerlessEnabled()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _onCreateDatabaseSuccess(offerThroughput: number, startKey: number): void {
|
||||||
|
this.isExecuting(false);
|
||||||
|
this.close();
|
||||||
|
this.container.refreshAllDatabases();
|
||||||
|
const addDatabasePaneSuccessMessage = {
|
||||||
|
database: ko.toJS({
|
||||||
|
id: this.databaseId(),
|
||||||
|
shared: this.databaseCreateNewShared(),
|
||||||
|
}),
|
||||||
|
offerThroughput: offerThroughput,
|
||||||
|
subscriptionType: userContext.subscriptionType,
|
||||||
|
subscriptionQuotaId: userContext.quotaId,
|
||||||
|
defaultsCheck: {
|
||||||
|
flight: userContext.addCollectionFlight,
|
||||||
|
},
|
||||||
|
dataExplorerArea: Constants.Areas.ContextualPane,
|
||||||
|
};
|
||||||
|
TelemetryProcessor.traceSuccess(Action.CreateDatabase, addDatabasePaneSuccessMessage, startKey);
|
||||||
|
this.resetData();
|
||||||
|
}
|
||||||
|
|
||||||
|
private _onCreateDatabaseFailure(error: any, offerThroughput: number, startKey: number): void {
|
||||||
|
this.isExecuting(false);
|
||||||
|
const errorMessage = getErrorMessage(error);
|
||||||
|
this.formErrors(errorMessage);
|
||||||
|
this.formErrorsDetails(errorMessage);
|
||||||
|
const addDatabasePaneFailedMessage = {
|
||||||
|
database: ko.toJS({
|
||||||
|
id: this.databaseId(),
|
||||||
|
shared: this.databaseCreateNewShared(),
|
||||||
|
}),
|
||||||
|
offerThroughput: offerThroughput,
|
||||||
|
subscriptionType: userContext.subscriptionType,
|
||||||
|
subscriptionQuotaId: userContext.quotaId,
|
||||||
|
defaultsCheck: {
|
||||||
|
flight: userContext.addCollectionFlight,
|
||||||
|
},
|
||||||
|
dataExplorerArea: Constants.Areas.ContextualPane,
|
||||||
|
error: errorMessage,
|
||||||
|
errorStack: getErrorStack(error),
|
||||||
|
};
|
||||||
|
TelemetryProcessor.traceFailure(Action.CreateDatabase, addDatabasePaneFailedMessage, startKey);
|
||||||
|
}
|
||||||
|
|
||||||
|
private _getThroughput(): number {
|
||||||
|
const throughput: number = this.throughput();
|
||||||
|
return isNaN(throughput) ? 0 : Number(throughput);
|
||||||
|
}
|
||||||
|
|
||||||
|
private _computeOfferThroughput(): number {
|
||||||
|
if (!this.canConfigureThroughput()) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.isAutoPilotSelected()) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
return this._getThroughput();
|
||||||
|
}
|
||||||
|
|
||||||
|
private _isValid(): boolean {
|
||||||
|
// TODO add feature flag that disables validation for customers with custom accounts
|
||||||
|
if (this.isAutoPilotSelected()) {
|
||||||
|
const autoPilot = this._isAutoPilotSelectedAndWhatTier();
|
||||||
|
if (
|
||||||
|
!autoPilot ||
|
||||||
|
!autoPilot.maxThroughput ||
|
||||||
|
!AutoPilotUtils.isValidAutoPilotThroughput(autoPilot.maxThroughput)
|
||||||
|
) {
|
||||||
|
this.formErrors(
|
||||||
|
`Please enter a value greater than ${AutoPilotUtils.minAutoPilotThroughput} for autopilot throughput`
|
||||||
|
);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const throughput = this._getThroughput();
|
||||||
|
|
||||||
|
if (throughput > SharedConstants.CollectionCreation.DefaultCollectionRUs100K && !this.throughputSpendAck()) {
|
||||||
|
this.formErrors(`Please acknowledge the estimated daily spend.`);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const autoscaleThroughput = this.maxAutoPilotThroughputSet() * 1;
|
||||||
|
|
||||||
|
if (
|
||||||
|
this.isAutoPilotSelected() &&
|
||||||
|
autoscaleThroughput > SharedConstants.CollectionCreation.DefaultCollectionRUs100K &&
|
||||||
|
!this.throughputSpendAck()
|
||||||
|
) {
|
||||||
|
this.formErrors(`Please acknowledge the estimated monthly spend.`);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _isAutoPilotSelectedAndWhatTier(): DataModels.AutoPilotCreationSettings {
|
||||||
|
if (this.isAutoPilotSelected() && this.maxAutoPilotThroughputSet()) {
|
||||||
|
return {
|
||||||
|
maxThroughput: this.maxAutoPilotThroughputSet() * 1,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _updateThroughputLimitByDatabase() {
|
||||||
|
const throughputDefaults = this.container.collectionCreationDefaults.throughput;
|
||||||
|
this.throughput(throughputDefaults.shared);
|
||||||
|
this.maxThroughputRU(throughputDefaults.unlimitedmax);
|
||||||
|
this.minThroughputRU(throughputDefaults.unlimitedmin);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -118,7 +118,7 @@ export const AddDatabasePane: FunctionComponent<AddDatabasePaneProps> = ({
|
|||||||
subscriptionType: SubscriptionType[subscriptionType],
|
subscriptionType: SubscriptionType[subscriptionType],
|
||||||
subscriptionQuotaId: userContext.quotaId,
|
subscriptionQuotaId: userContext.quotaId,
|
||||||
defaultsCheck: {
|
defaultsCheck: {
|
||||||
flight: container.flight(),
|
flight: userContext.addCollectionFlight,
|
||||||
},
|
},
|
||||||
dataExplorerArea: Constants.Areas.ContextualPane,
|
dataExplorerArea: Constants.Areas.ContextualPane,
|
||||||
};
|
};
|
||||||
@@ -129,7 +129,7 @@ export const AddDatabasePane: FunctionComponent<AddDatabasePaneProps> = ({
|
|||||||
subscriptionQuotaId: userContext.quotaId,
|
subscriptionQuotaId: userContext.quotaId,
|
||||||
defaultsCheck: {
|
defaultsCheck: {
|
||||||
throughput: throughput,
|
throughput: throughput,
|
||||||
flight: container.flight(),
|
flight: userContext.addCollectionFlight,
|
||||||
},
|
},
|
||||||
dataExplorerArea: Constants.Areas.ContextualPane,
|
dataExplorerArea: Constants.Areas.ContextualPane,
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import { getErrorMessage, getErrorStack } from "../../Common/ErrorHandlingUtils"
|
|||||||
import { HashMap } from "../../Common/HashMap";
|
import { HashMap } from "../../Common/HashMap";
|
||||||
import { configContext, Platform } from "../../ConfigContext";
|
import { configContext, Platform } from "../../ConfigContext";
|
||||||
import * as DataModels from "../../Contracts/DataModels";
|
import * as DataModels from "../../Contracts/DataModels";
|
||||||
import { SubscriptionType } from "../../Contracts/SubscriptionType";
|
|
||||||
import * as ViewModels from "../../Contracts/ViewModels";
|
import * as ViewModels from "../../Contracts/ViewModels";
|
||||||
import * as AddCollectionUtility from "../../Shared/AddCollectionUtility";
|
import * as AddCollectionUtility from "../../Shared/AddCollectionUtility";
|
||||||
import * as SharedConstants from "../../Shared/Constants";
|
import * as SharedConstants from "../../Shared/Constants";
|
||||||
@@ -117,10 +116,6 @@ export default class CassandraAddCollectionPane extends ContextualPaneBase {
|
|||||||
|
|
||||||
this.resetData();
|
this.resetData();
|
||||||
|
|
||||||
this.container.flight.subscribe(() => {
|
|
||||||
this.resetData();
|
|
||||||
});
|
|
||||||
|
|
||||||
this.requestUnitsUsageCostDedicated = ko.computed(() => {
|
this.requestUnitsUsageCostDedicated = ko.computed(() => {
|
||||||
const account = this.container.databaseAccount();
|
const account = this.container.databaseAccount();
|
||||||
if (!account) {
|
if (!account) {
|
||||||
@@ -306,12 +301,12 @@ export default class CassandraAddCollectionPane extends ContextualPaneBase {
|
|||||||
partitionKey: "",
|
partitionKey: "",
|
||||||
databaseId: this.keyspaceId(),
|
databaseId: this.keyspaceId(),
|
||||||
}),
|
}),
|
||||||
subscriptionType: SubscriptionType[this.container.subscriptionType()],
|
subscriptionType: userContext.subscriptionType,
|
||||||
subscriptionQuotaId: userContext.quotaId,
|
subscriptionQuotaId: userContext.quotaId,
|
||||||
defaultsCheck: {
|
defaultsCheck: {
|
||||||
storage: "u",
|
storage: "u",
|
||||||
throughput: this.throughput(),
|
throughput: this.throughput(),
|
||||||
flight: this.container.flight(),
|
flight: userContext.addCollectionFlight,
|
||||||
},
|
},
|
||||||
dataExplorerArea: Constants.Areas.ContextualPane,
|
dataExplorerArea: Constants.Areas.ContextualPane,
|
||||||
};
|
};
|
||||||
@@ -358,12 +353,12 @@ export default class CassandraAddCollectionPane extends ContextualPaneBase {
|
|||||||
hasDedicatedThroughput: this.dedicateTableThroughput(),
|
hasDedicatedThroughput: this.dedicateTableThroughput(),
|
||||||
}),
|
}),
|
||||||
keyspaceHasSharedOffer: this.keyspaceHasSharedOffer(),
|
keyspaceHasSharedOffer: this.keyspaceHasSharedOffer(),
|
||||||
subscriptionType: SubscriptionType[this.container.subscriptionType()],
|
subscriptionType: userContext.subscriptionType,
|
||||||
subscriptionQuotaId: userContext.quotaId,
|
subscriptionQuotaId: userContext.quotaId,
|
||||||
defaultsCheck: {
|
defaultsCheck: {
|
||||||
storage: "u",
|
storage: "u",
|
||||||
throughput: this.throughput(),
|
throughput: this.throughput(),
|
||||||
flight: this.container.flight(),
|
flight: userContext.addCollectionFlight,
|
||||||
},
|
},
|
||||||
dataExplorerArea: Constants.Areas.ContextualPane,
|
dataExplorerArea: Constants.Areas.ContextualPane,
|
||||||
toCreateKeyspace: toCreateKeyspace,
|
toCreateKeyspace: toCreateKeyspace,
|
||||||
@@ -402,12 +397,12 @@ export default class CassandraAddCollectionPane extends ContextualPaneBase {
|
|||||||
hasDedicatedThroughput: this.dedicateTableThroughput(),
|
hasDedicatedThroughput: this.dedicateTableThroughput(),
|
||||||
}),
|
}),
|
||||||
keyspaceHasSharedOffer: this.keyspaceHasSharedOffer(),
|
keyspaceHasSharedOffer: this.keyspaceHasSharedOffer(),
|
||||||
subscriptionType: SubscriptionType[this.container.subscriptionType()],
|
subscriptionType: userContext.subscriptionType,
|
||||||
subscriptionQuotaId: userContext.quotaId,
|
subscriptionQuotaId: userContext.quotaId,
|
||||||
defaultsCheck: {
|
defaultsCheck: {
|
||||||
storage: "u",
|
storage: "u",
|
||||||
throughput: this.throughput(),
|
throughput: this.throughput(),
|
||||||
flight: this.container.flight(),
|
flight: userContext.addCollectionFlight,
|
||||||
},
|
},
|
||||||
dataExplorerArea: Constants.Areas.ContextualPane,
|
dataExplorerArea: Constants.Areas.ContextualPane,
|
||||||
toCreateKeyspace: toCreateKeyspace,
|
toCreateKeyspace: toCreateKeyspace,
|
||||||
@@ -430,12 +425,12 @@ export default class CassandraAddCollectionPane extends ContextualPaneBase {
|
|||||||
hasDedicatedThroughput: this.dedicateTableThroughput(),
|
hasDedicatedThroughput: this.dedicateTableThroughput(),
|
||||||
},
|
},
|
||||||
keyspaceHasSharedOffer: this.keyspaceHasSharedOffer(),
|
keyspaceHasSharedOffer: this.keyspaceHasSharedOffer(),
|
||||||
subscriptionType: SubscriptionType[this.container.subscriptionType()],
|
subscriptionType: userContext.subscriptionType,
|
||||||
subscriptionQuotaId: userContext.quotaId,
|
subscriptionQuotaId: userContext.quotaId,
|
||||||
defaultsCheck: {
|
defaultsCheck: {
|
||||||
storage: "u",
|
storage: "u",
|
||||||
throughput: this.throughput(),
|
throughput: this.throughput(),
|
||||||
flight: this.container.flight(),
|
flight: userContext.addCollectionFlight,
|
||||||
},
|
},
|
||||||
dataExplorerArea: Constants.Areas.ContextualPane,
|
dataExplorerArea: Constants.Areas.ContextualPane,
|
||||||
toCreateKeyspace: toCreateKeyspace,
|
toCreateKeyspace: toCreateKeyspace,
|
||||||
|
|||||||
@@ -7,7 +7,6 @@ import GraphStylingPaneTemplate from "./GraphStylingPane.html";
|
|||||||
import SetupNotebooksPaneTemplate from "./SetupNotebooksPane.html";
|
import SetupNotebooksPaneTemplate from "./SetupNotebooksPane.html";
|
||||||
import StringInputPaneTemplate from "./StringInputPane.html";
|
import StringInputPaneTemplate from "./StringInputPane.html";
|
||||||
import TableAddEntityPaneTemplate from "./Tables/TableAddEntityPane.html";
|
import TableAddEntityPaneTemplate from "./Tables/TableAddEntityPane.html";
|
||||||
import TableColumnOptionsPaneTemplate from "./Tables/TableColumnOptionsPane.html";
|
|
||||||
import TableEditEntityPaneTemplate from "./Tables/TableEditEntityPane.html";
|
import TableEditEntityPaneTemplate from "./Tables/TableEditEntityPane.html";
|
||||||
import TableQuerySelectPaneTemplate from "./Tables/TableQuerySelectPane.html";
|
import TableQuerySelectPaneTemplate from "./Tables/TableQuerySelectPane.html";
|
||||||
|
|
||||||
@@ -71,15 +70,6 @@ export class TableEditEntityPaneComponent {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class TableColumnOptionsPaneComponent {
|
|
||||||
constructor() {
|
|
||||||
return {
|
|
||||||
viewModel: PaneComponent,
|
|
||||||
template: TableColumnOptionsPaneTemplate,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export class TableQuerySelectPaneComponent {
|
export class TableQuerySelectPaneComponent {
|
||||||
constructor() {
|
constructor() {
|
||||||
return {
|
return {
|
||||||
|
|||||||
@@ -196,31 +196,6 @@ exports[`Settings Pane should render Default properly 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
TableColumnOptionsPane {
|
|
||||||
"allSelected": [Function],
|
|
||||||
"anyColumnSelected": [Function],
|
|
||||||
"availableColumnsLabel": "Available Columns",
|
|
||||||
"canMoveDown": [Function],
|
|
||||||
"canMoveUp": [Function],
|
|
||||||
"canSelectAll": [Function],
|
|
||||||
"columnOptions": [Function],
|
|
||||||
"container": [Circular],
|
|
||||||
"firstFieldHasFocus": [Function],
|
|
||||||
"formErrors": [Function],
|
|
||||||
"formErrorsDetails": [Function],
|
|
||||||
"handleClick": [Function],
|
|
||||||
"id": "tablecolumnoptionspane",
|
|
||||||
"instructionLabel": "Choose the columns and the order in which you want to display them in the table.",
|
|
||||||
"isExecuting": [Function],
|
|
||||||
"isTemplateReady": [Function],
|
|
||||||
"moveDownLabel": "Move Down",
|
|
||||||
"moveUpLabel": "Move Up",
|
|
||||||
"noColumnSelectedWarning": "At least one column should be selected.",
|
|
||||||
"selectedColumnOption": null,
|
|
||||||
"title": [Function],
|
|
||||||
"titleLabel": "Column Options",
|
|
||||||
"visible": [Function],
|
|
||||||
},
|
|
||||||
QuerySelectPane {
|
QuerySelectPane {
|
||||||
"allSelected": [Function],
|
"allSelected": [Function],
|
||||||
"anyColumnSelected": [Function],
|
"anyColumnSelected": [Function],
|
||||||
@@ -583,7 +558,6 @@ exports[`Settings Pane should render Default properly 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"flight": [Function],
|
|
||||||
"graphStylingPane": GraphStylingPane {
|
"graphStylingPane": GraphStylingPane {
|
||||||
"container": [Circular],
|
"container": [Circular],
|
||||||
"firstFieldHasFocus": [Function],
|
"firstFieldHasFocus": [Function],
|
||||||
@@ -605,7 +579,6 @@ exports[`Settings Pane should render Default properly 1`] = `
|
|||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"hasStorageAnalyticsAfecFeature": [Function],
|
"hasStorageAnalyticsAfecFeature": [Function],
|
||||||
"hasWriteAccess": [Function],
|
|
||||||
"isAccountReady": [Function],
|
"isAccountReady": [Function],
|
||||||
"isAutoscaleDefaultEnabled": [Function],
|
"isAutoscaleDefaultEnabled": [Function],
|
||||||
"isCopyNotebookPaneEnabled": [Function],
|
"isCopyNotebookPaneEnabled": [Function],
|
||||||
@@ -758,32 +731,6 @@ exports[`Settings Pane should render Default properly 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"subscriptionType": [Function],
|
|
||||||
"tableColumnOptionsPane": TableColumnOptionsPane {
|
|
||||||
"allSelected": [Function],
|
|
||||||
"anyColumnSelected": [Function],
|
|
||||||
"availableColumnsLabel": "Available Columns",
|
|
||||||
"canMoveDown": [Function],
|
|
||||||
"canMoveUp": [Function],
|
|
||||||
"canSelectAll": [Function],
|
|
||||||
"columnOptions": [Function],
|
|
||||||
"container": [Circular],
|
|
||||||
"firstFieldHasFocus": [Function],
|
|
||||||
"formErrors": [Function],
|
|
||||||
"formErrorsDetails": [Function],
|
|
||||||
"handleClick": [Function],
|
|
||||||
"id": "tablecolumnoptionspane",
|
|
||||||
"instructionLabel": "Choose the columns and the order in which you want to display them in the table.",
|
|
||||||
"isExecuting": [Function],
|
|
||||||
"isTemplateReady": [Function],
|
|
||||||
"moveDownLabel": "Move Down",
|
|
||||||
"moveUpLabel": "Move Up",
|
|
||||||
"noColumnSelectedWarning": "At least one column should be selected.",
|
|
||||||
"selectedColumnOption": null,
|
|
||||||
"title": [Function],
|
|
||||||
"titleLabel": "Column Options",
|
|
||||||
"visible": [Function],
|
|
||||||
},
|
|
||||||
"tabsManager": TabsManager {
|
"tabsManager": TabsManager {
|
||||||
"activeTab": [Function],
|
"activeTab": [Function],
|
||||||
"openedTabs": [Function],
|
"openedTabs": [Function],
|
||||||
@@ -1090,31 +1037,6 @@ exports[`Settings Pane should render Gremlin properly 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
TableColumnOptionsPane {
|
|
||||||
"allSelected": [Function],
|
|
||||||
"anyColumnSelected": [Function],
|
|
||||||
"availableColumnsLabel": "Available Columns",
|
|
||||||
"canMoveDown": [Function],
|
|
||||||
"canMoveUp": [Function],
|
|
||||||
"canSelectAll": [Function],
|
|
||||||
"columnOptions": [Function],
|
|
||||||
"container": [Circular],
|
|
||||||
"firstFieldHasFocus": [Function],
|
|
||||||
"formErrors": [Function],
|
|
||||||
"formErrorsDetails": [Function],
|
|
||||||
"handleClick": [Function],
|
|
||||||
"id": "tablecolumnoptionspane",
|
|
||||||
"instructionLabel": "Choose the columns and the order in which you want to display them in the table.",
|
|
||||||
"isExecuting": [Function],
|
|
||||||
"isTemplateReady": [Function],
|
|
||||||
"moveDownLabel": "Move Down",
|
|
||||||
"moveUpLabel": "Move Up",
|
|
||||||
"noColumnSelectedWarning": "At least one column should be selected.",
|
|
||||||
"selectedColumnOption": null,
|
|
||||||
"title": [Function],
|
|
||||||
"titleLabel": "Column Options",
|
|
||||||
"visible": [Function],
|
|
||||||
},
|
|
||||||
QuerySelectPane {
|
QuerySelectPane {
|
||||||
"allSelected": [Function],
|
"allSelected": [Function],
|
||||||
"anyColumnSelected": [Function],
|
"anyColumnSelected": [Function],
|
||||||
@@ -1477,7 +1399,6 @@ exports[`Settings Pane should render Gremlin properly 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"flight": [Function],
|
|
||||||
"graphStylingPane": GraphStylingPane {
|
"graphStylingPane": GraphStylingPane {
|
||||||
"container": [Circular],
|
"container": [Circular],
|
||||||
"firstFieldHasFocus": [Function],
|
"firstFieldHasFocus": [Function],
|
||||||
@@ -1499,7 +1420,6 @@ exports[`Settings Pane should render Gremlin properly 1`] = `
|
|||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"hasStorageAnalyticsAfecFeature": [Function],
|
"hasStorageAnalyticsAfecFeature": [Function],
|
||||||
"hasWriteAccess": [Function],
|
|
||||||
"isAccountReady": [Function],
|
"isAccountReady": [Function],
|
||||||
"isAutoscaleDefaultEnabled": [Function],
|
"isAutoscaleDefaultEnabled": [Function],
|
||||||
"isCopyNotebookPaneEnabled": [Function],
|
"isCopyNotebookPaneEnabled": [Function],
|
||||||
@@ -1652,32 +1572,6 @@ exports[`Settings Pane should render Gremlin properly 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"subscriptionType": [Function],
|
|
||||||
"tableColumnOptionsPane": TableColumnOptionsPane {
|
|
||||||
"allSelected": [Function],
|
|
||||||
"anyColumnSelected": [Function],
|
|
||||||
"availableColumnsLabel": "Available Columns",
|
|
||||||
"canMoveDown": [Function],
|
|
||||||
"canMoveUp": [Function],
|
|
||||||
"canSelectAll": [Function],
|
|
||||||
"columnOptions": [Function],
|
|
||||||
"container": [Circular],
|
|
||||||
"firstFieldHasFocus": [Function],
|
|
||||||
"formErrors": [Function],
|
|
||||||
"formErrorsDetails": [Function],
|
|
||||||
"handleClick": [Function],
|
|
||||||
"id": "tablecolumnoptionspane",
|
|
||||||
"instructionLabel": "Choose the columns and the order in which you want to display them in the table.",
|
|
||||||
"isExecuting": [Function],
|
|
||||||
"isTemplateReady": [Function],
|
|
||||||
"moveDownLabel": "Move Down",
|
|
||||||
"moveUpLabel": "Move Up",
|
|
||||||
"noColumnSelectedWarning": "At least one column should be selected.",
|
|
||||||
"selectedColumnOption": null,
|
|
||||||
"title": [Function],
|
|
||||||
"titleLabel": "Column Options",
|
|
||||||
"visible": [Function],
|
|
||||||
},
|
|
||||||
"tabsManager": TabsManager {
|
"tabsManager": TabsManager {
|
||||||
"activeTab": [Function],
|
"activeTab": [Function],
|
||||||
"openedTabs": [Function],
|
"openedTabs": [Function],
|
||||||
|
|||||||
@@ -1,78 +0,0 @@
|
|||||||
<div data-bind="visible: visible">
|
|
||||||
<div
|
|
||||||
class="contextual-pane-out"
|
|
||||||
data-bind="
|
|
||||||
click: cancel,
|
|
||||||
clickBubble: false"
|
|
||||||
></div>
|
|
||||||
<div class="contextual-pane" id="tablecolumnoptionspane">
|
|
||||||
<!-- Table Column Options form - Start -->
|
|
||||||
<div class="contextual-pane-in">
|
|
||||||
<form
|
|
||||||
class="paneContentContainer"
|
|
||||||
data-bind="
|
|
||||||
submit: submit"
|
|
||||||
>
|
|
||||||
<!-- Table Column Options header - Start -->
|
|
||||||
<div class="firstdivbg headerline">
|
|
||||||
Column Options
|
|
||||||
<div
|
|
||||||
class="closeImg"
|
|
||||||
role="button"
|
|
||||||
aria-label="Close pane"
|
|
||||||
tabindex="0"
|
|
||||||
data-bind="
|
|
||||||
click: cancel"
|
|
||||||
>
|
|
||||||
<img src="../../../../images/close-black.svg" title="Close" alt="Close" />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<!-- Table Column Options header - End -->
|
|
||||||
<div class="paneMainContent paneContentContainer">
|
|
||||||
<div><span>Choose the columns and the order in which you want to display them in the table.</span></div>
|
|
||||||
<div class="column-options">
|
|
||||||
<div class="columns-border">
|
|
||||||
<input class="all-select-check" type="checkbox" data-bind="checked: allSelected" />
|
|
||||||
<label
|
|
||||||
style="font-weight: 700"
|
|
||||||
id="availableColumnsLabel"
|
|
||||||
data-bind="text: availableColumnsLabel"
|
|
||||||
></label>
|
|
||||||
<span class="column-arrows-svg" data-bind="click: moveDown, enable: canMoveDown">
|
|
||||||
<img class="column-opt-arrow-Img" src="/Down.svg" alt="Move down" />
|
|
||||||
</span>
|
|
||||||
<span class="column-arrows-svg" data-bind="click: moveUp, enable: canMoveUp">
|
|
||||||
<img class="column-opt-arrow-Img" src="/Up.svg" alt="Move up" />
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div class="content">
|
|
||||||
<section>
|
|
||||||
<ul data-bind="foreach: columnOptions" aria-labelledby="availableColumnsLabel" tabindex="0">
|
|
||||||
<li
|
|
||||||
class="list-item columns-border"
|
|
||||||
data-bind="attr: { title: columnName }, click: $parent.handleClick "
|
|
||||||
>
|
|
||||||
<input
|
|
||||||
type="checkbox"
|
|
||||||
for="columnName"
|
|
||||||
data-bind="attr: { title: columnName, 'aria-selected': (selected()? 'true': 'false') }, checked: selected"
|
|
||||||
/>
|
|
||||||
<label id="columnName" data-bind="text: columnName"></label>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</section>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="row-label" data-bind="style: { visibility: anyColumnSelected() ? 'hidden': 'visible' }">
|
|
||||||
<label class="warning" role="alert" aria-atomic="true" data-bind="text: noColumnSelectedWarning"></label>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="paneFooter">
|
|
||||||
<div class="leftpanel-okbut"><input type="submit" value="OK" class="btncreatecoll1" /></div>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
<!-- Table Column Options form - End -->
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
@@ -1,195 +0,0 @@
|
|||||||
import * as ko from "knockout";
|
|
||||||
import * as ViewModels from "../../../Contracts/ViewModels";
|
|
||||||
import * as DataTableOperations from "../../Tables/DataTable/DataTableOperations";
|
|
||||||
import TableEntityListViewModel from "../../Tables/DataTable/TableEntityListViewModel";
|
|
||||||
import { ContextualPaneBase } from "../ContextualPaneBase";
|
|
||||||
import _ from "underscore";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Represents an item shown in the available columns.
|
|
||||||
* columnName: the name of the column.
|
|
||||||
* selected: indicate whether user wants to display the column in the table.
|
|
||||||
* order: the order in the initial table. E.g.,
|
|
||||||
* Order array of initial table: I = [0, 1, 2, 3, 4, 5, 6, 7, 8] <----> {prop0, prop1, prop2, prop3, prop4, prop5, prop6, prop7, prop8}
|
|
||||||
* Order array of current table: C = [0, 1, 2, 6, 7, 3, 4, 5, 8] <----> {prop0, prop1, prop2, prop6, prop7, prop3, prop4, prop5, prop8}
|
|
||||||
* if order = 6, then this column will be the one with column name prop6
|
|
||||||
* index: index in the observable array, this used for selection and moving up/down.
|
|
||||||
*/
|
|
||||||
interface IColumnOption {
|
|
||||||
columnName: ko.Observable<string>;
|
|
||||||
selected: ko.Observable<boolean>;
|
|
||||||
order: number;
|
|
||||||
index: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface IColumnSetting {
|
|
||||||
columnNames: string[];
|
|
||||||
visible?: boolean[];
|
|
||||||
order?: number[];
|
|
||||||
}
|
|
||||||
|
|
||||||
export class TableColumnOptionsPane extends ContextualPaneBase {
|
|
||||||
public titleLabel: string = "Column Options";
|
|
||||||
public instructionLabel: string = "Choose the columns and the order in which you want to display them in the table.";
|
|
||||||
public availableColumnsLabel: string = "Available Columns";
|
|
||||||
public moveUpLabel: string = "Move Up";
|
|
||||||
public moveDownLabel: string = "Move Down";
|
|
||||||
public noColumnSelectedWarning: string = "At least one column should be selected.";
|
|
||||||
|
|
||||||
public columnOptions: ko.ObservableArray<IColumnOption>;
|
|
||||||
public allSelected: ko.Computed<boolean>;
|
|
||||||
public anyColumnSelected: ko.Computed<boolean>;
|
|
||||||
public canSelectAll: ko.Computed<boolean>;
|
|
||||||
public canMoveUp: ko.Observable<boolean>;
|
|
||||||
public canMoveDown: ko.Observable<boolean>;
|
|
||||||
|
|
||||||
public tableViewModel: TableEntityListViewModel;
|
|
||||||
public parameters: IColumnSetting;
|
|
||||||
|
|
||||||
private selectedColumnOption: IColumnOption = null;
|
|
||||||
|
|
||||||
constructor(options: ViewModels.PaneOptions) {
|
|
||||||
super(options);
|
|
||||||
|
|
||||||
this.columnOptions = ko.observableArray<IColumnOption>();
|
|
||||||
this.anyColumnSelected = ko.computed<boolean>(() => {
|
|
||||||
return _.some(this.columnOptions(), (value: IColumnOption) => {
|
|
||||||
return value.selected();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
this.canSelectAll = ko.computed<boolean>(() => {
|
|
||||||
return _.some(this.columnOptions(), (value: IColumnOption) => {
|
|
||||||
return !value.selected();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
this.canMoveUp = ko.observable<boolean>(false);
|
|
||||||
this.canMoveDown = ko.observable<boolean>(false);
|
|
||||||
|
|
||||||
this.allSelected = ko.pureComputed<boolean>({
|
|
||||||
read: () => {
|
|
||||||
return !this.canSelectAll();
|
|
||||||
},
|
|
||||||
write: (value) => {
|
|
||||||
if (value) {
|
|
||||||
this.selectAll();
|
|
||||||
} else {
|
|
||||||
this.clearAll();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
owner: this,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public submit() {
|
|
||||||
var newColumnSetting = this.getParameters();
|
|
||||||
DataTableOperations.reorderColumns(this.tableViewModel.table, newColumnSetting.order).then(() => {
|
|
||||||
DataTableOperations.filterColumns(this.tableViewModel.table, newColumnSetting.visible);
|
|
||||||
this.visible(false);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
public open() {
|
|
||||||
this.setDisplayedColumns(this.parameters.columnNames, this.parameters.order, this.parameters.visible);
|
|
||||||
super.open();
|
|
||||||
}
|
|
||||||
|
|
||||||
private getParameters(): IColumnSetting {
|
|
||||||
var newColumnSettings: IColumnSetting = <IColumnSetting>{
|
|
||||||
columnNames: [],
|
|
||||||
order: [],
|
|
||||||
visible: [],
|
|
||||||
};
|
|
||||||
this.columnOptions().map((value: IColumnOption) => {
|
|
||||||
newColumnSettings.columnNames.push(value.columnName());
|
|
||||||
newColumnSettings.order.push(value.order);
|
|
||||||
newColumnSettings.visible.push(value.selected());
|
|
||||||
});
|
|
||||||
return newColumnSettings;
|
|
||||||
}
|
|
||||||
|
|
||||||
public setDisplayedColumns(columnNames: string[], order: number[], visible: boolean[]): void {
|
|
||||||
var options: IColumnOption[] = order.map((value: number, index: number) => {
|
|
||||||
var columnOption: IColumnOption = {
|
|
||||||
columnName: ko.observable<string>(columnNames[index]),
|
|
||||||
order: value,
|
|
||||||
selected: ko.observable<boolean>(visible[index]),
|
|
||||||
index: index,
|
|
||||||
};
|
|
||||||
return columnOption;
|
|
||||||
});
|
|
||||||
this.columnOptions(options);
|
|
||||||
}
|
|
||||||
|
|
||||||
public selectAll(): void {
|
|
||||||
const columnOptions = this.columnOptions && this.columnOptions();
|
|
||||||
columnOptions &&
|
|
||||||
columnOptions.forEach((value: IColumnOption) => {
|
|
||||||
value.selected(true);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public clearAll(): void {
|
|
||||||
const columnOptions = this.columnOptions && this.columnOptions();
|
|
||||||
columnOptions &&
|
|
||||||
columnOptions.forEach((value: IColumnOption) => {
|
|
||||||
value.selected(false);
|
|
||||||
});
|
|
||||||
|
|
||||||
if (columnOptions && columnOptions.length > 0) {
|
|
||||||
columnOptions[0].selected(true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public moveUp(): void {
|
|
||||||
if (this.selectedColumnOption) {
|
|
||||||
var currentSelectedIndex: number = this.selectedColumnOption.index;
|
|
||||||
var swapTargetIndex: number = currentSelectedIndex - 1;
|
|
||||||
//Debug.assert(currentSelectedIndex > 0);
|
|
||||||
|
|
||||||
this.swapColumnOption(this.columnOptions(), swapTargetIndex, currentSelectedIndex);
|
|
||||||
this.selectTargetItem($(`div.column-options li:eq(${swapTargetIndex})`), this.columnOptions()[swapTargetIndex]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public moveDown(): void {
|
|
||||||
if (this.selectedColumnOption) {
|
|
||||||
var currentSelectedIndex: number = this.selectedColumnOption.index;
|
|
||||||
var swapTargetIndex: number = currentSelectedIndex + 1;
|
|
||||||
//Debug.assert(currentSelectedIndex < (this.columnOptions().length - 1));
|
|
||||||
|
|
||||||
this.swapColumnOption(this.columnOptions(), swapTargetIndex, currentSelectedIndex);
|
|
||||||
this.selectTargetItem($(`div.column-options li:eq(${swapTargetIndex})`), this.columnOptions()[swapTargetIndex]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public handleClick = (data: IColumnOption, event: KeyboardEvent): boolean => {
|
|
||||||
this.selectTargetItem($(event.currentTarget), data);
|
|
||||||
return true;
|
|
||||||
};
|
|
||||||
|
|
||||||
private selectTargetItem($target: JQuery, targetColumn: IColumnOption): void {
|
|
||||||
this.selectedColumnOption = targetColumn;
|
|
||||||
|
|
||||||
this.canMoveUp(targetColumn.index !== 0);
|
|
||||||
this.canMoveDown(targetColumn.index !== this.columnOptions().length - 1);
|
|
||||||
|
|
||||||
$(".list-item.selected").removeClass("selected");
|
|
||||||
$target.addClass("selected");
|
|
||||||
}
|
|
||||||
|
|
||||||
private swapColumnOption(options: IColumnOption[], indexA: number, indexB: number): void {
|
|
||||||
var tempColumnName: string = options[indexA].columnName();
|
|
||||||
var tempSelected: boolean = options[indexA].selected();
|
|
||||||
var tempOrder: number = options[indexA].order;
|
|
||||||
|
|
||||||
options[indexA].columnName(options[indexB].columnName());
|
|
||||||
options[indexB].columnName(tempColumnName);
|
|
||||||
|
|
||||||
options[indexA].selected(options[indexB].selected());
|
|
||||||
options[indexB].selected(tempSelected);
|
|
||||||
|
|
||||||
options[indexA].order = options[indexB].order;
|
|
||||||
options[indexB].order = tempOrder;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -196,31 +196,6 @@ exports[`Upload Items Pane should render Default properly 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
TableColumnOptionsPane {
|
|
||||||
"allSelected": [Function],
|
|
||||||
"anyColumnSelected": [Function],
|
|
||||||
"availableColumnsLabel": "Available Columns",
|
|
||||||
"canMoveDown": [Function],
|
|
||||||
"canMoveUp": [Function],
|
|
||||||
"canSelectAll": [Function],
|
|
||||||
"columnOptions": [Function],
|
|
||||||
"container": [Circular],
|
|
||||||
"firstFieldHasFocus": [Function],
|
|
||||||
"formErrors": [Function],
|
|
||||||
"formErrorsDetails": [Function],
|
|
||||||
"handleClick": [Function],
|
|
||||||
"id": "tablecolumnoptionspane",
|
|
||||||
"instructionLabel": "Choose the columns and the order in which you want to display them in the table.",
|
|
||||||
"isExecuting": [Function],
|
|
||||||
"isTemplateReady": [Function],
|
|
||||||
"moveDownLabel": "Move Down",
|
|
||||||
"moveUpLabel": "Move Up",
|
|
||||||
"noColumnSelectedWarning": "At least one column should be selected.",
|
|
||||||
"selectedColumnOption": null,
|
|
||||||
"title": [Function],
|
|
||||||
"titleLabel": "Column Options",
|
|
||||||
"visible": [Function],
|
|
||||||
},
|
|
||||||
QuerySelectPane {
|
QuerySelectPane {
|
||||||
"allSelected": [Function],
|
"allSelected": [Function],
|
||||||
"anyColumnSelected": [Function],
|
"anyColumnSelected": [Function],
|
||||||
@@ -583,7 +558,6 @@ exports[`Upload Items Pane should render Default properly 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"flight": [Function],
|
|
||||||
"graphStylingPane": GraphStylingPane {
|
"graphStylingPane": GraphStylingPane {
|
||||||
"container": [Circular],
|
"container": [Circular],
|
||||||
"firstFieldHasFocus": [Function],
|
"firstFieldHasFocus": [Function],
|
||||||
@@ -605,7 +579,6 @@ exports[`Upload Items Pane should render Default properly 1`] = `
|
|||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"hasStorageAnalyticsAfecFeature": [Function],
|
"hasStorageAnalyticsAfecFeature": [Function],
|
||||||
"hasWriteAccess": [Function],
|
|
||||||
"isAccountReady": [Function],
|
"isAccountReady": [Function],
|
||||||
"isAutoscaleDefaultEnabled": [Function],
|
"isAutoscaleDefaultEnabled": [Function],
|
||||||
"isCopyNotebookPaneEnabled": [Function],
|
"isCopyNotebookPaneEnabled": [Function],
|
||||||
@@ -758,32 +731,6 @@ exports[`Upload Items Pane should render Default properly 1`] = `
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"subscriptionType": [Function],
|
|
||||||
"tableColumnOptionsPane": TableColumnOptionsPane {
|
|
||||||
"allSelected": [Function],
|
|
||||||
"anyColumnSelected": [Function],
|
|
||||||
"availableColumnsLabel": "Available Columns",
|
|
||||||
"canMoveDown": [Function],
|
|
||||||
"canMoveUp": [Function],
|
|
||||||
"canSelectAll": [Function],
|
|
||||||
"columnOptions": [Function],
|
|
||||||
"container": [Circular],
|
|
||||||
"firstFieldHasFocus": [Function],
|
|
||||||
"formErrors": [Function],
|
|
||||||
"formErrorsDetails": [Function],
|
|
||||||
"handleClick": [Function],
|
|
||||||
"id": "tablecolumnoptionspane",
|
|
||||||
"instructionLabel": "Choose the columns and the order in which you want to display them in the table.",
|
|
||||||
"isExecuting": [Function],
|
|
||||||
"isTemplateReady": [Function],
|
|
||||||
"moveDownLabel": "Move Down",
|
|
||||||
"moveUpLabel": "Move Up",
|
|
||||||
"noColumnSelectedWarning": "At least one column should be selected.",
|
|
||||||
"selectedColumnOption": null,
|
|
||||||
"title": [Function],
|
|
||||||
"titleLabel": "Column Options",
|
|
||||||
"visible": [Function],
|
|
||||||
},
|
|
||||||
"tabsManager": TabsManager {
|
"tabsManager": TabsManager {
|
||||||
"activeTab": [Function],
|
"activeTab": [Function],
|
||||||
"openedTabs": [Function],
|
"openedTabs": [Function],
|
||||||
|
|||||||
@@ -197,31 +197,6 @@ exports[`Delete Database Confirmation Pane submit() Should call delete database
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
TableColumnOptionsPane {
|
|
||||||
"allSelected": [Function],
|
|
||||||
"anyColumnSelected": [Function],
|
|
||||||
"availableColumnsLabel": "Available Columns",
|
|
||||||
"canMoveDown": [Function],
|
|
||||||
"canMoveUp": [Function],
|
|
||||||
"canSelectAll": [Function],
|
|
||||||
"columnOptions": [Function],
|
|
||||||
"container": [Circular],
|
|
||||||
"firstFieldHasFocus": [Function],
|
|
||||||
"formErrors": [Function],
|
|
||||||
"formErrorsDetails": [Function],
|
|
||||||
"handleClick": [Function],
|
|
||||||
"id": "tablecolumnoptionspane",
|
|
||||||
"instructionLabel": "Choose the columns and the order in which you want to display them in the table.",
|
|
||||||
"isExecuting": [Function],
|
|
||||||
"isTemplateReady": [Function],
|
|
||||||
"moveDownLabel": "Move Down",
|
|
||||||
"moveUpLabel": "Move Up",
|
|
||||||
"noColumnSelectedWarning": "At least one column should be selected.",
|
|
||||||
"selectedColumnOption": null,
|
|
||||||
"title": [Function],
|
|
||||||
"titleLabel": "Column Options",
|
|
||||||
"visible": [Function],
|
|
||||||
},
|
|
||||||
QuerySelectPane {
|
QuerySelectPane {
|
||||||
"allSelected": [Function],
|
"allSelected": [Function],
|
||||||
"anyColumnSelected": [Function],
|
"anyColumnSelected": [Function],
|
||||||
@@ -584,7 +559,6 @@ exports[`Delete Database Confirmation Pane submit() Should call delete database
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"flight": [Function],
|
|
||||||
"graphStylingPane": GraphStylingPane {
|
"graphStylingPane": GraphStylingPane {
|
||||||
"container": [Circular],
|
"container": [Circular],
|
||||||
"firstFieldHasFocus": [Function],
|
"firstFieldHasFocus": [Function],
|
||||||
@@ -606,7 +580,6 @@ exports[`Delete Database Confirmation Pane submit() Should call delete database
|
|||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"hasStorageAnalyticsAfecFeature": [Function],
|
"hasStorageAnalyticsAfecFeature": [Function],
|
||||||
"hasWriteAccess": [Function],
|
|
||||||
"isAccountReady": [Function],
|
"isAccountReady": [Function],
|
||||||
"isAutoscaleDefaultEnabled": [Function],
|
"isAutoscaleDefaultEnabled": [Function],
|
||||||
"isCopyNotebookPaneEnabled": [Function],
|
"isCopyNotebookPaneEnabled": [Function],
|
||||||
@@ -763,32 +736,6 @@ exports[`Delete Database Confirmation Pane submit() Should call delete database
|
|||||||
"title": [Function],
|
"title": [Function],
|
||||||
"visible": [Function],
|
"visible": [Function],
|
||||||
},
|
},
|
||||||
"subscriptionType": [Function],
|
|
||||||
"tableColumnOptionsPane": TableColumnOptionsPane {
|
|
||||||
"allSelected": [Function],
|
|
||||||
"anyColumnSelected": [Function],
|
|
||||||
"availableColumnsLabel": "Available Columns",
|
|
||||||
"canMoveDown": [Function],
|
|
||||||
"canMoveUp": [Function],
|
|
||||||
"canSelectAll": [Function],
|
|
||||||
"columnOptions": [Function],
|
|
||||||
"container": [Circular],
|
|
||||||
"firstFieldHasFocus": [Function],
|
|
||||||
"formErrors": [Function],
|
|
||||||
"formErrorsDetails": [Function],
|
|
||||||
"handleClick": [Function],
|
|
||||||
"id": "tablecolumnoptionspane",
|
|
||||||
"instructionLabel": "Choose the columns and the order in which you want to display them in the table.",
|
|
||||||
"isExecuting": [Function],
|
|
||||||
"isTemplateReady": [Function],
|
|
||||||
"moveDownLabel": "Move Down",
|
|
||||||
"moveUpLabel": "Move Up",
|
|
||||||
"noColumnSelectedWarning": "At least one column should be selected.",
|
|
||||||
"selectedColumnOption": null,
|
|
||||||
"title": [Function],
|
|
||||||
"titleLabel": "Column Options",
|
|
||||||
"visible": [Function],
|
|
||||||
},
|
|
||||||
"tabsManager": TabsManager {
|
"tabsManager": TabsManager {
|
||||||
"activeTab": [Function],
|
"activeTab": [Function],
|
||||||
"openedTabs": [Function],
|
"openedTabs": [Function],
|
||||||
|
|||||||
@@ -1,12 +1,8 @@
|
|||||||
import _ from "underscore";
|
|
||||||
import Q from "q";
|
import Q from "q";
|
||||||
import * as DataTableUtilities from "./DataTableUtilities";
|
|
||||||
import * as DataTableOperations from "./DataTableOperations";
|
|
||||||
import TableEntityListViewModel from "./TableEntityListViewModel";
|
|
||||||
import * as Entities from "../Entities";
|
|
||||||
import * as ViewModels from "../../../Contracts/ViewModels";
|
|
||||||
import * as TableColumnOptionsPane from "../../Panes/Tables/TableColumnOptionsPane";
|
|
||||||
import Explorer from "../../Explorer";
|
import Explorer from "../../Explorer";
|
||||||
|
import * as Entities from "../Entities";
|
||||||
|
import * as DataTableUtilities from "./DataTableUtilities";
|
||||||
|
import TableEntityListViewModel from "./TableEntityListViewModel";
|
||||||
|
|
||||||
export default class TableCommands {
|
export default class TableCommands {
|
||||||
// Command Ids
|
// Command Ids
|
||||||
@@ -92,64 +88,6 @@ export default class TableCommands {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
public customizeColumnsCommand(viewModel: TableEntityListViewModel): Q.Promise<any> {
|
|
||||||
var table: DataTables.DataTable = viewModel.table;
|
|
||||||
var displayedColumnNames: string[] = DataTableOperations.getDataTableHeaders(table);
|
|
||||||
var columnsCount: number = displayedColumnNames.length;
|
|
||||||
var currentOrder: number[] = DataTableOperations.getInitialOrder(columnsCount);
|
|
||||||
//Debug.assert(!!table && !!currentOrder && displayedColumnNames.length === currentOrder.length);
|
|
||||||
|
|
||||||
var currentSettings: boolean[];
|
|
||||||
try {
|
|
||||||
currentSettings = currentOrder.map((value: number, index: number) => {
|
|
||||||
return table.column(index).visible();
|
|
||||||
});
|
|
||||||
} catch (err) {
|
|
||||||
// Error
|
|
||||||
}
|
|
||||||
|
|
||||||
let parameters: TableColumnOptionsPane.IColumnSetting = <TableColumnOptionsPane.IColumnSetting>{
|
|
||||||
columnNames: displayedColumnNames,
|
|
||||||
order: currentOrder,
|
|
||||||
visible: currentSettings,
|
|
||||||
};
|
|
||||||
|
|
||||||
this._container.tableColumnOptionsPane.tableViewModel = viewModel;
|
|
||||||
this._container.tableColumnOptionsPane.parameters = parameters;
|
|
||||||
this._container.tableColumnOptionsPane.open();
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public reorderColumnsBasedOnSelectedEntities(viewModel: TableEntityListViewModel): Q.Promise<boolean> {
|
|
||||||
var selected = viewModel.selected();
|
|
||||||
if (!selected || !selected.length) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
var table = viewModel.table;
|
|
||||||
var currentColumnNames: string[] = DataTableOperations.getDataTableHeaders(table);
|
|
||||||
var headersCount: number = currentColumnNames.length;
|
|
||||||
|
|
||||||
var headersUnion: string[] = DataTableUtilities.getPropertyIntersectionFromTableEntities(
|
|
||||||
selected,
|
|
||||||
viewModel.queryTablesTab.container.isPreferredApiCassandra()
|
|
||||||
);
|
|
||||||
|
|
||||||
// An array with elements representing indexes of selected entities' header union out of initial headers.
|
|
||||||
var orderOfLeftHeaders: number[] = headersUnion.map((item: string) => currentColumnNames.indexOf(item));
|
|
||||||
|
|
||||||
// An array with elements representing initial order of the table.
|
|
||||||
var initialOrder: number[] = DataTableOperations.getInitialOrder(headersCount);
|
|
||||||
|
|
||||||
// An array with elements representing indexes of headers not present in selected entities' header union.
|
|
||||||
var orderOfRightHeaders: number[] = _.difference(initialOrder, orderOfLeftHeaders);
|
|
||||||
|
|
||||||
// This will be the target order, with headers in selected entities on the left while others on the right, both in the initial order, respectively.
|
|
||||||
var targetOrder: number[] = orderOfLeftHeaders.concat(orderOfRightHeaders);
|
|
||||||
|
|
||||||
return DataTableOperations.reorderColumns(table, targetOrder);
|
|
||||||
}
|
|
||||||
|
|
||||||
public resetColumns(viewModel: TableEntityListViewModel): void {
|
public resetColumns(viewModel: TableEntityListViewModel): void {
|
||||||
viewModel.reloadTable();
|
viewModel.reloadTable();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,85 +0,0 @@
|
|||||||
<div
|
|
||||||
class="tab-pane flexContainer"
|
|
||||||
data-bind="
|
|
||||||
attr:{
|
|
||||||
id: tabId
|
|
||||||
},
|
|
||||||
visible: isActive"
|
|
||||||
role="tabpanel"
|
|
||||||
>
|
|
||||||
<div class="warningErrorContainer scaleWarningContainer" data-bind="visible: shouldShowStatusBar">
|
|
||||||
<div>
|
|
||||||
<div class="warningErrorContent" data-bind="visible: shouldShowNotificationStatusPrompt">
|
|
||||||
<span><img src="/info_color.svg" alt="Info" /></span>
|
|
||||||
<span class="warningErrorDetailsLinkContainer" data-bind="html: notificationStatusInfo"></span>
|
|
||||||
</div>
|
|
||||||
<div class="warningErrorContent" data-bind="visible: !shouldShowNotificationStatusPrompt()">
|
|
||||||
<span><img src="/warning.svg" alt="Warning" /></span>
|
|
||||||
<span class="warningErrorDetailsLinkContainer" data-bind="html: warningMessage"></span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="tabForm scaleSettingScrollable">
|
|
||||||
<div class="scaleDivison" aria-label="Scale" aria-controls="scaleRegion">
|
|
||||||
<span class="scaleSettingTitle">Scale</span>
|
|
||||||
</div>
|
|
||||||
<div class="freeTierInfoBanner" data-bind="visible: isFreeTierAccount">
|
|
||||||
<span class="freeTierInfoIcon"><img src="/info_color.svg" alt="Info" /></span>
|
|
||||||
<span class="freeTierInfoMessage"
|
|
||||||
>With free tier, you'll get the first 400 RU/s and 5 GB of storage in this account for free. To keep your
|
|
||||||
account free, keep the total RU/s across all resources in the account to 400 RU/s.
|
|
||||||
<a
|
|
||||||
href="https://docs.microsoft.com/en-us/azure/cosmos-db/understand-your-bill#billing-examples-with-free-tier-accounts"
|
|
||||||
target="_blank"
|
|
||||||
>
|
|
||||||
Learn more.</a
|
|
||||||
>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div class="ssTextAllignment" id="scaleRegion">
|
|
||||||
<throughput-input-autopilot-v3
|
|
||||||
params="{
|
|
||||||
testId: testId,
|
|
||||||
class: 'scaleForm dirty',
|
|
||||||
value: throughput,
|
|
||||||
minimum: minRUs,
|
|
||||||
maximum: maxRUThroughputInputLimit,
|
|
||||||
canExceedMaximumValue: canThroughputExceedMaximumValue,
|
|
||||||
step: throughputIncreaseFactor,
|
|
||||||
label: throughputTitle,
|
|
||||||
ariaLabel: throughputAriaLabel,
|
|
||||||
costsVisible: costsVisible,
|
|
||||||
requestUnitsUsageCost: requestUnitsUsageCost,
|
|
||||||
throughputAutoPilotRadioId: throughputAutoPilotRadioId,
|
|
||||||
throughputProvisionedRadioId: throughputProvisionedRadioId,
|
|
||||||
throughputModeRadioName: throughputModeRadioName,
|
|
||||||
isAutoPilotSelected: isAutoPilotSelected,
|
|
||||||
maxAutoPilotThroughputSet: autoPilotThroughput,
|
|
||||||
autoPilotUsageCost: autoPilotUsageCost,
|
|
||||||
canExceedMaximumValue: canExceedMaximumValue,
|
|
||||||
overrideWithAutoPilotSettings: overrideWithAutoPilotSettings,
|
|
||||||
overrideWithProvisionedThroughputSettings: overrideWithProvisionedThroughputSettings,
|
|
||||||
freeTierExceedThroughputWarning: freeTierExceedThroughputWarning
|
|
||||||
}"
|
|
||||||
>
|
|
||||||
</throughput-input-autopilot-v3>
|
|
||||||
|
|
||||||
<div class="estimatedCost" data-bind="visible: costsVisible">
|
|
||||||
<p data-bind="visible: minRUAnotationVisible">
|
|
||||||
<span>Learn more about minimum throughput </span>
|
|
||||||
<a href="https://docs.microsoft.com/azure/cosmos-db/set-throughput" target="_blank">here.</a>
|
|
||||||
</p>
|
|
||||||
<p data-bind="visible: canRequestSupport">
|
|
||||||
<!-- TODO: Replace link with call to the Azure Support blade -->
|
|
||||||
<a href="https://aka.ms/cosmosdbfeedback?subject=Cosmos%20DB%20More%20Throughput%20Request"
|
|
||||||
>Contact support</a
|
|
||||||
>
|
|
||||||
for more than <span data-bind="text: maxRUsText"></span> RU/s
|
|
||||||
</p>
|
|
||||||
<p data-bind="visible: shouldDisplayPortalUsePrompt">
|
|
||||||
Use Data Explorer from Azure Portal to request more than <span data-bind="text: maxRUsText"></span> RU/s
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
@@ -1,489 +0,0 @@
|
|||||||
import * as ko from "knockout";
|
|
||||||
import Q from "q";
|
|
||||||
import DiscardIcon from "../../../images/discard.svg";
|
|
||||||
import SaveIcon from "../../../images/save-cosmos.svg";
|
|
||||||
import * as Constants from "../../Common/Constants";
|
|
||||||
import { updateOffer } from "../../Common/dataAccess/updateOffer";
|
|
||||||
import editable from "../../Common/EditableUtility";
|
|
||||||
import { getErrorMessage, getErrorStack } from "../../Common/ErrorHandlingUtils";
|
|
||||||
import { configContext, Platform } from "../../ConfigContext";
|
|
||||||
import * as DataModels from "../../Contracts/DataModels";
|
|
||||||
import * as ViewModels from "../../Contracts/ViewModels";
|
|
||||||
import * as SharedConstants from "../../Shared/Constants";
|
|
||||||
import { Action } from "../../Shared/Telemetry/TelemetryConstants";
|
|
||||||
import * as TelemetryProcessor from "../../Shared/Telemetry/TelemetryProcessor";
|
|
||||||
import { userContext } from "../../UserContext";
|
|
||||||
import * as AutoPilotUtils from "../../Utils/AutoPilotUtils";
|
|
||||||
import * as PricingUtils from "../../Utils/PricingUtils";
|
|
||||||
import { CommandButtonComponentProps } from "../Controls/CommandButton/CommandButtonComponent";
|
|
||||||
import Explorer from "../Explorer";
|
|
||||||
import template from "./DatabaseSettingsTab.html";
|
|
||||||
import TabsBase from "./TabsBase";
|
|
||||||
|
|
||||||
const updateThroughputBeyondLimitWarningMessage: string = `
|
|
||||||
You are about to request an increase in throughput beyond the pre-allocated capacity.
|
|
||||||
The service will scale out and increase throughput for the selected database.
|
|
||||||
This operation will take 1-3 business days to complete. You can track the status of this request in Notifications.`;
|
|
||||||
|
|
||||||
const updateThroughputDelayedApplyWarningMessage: string = `
|
|
||||||
You are about to request an increase in throughput beyond the pre-allocated capacity.
|
|
||||||
This operation will take some time to complete.`;
|
|
||||||
|
|
||||||
const currentThroughput: (isAutoscale: boolean, throughput: number) => string = (isAutoscale, throughput) =>
|
|
||||||
isAutoscale
|
|
||||||
? `Current autoscale throughput: ${Math.round(throughput / 10)} - ${throughput} RU/s`
|
|
||||||
: `Current manual throughput: ${throughput} RU/s`;
|
|
||||||
|
|
||||||
const throughputApplyShortDelayMessage = (isAutoscale: boolean, throughput: number, databaseName: string) =>
|
|
||||||
`A request to increase the throughput is currently in progress.
|
|
||||||
This operation will take some time to complete.<br />
|
|
||||||
Database: ${databaseName}, ${currentThroughput(isAutoscale, throughput)}`;
|
|
||||||
|
|
||||||
const throughputApplyLongDelayMessage = (isAutoscale: boolean, throughput: number, databaseName: string) =>
|
|
||||||
`A request to increase the throughput is currently in progress.
|
|
||||||
This operation will take 1-3 business days to complete. View the latest status in Notifications.<br />
|
|
||||||
Database: ${databaseName}, ${currentThroughput(isAutoscale, throughput)}`;
|
|
||||||
|
|
||||||
export default class DatabaseSettingsTab extends TabsBase implements ViewModels.WaitsForTemplate {
|
|
||||||
public static readonly component = { name: "database-settings-tab", template };
|
|
||||||
// editables
|
|
||||||
public isAutoPilotSelected: ViewModels.Editable<boolean>;
|
|
||||||
public throughput: ViewModels.Editable<number>;
|
|
||||||
public autoPilotThroughput: ViewModels.Editable<number>;
|
|
||||||
public throughputIncreaseFactor: number = Constants.ClientDefaults.databaseThroughputIncreaseFactor;
|
|
||||||
|
|
||||||
public saveSettingsButton: ViewModels.Button;
|
|
||||||
public discardSettingsChangesButton: ViewModels.Button;
|
|
||||||
|
|
||||||
public canRequestSupport: ko.PureComputed<boolean>;
|
|
||||||
public canThroughputExceedMaximumValue: ko.Computed<boolean>;
|
|
||||||
public costsVisible: ko.Computed<boolean>;
|
|
||||||
public displayedError: ko.Observable<string>;
|
|
||||||
public isFreeTierAccount: ko.Computed<boolean>;
|
|
||||||
public isTemplateReady: ko.Observable<boolean>;
|
|
||||||
public minRUAnotationVisible: ko.Computed<boolean>;
|
|
||||||
public minRUs: ko.Observable<number>;
|
|
||||||
public maxRUs: ko.Observable<number>;
|
|
||||||
public maxRUsText: ko.PureComputed<string>;
|
|
||||||
public maxRUThroughputInputLimit: ko.Computed<number>;
|
|
||||||
public notificationStatusInfo: ko.Observable<string>;
|
|
||||||
public pendingNotification: ko.Observable<DataModels.Notification>;
|
|
||||||
public requestUnitsUsageCost: ko.PureComputed<string>;
|
|
||||||
public autoscaleCost: ko.PureComputed<string>;
|
|
||||||
public shouldShowNotificationStatusPrompt: ko.Computed<boolean>;
|
|
||||||
public shouldDisplayPortalUsePrompt: ko.Computed<boolean>;
|
|
||||||
public shouldShowStatusBar: ko.Computed<boolean>;
|
|
||||||
public throughputTitle: ko.PureComputed<string>;
|
|
||||||
public throughputAriaLabel: ko.PureComputed<string>;
|
|
||||||
public autoPilotUsageCost: ko.PureComputed<string>;
|
|
||||||
public warningMessage: ko.Computed<string>;
|
|
||||||
public canExceedMaximumValue: ko.PureComputed<boolean>;
|
|
||||||
public overrideWithAutoPilotSettings: ko.Computed<boolean>;
|
|
||||||
public overrideWithProvisionedThroughputSettings: ko.Computed<boolean>;
|
|
||||||
public testId: string;
|
|
||||||
public throughputAutoPilotRadioId: string;
|
|
||||||
public throughputProvisionedRadioId: string;
|
|
||||||
public throughputModeRadioName: string;
|
|
||||||
public freeTierExceedThroughputWarning: ko.Computed<string>;
|
|
||||||
|
|
||||||
private _hasProvisioningTypeChanged: ko.Computed<boolean>;
|
|
||||||
private _wasAutopilotOriginallySet: ko.Observable<boolean>;
|
|
||||||
private _offerReplacePending: ko.Observable<boolean>;
|
|
||||||
private container: Explorer;
|
|
||||||
|
|
||||||
constructor(options: ViewModels.TabOptions) {
|
|
||||||
super(options);
|
|
||||||
|
|
||||||
this.container = options.node && (options.node as ViewModels.Database).container;
|
|
||||||
this.canExceedMaximumValue = ko.pureComputed(() => this.container.canExceedMaximumValue());
|
|
||||||
|
|
||||||
// html element ids
|
|
||||||
this.testId = `scaleSettingThroughputValue${this.tabId}`;
|
|
||||||
this.throughputAutoPilotRadioId = `editContainerThroughput-autoPilotRadio${this.tabId}`;
|
|
||||||
this.throughputProvisionedRadioId = `editContainerThroughput-manualRadio${this.tabId}`;
|
|
||||||
this.throughputModeRadioName = `throughputModeRadio${this.tabId}`;
|
|
||||||
|
|
||||||
this.throughput = editable.observable<number>();
|
|
||||||
this._wasAutopilotOriginallySet = ko.observable(false);
|
|
||||||
this.isAutoPilotSelected = editable.observable(false);
|
|
||||||
this.autoPilotThroughput = editable.observable<number>();
|
|
||||||
|
|
||||||
const autoscaleMaxThroughput = this.database?.offer()?.autoscaleMaxThroughput;
|
|
||||||
if (autoscaleMaxThroughput) {
|
|
||||||
if (AutoPilotUtils.isValidAutoPilotThroughput(autoscaleMaxThroughput)) {
|
|
||||||
this._wasAutopilotOriginallySet(true);
|
|
||||||
this.isAutoPilotSelected(true);
|
|
||||||
this.autoPilotThroughput(autoscaleMaxThroughput);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
this._hasProvisioningTypeChanged = ko.pureComputed<boolean>(() => {
|
|
||||||
if (this._wasAutopilotOriginallySet() !== this.isAutoPilotSelected()) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
});
|
|
||||||
|
|
||||||
this.autoPilotUsageCost = ko.pureComputed<string>(() => {
|
|
||||||
const autoPilot = this.autoPilotThroughput();
|
|
||||||
if (!autoPilot) {
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
return PricingUtils.getAutoPilotV3SpendHtml(autoPilot, true /* isDatabaseThroughput */);
|
|
||||||
});
|
|
||||||
|
|
||||||
this.requestUnitsUsageCost = ko.pureComputed(() => {
|
|
||||||
const account = userContext.databaseAccount;
|
|
||||||
if (!account) {
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
const regions =
|
|
||||||
(account &&
|
|
||||||
account.properties &&
|
|
||||||
account.properties.readLocations &&
|
|
||||||
account.properties.readLocations.length) ||
|
|
||||||
1;
|
|
||||||
const multimaster = (account && account.properties && account.properties.enableMultipleWriteLocations) || false;
|
|
||||||
|
|
||||||
let estimatedSpend: string;
|
|
||||||
if (!this.isAutoPilotSelected()) {
|
|
||||||
estimatedSpend = PricingUtils.getEstimatedSpendHtml(
|
|
||||||
// if migrating from autoscale to manual, we use the autoscale RUs value as that is what will be set...
|
|
||||||
this.overrideWithAutoPilotSettings() ? this.autoPilotThroughput() : this.throughput(),
|
|
||||||
userContext.portalEnv,
|
|
||||||
regions,
|
|
||||||
multimaster
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
estimatedSpend = PricingUtils.getEstimatedAutoscaleSpendHtml(
|
|
||||||
this.autoPilotThroughput(),
|
|
||||||
userContext.portalEnv,
|
|
||||||
regions,
|
|
||||||
multimaster
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return estimatedSpend;
|
|
||||||
});
|
|
||||||
|
|
||||||
this.costsVisible = ko.computed(() => {
|
|
||||||
return configContext.platform !== Platform.Emulator;
|
|
||||||
});
|
|
||||||
|
|
||||||
this.shouldDisplayPortalUsePrompt = ko.pureComputed<boolean>(() => configContext.platform === Platform.Hosted);
|
|
||||||
this.canThroughputExceedMaximumValue = ko.pureComputed<boolean>(
|
|
||||||
() => configContext.platform === Platform.Portal && !this.container.isRunningOnNationalCloud()
|
|
||||||
);
|
|
||||||
this.canRequestSupport = ko.pureComputed(() => {
|
|
||||||
if (
|
|
||||||
configContext.platform === Platform.Emulator ||
|
|
||||||
configContext.platform === Platform.Hosted ||
|
|
||||||
this.canThroughputExceedMaximumValue()
|
|
||||||
) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
});
|
|
||||||
|
|
||||||
this.overrideWithAutoPilotSettings = ko.pureComputed(() => {
|
|
||||||
return this._hasProvisioningTypeChanged() && this._wasAutopilotOriginallySet();
|
|
||||||
});
|
|
||||||
|
|
||||||
this.overrideWithProvisionedThroughputSettings = ko.pureComputed(() => {
|
|
||||||
return this._hasProvisioningTypeChanged() && !this._wasAutopilotOriginallySet();
|
|
||||||
});
|
|
||||||
|
|
||||||
this.minRUs = ko.observable<number>(
|
|
||||||
this.database.offer()?.minimumThroughput || this.container.collectionCreationDefaults.throughput.unlimitedmin
|
|
||||||
);
|
|
||||||
|
|
||||||
this.minRUAnotationVisible = ko.computed<boolean>(() => {
|
|
||||||
return PricingUtils.isLargerThanDefaultMinRU(this.minRUs());
|
|
||||||
});
|
|
||||||
|
|
||||||
this.maxRUs = ko.observable<number>(this.container.collectionCreationDefaults.throughput.unlimitedmax);
|
|
||||||
|
|
||||||
this.maxRUThroughputInputLimit = ko.pureComputed<number>(() => {
|
|
||||||
if (configContext.platform === Platform.Hosted) {
|
|
||||||
return SharedConstants.CollectionCreation.DefaultCollectionRUs1Million;
|
|
||||||
}
|
|
||||||
|
|
||||||
return this.maxRUs();
|
|
||||||
});
|
|
||||||
|
|
||||||
this.maxRUsText = ko.pureComputed(() => {
|
|
||||||
return SharedConstants.CollectionCreation.DefaultCollectionRUs1Million.toLocaleString();
|
|
||||||
});
|
|
||||||
|
|
||||||
this.throughputTitle = ko.pureComputed<string>(() => {
|
|
||||||
if (this.isAutoPilotSelected()) {
|
|
||||||
return AutoPilotUtils.getAutoPilotHeaderText();
|
|
||||||
}
|
|
||||||
|
|
||||||
return `Throughput (${this.minRUs().toLocaleString()} - unlimited RU/s)`;
|
|
||||||
});
|
|
||||||
|
|
||||||
this.throughputAriaLabel = ko.pureComputed<string>(() => {
|
|
||||||
return this.throughputTitle() + this.requestUnitsUsageCost();
|
|
||||||
});
|
|
||||||
this.pendingNotification = ko.observable<DataModels.Notification>();
|
|
||||||
this._offerReplacePending = ko.observable<boolean>(!!this.database.offer()?.offerReplacePending);
|
|
||||||
this.notificationStatusInfo = ko.observable<string>("");
|
|
||||||
this.shouldShowNotificationStatusPrompt = ko.computed<boolean>(() => this.notificationStatusInfo().length > 0);
|
|
||||||
this.warningMessage = ko.computed<string>(() => {
|
|
||||||
if (this.overrideWithProvisionedThroughputSettings()) {
|
|
||||||
return AutoPilotUtils.manualToAutoscaleDisclaimer;
|
|
||||||
}
|
|
||||||
|
|
||||||
const offer = this.database.offer();
|
|
||||||
if (offer?.offerReplacePending) {
|
|
||||||
const throughput = offer.manualThroughput || offer.autoscaleMaxThroughput;
|
|
||||||
return throughputApplyShortDelayMessage(this.isAutoPilotSelected(), throughput, this.database.id());
|
|
||||||
}
|
|
||||||
|
|
||||||
if (
|
|
||||||
this.throughput() > SharedConstants.CollectionCreation.DefaultCollectionRUs1Million &&
|
|
||||||
this.canThroughputExceedMaximumValue()
|
|
||||||
) {
|
|
||||||
return updateThroughputBeyondLimitWarningMessage;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.throughput() > this.maxRUs()) {
|
|
||||||
return updateThroughputDelayedApplyWarningMessage;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.pendingNotification()) {
|
|
||||||
const matches: string[] = this.pendingNotification().description.match("Throughput update for (.*) RU/s");
|
|
||||||
const throughput: number = matches.length > 1 && Number(matches[1]);
|
|
||||||
|
|
||||||
if (throughput) {
|
|
||||||
return throughputApplyLongDelayMessage(this.isAutoPilotSelected(), throughput, this.database.id());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return "";
|
|
||||||
});
|
|
||||||
|
|
||||||
this.warningMessage.subscribe((warning: string) => {
|
|
||||||
if (warning.length > 0) {
|
|
||||||
this.notificationStatusInfo("");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
this.shouldShowStatusBar = ko.computed<boolean>(
|
|
||||||
() => this.shouldShowNotificationStatusPrompt() || (this.warningMessage && this.warningMessage().length > 0)
|
|
||||||
);
|
|
||||||
|
|
||||||
this.displayedError = ko.observable<string>("");
|
|
||||||
|
|
||||||
this._setBaseline();
|
|
||||||
|
|
||||||
this.saveSettingsButton = {
|
|
||||||
enabled: ko.computed<boolean>(() => {
|
|
||||||
if (this._hasProvisioningTypeChanged()) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this._offerReplacePending && this._offerReplacePending()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
const isAutoPilot = this.isAutoPilotSelected();
|
|
||||||
const isManual = !this.isAutoPilotSelected();
|
|
||||||
if (isAutoPilot) {
|
|
||||||
if (!AutoPilotUtils.isValidAutoPilotThroughput(this.autoPilotThroughput())) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (this.isAutoPilotSelected.editableIsDirty()) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
if (this.autoPilotThroughput.editableIsDirty()) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (isManual) {
|
|
||||||
if (!this.throughput()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.throughput() < this.minRUs()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (
|
|
||||||
!this.canThroughputExceedMaximumValue() &&
|
|
||||||
this.throughput() > SharedConstants.CollectionCreation.DefaultCollectionRUs1Million
|
|
||||||
) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.throughput.editableIsDirty()) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.isAutoPilotSelected.editableIsDirty()) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}),
|
|
||||||
|
|
||||||
visible: ko.computed<boolean>(() => {
|
|
||||||
return true;
|
|
||||||
}),
|
|
||||||
};
|
|
||||||
|
|
||||||
this.discardSettingsChangesButton = {
|
|
||||||
enabled: ko.computed<boolean>(() => {
|
|
||||||
if (this.throughput.editableIsDirty()) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
if (this.isAutoPilotSelected.editableIsDirty()) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
if (this.autoPilotThroughput.editableIsDirty()) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}),
|
|
||||||
|
|
||||||
visible: ko.computed<boolean>(() => {
|
|
||||||
return true;
|
|
||||||
}),
|
|
||||||
};
|
|
||||||
|
|
||||||
this.isTemplateReady = ko.observable<boolean>(false);
|
|
||||||
|
|
||||||
this.isFreeTierAccount = ko.computed<boolean>(() => {
|
|
||||||
const databaseAccount = userContext.databaseAccount;
|
|
||||||
return databaseAccount?.properties?.enableFreeTier;
|
|
||||||
});
|
|
||||||
|
|
||||||
this.freeTierExceedThroughputWarning = ko.computed<string>(() =>
|
|
||||||
this.isFreeTierAccount()
|
|
||||||
? "Billing will apply if you provision more than 400 RU/s of manual throughput, or if the resource scales beyond 400 RU/s with autoscale."
|
|
||||||
: ""
|
|
||||||
);
|
|
||||||
|
|
||||||
this._buildCommandBarOptions();
|
|
||||||
}
|
|
||||||
|
|
||||||
public onSaveClick = async (): Promise<any> => {
|
|
||||||
this.isExecutionError(false);
|
|
||||||
|
|
||||||
this.isExecuting(true);
|
|
||||||
|
|
||||||
const startKey: number = TelemetryProcessor.traceStart(Action.UpdateSettings, {
|
|
||||||
dataExplorerArea: Constants.Areas.Tab,
|
|
||||||
tabTitle: this.tabTitle(),
|
|
||||||
});
|
|
||||||
|
|
||||||
try {
|
|
||||||
const updateOfferParams: DataModels.UpdateOfferParams = {
|
|
||||||
databaseId: this.database.id(),
|
|
||||||
currentOffer: this.database.offer(),
|
|
||||||
autopilotThroughput: this.isAutoPilotSelected() ? this.autoPilotThroughput() : undefined,
|
|
||||||
manualThroughput: this.isAutoPilotSelected() ? undefined : this.throughput(),
|
|
||||||
};
|
|
||||||
|
|
||||||
if (this._hasProvisioningTypeChanged()) {
|
|
||||||
if (this.isAutoPilotSelected()) {
|
|
||||||
updateOfferParams.migrateToAutoPilot = true;
|
|
||||||
} else {
|
|
||||||
updateOfferParams.migrateToManual = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const updatedOffer: DataModels.Offer = await updateOffer(updateOfferParams);
|
|
||||||
this.database.offer(updatedOffer);
|
|
||||||
this.database.offer.valueHasMutated();
|
|
||||||
this._setBaseline();
|
|
||||||
this._wasAutopilotOriginallySet(this.isAutoPilotSelected());
|
|
||||||
} catch (error) {
|
|
||||||
this.isExecutionError(true);
|
|
||||||
console.error(error);
|
|
||||||
const errorMessage = getErrorMessage(error);
|
|
||||||
this.displayedError(errorMessage);
|
|
||||||
TelemetryProcessor.traceFailure(
|
|
||||||
Action.UpdateSettings,
|
|
||||||
{
|
|
||||||
databaseName: this.database && this.database.id(),
|
|
||||||
|
|
||||||
dataExplorerArea: Constants.Areas.Tab,
|
|
||||||
tabTitle: this.tabTitle(),
|
|
||||||
error: errorMessage,
|
|
||||||
errorStack: getErrorStack(error),
|
|
||||||
},
|
|
||||||
startKey
|
|
||||||
);
|
|
||||||
} finally {
|
|
||||||
this.isExecuting(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
public onRevertClick = (): Q.Promise<any> => {
|
|
||||||
this.throughput.setBaseline(this.throughput.getEditableOriginalValue());
|
|
||||||
this.isAutoPilotSelected.setBaseline(this.isAutoPilotSelected.getEditableOriginalValue());
|
|
||||||
this.autoPilotThroughput.setBaseline(this.autoPilotThroughput.getEditableOriginalValue());
|
|
||||||
|
|
||||||
return Q();
|
|
||||||
};
|
|
||||||
|
|
||||||
public async onActivate(): Promise<void> {
|
|
||||||
super.onActivate();
|
|
||||||
this.database.selectedSubnodeKind(ViewModels.CollectionTabKind.DatabaseSettings);
|
|
||||||
await this.database.loadOffer();
|
|
||||||
}
|
|
||||||
|
|
||||||
private _setBaseline() {
|
|
||||||
const offer = this.database && this.database.offer && this.database.offer();
|
|
||||||
this.isAutoPilotSelected.setBaseline(AutoPilotUtils.isValidAutoPilotThroughput(offer.autoscaleMaxThroughput));
|
|
||||||
this.autoPilotThroughput.setBaseline(offer.autoscaleMaxThroughput);
|
|
||||||
this.throughput.setBaseline(offer.manualThroughput);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected getTabsButtons(): CommandButtonComponentProps[] {
|
|
||||||
const buttons: CommandButtonComponentProps[] = [];
|
|
||||||
const label = "Save";
|
|
||||||
if (this.saveSettingsButton.visible()) {
|
|
||||||
buttons.push({
|
|
||||||
iconSrc: SaveIcon,
|
|
||||||
iconAlt: label,
|
|
||||||
onCommandClick: this.onSaveClick,
|
|
||||||
commandButtonLabel: label,
|
|
||||||
ariaLabel: label,
|
|
||||||
hasPopup: false,
|
|
||||||
disabled: !this.saveSettingsButton.enabled(),
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.discardSettingsChangesButton.visible()) {
|
|
||||||
const label = "Discard";
|
|
||||||
buttons.push({
|
|
||||||
iconSrc: DiscardIcon,
|
|
||||||
iconAlt: label,
|
|
||||||
onCommandClick: this.onRevertClick,
|
|
||||||
commandButtonLabel: label,
|
|
||||||
ariaLabel: label,
|
|
||||||
hasPopup: false,
|
|
||||||
disabled: !this.discardSettingsChangesButton.enabled(),
|
|
||||||
});
|
|
||||||
}
|
|
||||||
return buttons;
|
|
||||||
}
|
|
||||||
|
|
||||||
private _buildCommandBarOptions(): void {
|
|
||||||
ko.computed(() =>
|
|
||||||
ko.toJSON([
|
|
||||||
this.saveSettingsButton.visible,
|
|
||||||
this.saveSettingsButton.enabled,
|
|
||||||
this.discardSettingsChangesButton.visible,
|
|
||||||
this.discardSettingsChangesButton.enabled,
|
|
||||||
])
|
|
||||||
).subscribe(() => this.updateNavbarWithTabsButtons());
|
|
||||||
this.updateNavbarWithTabsButtons();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,39 +1,36 @@
|
|||||||
import * as _ from "underscore";
|
import { stringifyNotebook, toJS } from "@nteract/commutable";
|
||||||
import * as Q from "q";
|
|
||||||
import * as ko from "knockout";
|
import * as ko from "knockout";
|
||||||
|
import * as Q from "q";
|
||||||
import * as ViewModels from "../../Contracts/ViewModels";
|
import * as _ from "underscore";
|
||||||
import * as DataModels from "../../Contracts/DataModels";
|
|
||||||
import TabsBase from "./TabsBase";
|
|
||||||
|
|
||||||
import NewCellIcon from "../../../images/notebook/Notebook-insert-cell.svg";
|
|
||||||
import CutIcon from "../../../images/notebook/Notebook-cut.svg";
|
|
||||||
import CopyIcon from "../../../images/notebook/Notebook-copy.svg";
|
|
||||||
import PasteIcon from "../../../images/notebook/Notebook-paste.svg";
|
|
||||||
import RunIcon from "../../../images/notebook/Notebook-run.svg";
|
|
||||||
import RunAllIcon from "../../../images/notebook/Notebook-run-all.svg";
|
|
||||||
import RestartIcon from "../../../images/notebook/Notebook-restart.svg";
|
|
||||||
import SaveIcon from "../../../images/save-cosmos.svg";
|
|
||||||
import ClearAllOutputsIcon from "../../../images/notebook/Notebook-clear-all-outputs.svg";
|
import ClearAllOutputsIcon from "../../../images/notebook/Notebook-clear-all-outputs.svg";
|
||||||
import InterruptKernelIcon from "../../../images/notebook/Notebook-stop.svg";
|
import CopyIcon from "../../../images/notebook/Notebook-copy.svg";
|
||||||
import KillKernelIcon from "../../../images/notebook/Notebook-stop.svg";
|
import CutIcon from "../../../images/notebook/Notebook-cut.svg";
|
||||||
import * as TelemetryProcessor from "../../Shared/Telemetry/TelemetryProcessor";
|
import NewCellIcon from "../../../images/notebook/Notebook-insert-cell.svg";
|
||||||
import { Action, ActionModifiers, Source } from "../../Shared/Telemetry/TelemetryConstants";
|
import PasteIcon from "../../../images/notebook/Notebook-paste.svg";
|
||||||
|
import RestartIcon from "../../../images/notebook/Notebook-restart.svg";
|
||||||
|
import RunAllIcon from "../../../images/notebook/Notebook-run-all.svg";
|
||||||
|
import RunIcon from "../../../images/notebook/Notebook-run.svg";
|
||||||
|
import { default as InterruptKernelIcon, default as KillKernelIcon } from "../../../images/notebook/Notebook-stop.svg";
|
||||||
|
import SaveIcon from "../../../images/save-cosmos.svg";
|
||||||
import { Areas, ArmApiVersions } from "../../Common/Constants";
|
import { Areas, ArmApiVersions } from "../../Common/Constants";
|
||||||
|
import { configContext } from "../../ConfigContext";
|
||||||
|
import * as DataModels from "../../Contracts/DataModels";
|
||||||
|
import * as ViewModels from "../../Contracts/ViewModels";
|
||||||
|
import { trackEvent } from "../../Shared/appInsights";
|
||||||
|
import { Action, ActionModifiers, Source } from "../../Shared/Telemetry/TelemetryConstants";
|
||||||
|
import * as TelemetryProcessor from "../../Shared/Telemetry/TelemetryProcessor";
|
||||||
|
import { userContext } from "../../UserContext";
|
||||||
|
import * as NotebookConfigurationUtils from "../../Utils/NotebookConfigurationUtils";
|
||||||
|
import * as NotificationConsoleUtils from "../../Utils/NotificationConsoleUtils";
|
||||||
|
import { CommandButtonComponentProps } from "../Controls/CommandButton/CommandButtonComponent";
|
||||||
|
import Explorer from "../Explorer";
|
||||||
import * as CommandBarComponentButtonFactory from "../Menus/CommandBar/CommandBarComponentButtonFactory";
|
import * as CommandBarComponentButtonFactory from "../Menus/CommandBar/CommandBarComponentButtonFactory";
|
||||||
import { ConsoleDataType } from "../Menus/NotificationConsole/NotificationConsoleComponent";
|
import { ConsoleDataType } from "../Menus/NotificationConsole/NotificationConsoleComponent";
|
||||||
import * as NotificationConsoleUtils from "../../Utils/NotificationConsoleUtils";
|
|
||||||
import { NotebookComponentAdapter } from "../Notebook/NotebookComponent/NotebookComponentAdapter";
|
|
||||||
import * as NotebookConfigurationUtils from "../../Utils/NotebookConfigurationUtils";
|
|
||||||
import { KernelSpecsDisplay, NotebookClientV2 } from "../Notebook/NotebookClientV2";
|
import { KernelSpecsDisplay, NotebookClientV2 } from "../Notebook/NotebookClientV2";
|
||||||
import { configContext } from "../../ConfigContext";
|
import { NotebookComponentAdapter } from "../Notebook/NotebookComponent/NotebookComponentAdapter";
|
||||||
import Explorer from "../Explorer";
|
|
||||||
import { NotebookContentItem } from "../Notebook/NotebookContentItem";
|
import { NotebookContentItem } from "../Notebook/NotebookContentItem";
|
||||||
import { CommandButtonComponentProps } from "../Controls/CommandButton/CommandButtonComponent";
|
|
||||||
import { toJS, stringifyNotebook } from "@nteract/commutable";
|
|
||||||
import { appInsights } from "../../Shared/appInsights";
|
|
||||||
import { userContext } from "../../UserContext";
|
|
||||||
import template from "./NotebookV2Tab.html";
|
import template from "./NotebookV2Tab.html";
|
||||||
|
import TabsBase from "./TabsBase";
|
||||||
|
|
||||||
export interface NotebookTabOptions extends ViewModels.TabOptions {
|
export interface NotebookTabOptions extends ViewModels.TabOptions {
|
||||||
account: DataModels.DatabaseAccount;
|
account: DataModels.DatabaseAccount;
|
||||||
@@ -428,7 +425,7 @@ export default class NotebookTabV2 extends TabsBase {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
appInsights.trackEvent(
|
trackEvent(
|
||||||
{ name: "SparkPoolSelected" },
|
{ name: "SparkPoolSelected" },
|
||||||
{
|
{
|
||||||
subscriptionId: userContext.subscriptionId,
|
subscriptionId: userContext.subscriptionId,
|
||||||
|
|||||||
@@ -1,23 +1,22 @@
|
|||||||
import * as _ from "underscore";
|
|
||||||
import * as ko from "knockout";
|
import * as ko from "knockout";
|
||||||
import * as ViewModels from "../../Contracts/ViewModels";
|
import * as _ from "underscore";
|
||||||
import * as Constants from "../../Common/Constants";
|
import * as Constants from "../../Common/Constants";
|
||||||
|
import { readCollections } from "../../Common/dataAccess/readCollections";
|
||||||
|
import { readDatabaseOffer } from "../../Common/dataAccess/readDatabaseOffer";
|
||||||
|
import { getErrorMessage, getErrorStack } from "../../Common/ErrorHandlingUtils";
|
||||||
|
import * as Logger from "../../Common/Logger";
|
||||||
|
import { fetchPortalNotifications } from "../../Common/PortalNotifications";
|
||||||
import * as DataModels from "../../Contracts/DataModels";
|
import * as DataModels from "../../Contracts/DataModels";
|
||||||
|
import * as ViewModels from "../../Contracts/ViewModels";
|
||||||
|
import { IJunoResponse, JunoClient } from "../../Juno/JunoClient";
|
||||||
import { Action, ActionModifiers } from "../../Shared/Telemetry/TelemetryConstants";
|
import { Action, ActionModifiers } from "../../Shared/Telemetry/TelemetryConstants";
|
||||||
import DatabaseSettingsTab from "../Tabs/DatabaseSettingsTab";
|
import * as TelemetryProcessor from "../../Shared/Telemetry/TelemetryProcessor";
|
||||||
|
import { userContext } from "../../UserContext";
|
||||||
|
import * as NotificationConsoleUtils from "../../Utils/NotificationConsoleUtils";
|
||||||
|
import Explorer from "../Explorer";
|
||||||
|
import { ConsoleDataType } from "../Menus/NotificationConsole/NotificationConsoleComponent";
|
||||||
import { DatabaseSettingsTabV2 } from "../Tabs/SettingsTabV2";
|
import { DatabaseSettingsTabV2 } from "../Tabs/SettingsTabV2";
|
||||||
import Collection from "./Collection";
|
import Collection from "./Collection";
|
||||||
import * as TelemetryProcessor from "../../Shared/Telemetry/TelemetryProcessor";
|
|
||||||
import * as NotificationConsoleUtils from "../../Utils/NotificationConsoleUtils";
|
|
||||||
import { ConsoleDataType } from "../Menus/NotificationConsole/NotificationConsoleComponent";
|
|
||||||
import * as Logger from "../../Common/Logger";
|
|
||||||
import Explorer from "../Explorer";
|
|
||||||
import { readCollections } from "../../Common/dataAccess/readCollections";
|
|
||||||
import { JunoClient, IJunoResponse } from "../../Juno/JunoClient";
|
|
||||||
import { userContext } from "../../UserContext";
|
|
||||||
import { readDatabaseOffer } from "../../Common/dataAccess/readDatabaseOffer";
|
|
||||||
import { fetchPortalNotifications } from "../../Common/PortalNotifications";
|
|
||||||
import { getErrorMessage, getErrorStack } from "../../Common/ErrorHandlingUtils";
|
|
||||||
|
|
||||||
export default class Database implements ViewModels.Database {
|
export default class Database implements ViewModels.Database {
|
||||||
public nodeKind: string;
|
public nodeKind: string;
|
||||||
@@ -58,18 +57,13 @@ export default class Database implements ViewModels.Database {
|
|||||||
});
|
});
|
||||||
|
|
||||||
const pendingNotificationsPromise: Promise<DataModels.Notification> = this.getPendingThroughputSplitNotification();
|
const pendingNotificationsPromise: Promise<DataModels.Notification> = this.getPendingThroughputSplitNotification();
|
||||||
const useDatabaseSettingsTabV1 = userContext.features.enableDatabaseSettingsTabV1;
|
const tabKind = ViewModels.CollectionTabKind.DatabaseSettingsV2;
|
||||||
const tabKind: ViewModels.CollectionTabKind = useDatabaseSettingsTabV1
|
|
||||||
? ViewModels.CollectionTabKind.DatabaseSettings
|
|
||||||
: ViewModels.CollectionTabKind.DatabaseSettingsV2;
|
|
||||||
const matchingTabs = this.container.tabsManager.getTabs(tabKind, (tab) => tab.node?.id() === this.id());
|
const matchingTabs = this.container.tabsManager.getTabs(tabKind, (tab) => tab.node?.id() === this.id());
|
||||||
let settingsTab: DatabaseSettingsTab | DatabaseSettingsTabV2 = useDatabaseSettingsTabV1
|
let settingsTab = matchingTabs?.[0] as DatabaseSettingsTabV2;
|
||||||
? (matchingTabs?.[0] as DatabaseSettingsTab)
|
|
||||||
: (matchingTabs?.[0] as DatabaseSettingsTabV2);
|
|
||||||
if (!settingsTab) {
|
if (!settingsTab) {
|
||||||
const startKey: number = TelemetryProcessor.traceStart(Action.Tab, {
|
const startKey: number = TelemetryProcessor.traceStart(Action.Tab, {
|
||||||
databaseName: this.id(),
|
databaseName: this.id(),
|
||||||
|
|
||||||
dataExplorerArea: Constants.Areas.Tab,
|
dataExplorerArea: Constants.Areas.Tab,
|
||||||
tabTitle: "Scale",
|
tabTitle: "Scale",
|
||||||
});
|
});
|
||||||
@@ -77,9 +71,7 @@ export default class Database implements ViewModels.Database {
|
|||||||
(data: any) => {
|
(data: any) => {
|
||||||
const pendingNotification: DataModels.Notification = data?.[0];
|
const pendingNotification: DataModels.Notification = data?.[0];
|
||||||
const tabOptions: ViewModels.TabOptions = {
|
const tabOptions: ViewModels.TabOptions = {
|
||||||
tabKind: useDatabaseSettingsTabV1
|
tabKind,
|
||||||
? ViewModels.CollectionTabKind.DatabaseSettings
|
|
||||||
: ViewModels.CollectionTabKind.DatabaseSettingsV2,
|
|
||||||
title: "Scale",
|
title: "Scale",
|
||||||
tabPath: "",
|
tabPath: "",
|
||||||
node: this,
|
node: this,
|
||||||
@@ -90,9 +82,7 @@ export default class Database implements ViewModels.Database {
|
|||||||
onLoadStartKey: startKey,
|
onLoadStartKey: startKey,
|
||||||
onUpdateTabsButtons: this.container.onUpdateTabsButtons,
|
onUpdateTabsButtons: this.container.onUpdateTabsButtons,
|
||||||
};
|
};
|
||||||
settingsTab = useDatabaseSettingsTabV1
|
settingsTab = new DatabaseSettingsTabV2(tabOptions);
|
||||||
? new DatabaseSettingsTab(tabOptions)
|
|
||||||
: new DatabaseSettingsTabV2(tabOptions);
|
|
||||||
settingsTab.pendingNotification(pendingNotification);
|
settingsTab.pendingNotification(pendingNotification);
|
||||||
this.container.tabsManager.activateNewTab(settingsTab);
|
this.container.tabsManager.activateNewTab(settingsTab);
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
{
|
{
|
||||||
"DedicatedGatewayDescription": "Provision a dedicated gateway cluster for your Azure Cosmos DB account. A dedicated gateway is compute that is a front-end to data in your Azure Cosmos DB account. Your dedicated gateway automatically includes the integrated cache, which can improve read performance.",
|
"DedicatedGatewayDescription": "Provision a dedicated gateway cluster for your Azure Cosmos DB account. A dedicated gateway is compute that is a front-end to data in your Azure Cosmos DB account. Your dedicated gateway automatically includes the integrated cache, which can improve read performance.",
|
||||||
"DedicatedGateway": "Dedicated Gateway",
|
"DedicatedGateway": "Dedicated Gateway",
|
||||||
"Enable": "Enable",
|
"Provisioned": "Provisioned",
|
||||||
"Disable": "Disable",
|
"Deprovisioned": "Deprovisioned",
|
||||||
"LearnAboutDedicatedGateway": "Learn more about dedicated gateway.",
|
"LearnAboutDedicatedGateway": "Learn more about dedicated gateway.",
|
||||||
"DeprovisioningDetailsText": "Learn more about deprovisioning the dedicated gateway.",
|
"DeprovisioningDetailsText": "Learn more about deprovisioning the dedicated gateway.",
|
||||||
"DedicatedGatewayPricing": "Learn more about dedicated gateway pricing.",
|
"DedicatedGatewayPricing": "Learn more about dedicated gateway pricing.",
|
||||||
|
|||||||
@@ -234,7 +234,6 @@ const App: React.FunctionComponent = () => {
|
|||||||
<div data-bind='component: { name: "graph-styling-pane", params: { data: graphStylingPane} }' />
|
<div data-bind='component: { name: "graph-styling-pane", params: { data: graphStylingPane} }' />
|
||||||
<div data-bind='component: { name: "table-add-entity-pane", params: { data: addTableEntityPane} }' />
|
<div data-bind='component: { name: "table-add-entity-pane", params: { data: addTableEntityPane} }' />
|
||||||
<div data-bind='component: { name: "table-edit-entity-pane", params: { data: editTableEntityPane} }' />
|
<div data-bind='component: { name: "table-edit-entity-pane", params: { data: editTableEntityPane} }' />
|
||||||
<div data-bind='component: { name: "table-column-options-pane", params: { data: tableColumnOptionsPane} }' />
|
|
||||||
<div data-bind='component: { name: "table-query-select-pane", params: { data: querySelectPane} }' />
|
<div data-bind='component: { name: "table-query-select-pane", params: { data: querySelectPane} }' />
|
||||||
<div data-bind='component: { name: "cassandra-add-collection-pane", params: { data: cassandraAddCollectionPane} }' />
|
<div data-bind='component: { name: "cassandra-add-collection-pane", params: { data: cassandraAddCollectionPane} }' />
|
||||||
<div data-bind='component: { name: "string-input-pane", params: { data: stringInputPane} }' />
|
<div data-bind='component: { name: "string-input-pane", params: { data: stringInputPane} }' />
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ export type Features = {
|
|||||||
readonly canExceedMaximumValue: boolean;
|
readonly canExceedMaximumValue: boolean;
|
||||||
readonly cosmosdb: boolean;
|
readonly cosmosdb: boolean;
|
||||||
readonly enableChangeFeedPolicy: boolean;
|
readonly enableChangeFeedPolicy: boolean;
|
||||||
readonly enableDatabaseSettingsTabV1: boolean;
|
|
||||||
readonly enableFixedCollectionWithSharedThroughput: boolean;
|
readonly enableFixedCollectionWithSharedThroughput: boolean;
|
||||||
readonly enableKOPanel: boolean;
|
readonly enableKOPanel: boolean;
|
||||||
readonly enableNotebooks: boolean;
|
readonly enableNotebooks: boolean;
|
||||||
@@ -40,7 +39,6 @@ export function extractFeatures(given = new URLSearchParams()): Features {
|
|||||||
canExceedMaximumValue: "true" === get("canexceedmaximumvalue"),
|
canExceedMaximumValue: "true" === get("canexceedmaximumvalue"),
|
||||||
cosmosdb: "true" === get("cosmosdb"),
|
cosmosdb: "true" === get("cosmosdb"),
|
||||||
enableChangeFeedPolicy: "true" === get("enablechangefeedpolicy"),
|
enableChangeFeedPolicy: "true" === get("enablechangefeedpolicy"),
|
||||||
enableDatabaseSettingsTabV1: "true" === get("enabledbsettingsv1"),
|
|
||||||
enableFixedCollectionWithSharedThroughput: "true" === get("enablefixedcollectionwithsharedthroughput"),
|
enableFixedCollectionWithSharedThroughput: "true" === get("enablefixedcollectionwithsharedthroughput"),
|
||||||
enableKOPanel: "true" === get("enablekopanel"),
|
enableKOPanel: "true" === get("enablekopanel"),
|
||||||
enableNotebooks: "true" === get("enablenotebooks"),
|
enableNotebooks: "true" === get("enablenotebooks"),
|
||||||
|
|||||||
@@ -23,9 +23,11 @@ const loadTranslationFile = async (className: string): Promise<void> => {
|
|||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
let translations: any;
|
let translations: any;
|
||||||
try {
|
try {
|
||||||
translations = await import(`../Localization/${language}/${fileName}`);
|
translations = await import(
|
||||||
|
/* webpackChunkName: "Localization-[request]" */ `../Localization/${language}/${fileName}`
|
||||||
|
);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
translations = await import(`../Localization/en/${fileName}`);
|
translations = await import(/* webpackChunkName: "Localization-en-[request]" */ `../Localization/en/${fileName}`);
|
||||||
}
|
}
|
||||||
i18n.addResourceBundle(language, className, translations.default, true);
|
i18n.addResourceBundle(language, className, translations.default, true);
|
||||||
};
|
};
|
||||||
@@ -39,12 +41,12 @@ const getDescriptor = async (selfServeType: SelfServeType): Promise<SelfServeDes
|
|||||||
switch (selfServeType) {
|
switch (selfServeType) {
|
||||||
case SelfServeType.example: {
|
case SelfServeType.example: {
|
||||||
const SelfServeExample = await import(/* webpackChunkName: "SelfServeExample" */ "./Example/SelfServeExample");
|
const SelfServeExample = await import(/* webpackChunkName: "SelfServeExample" */ "./Example/SelfServeExample");
|
||||||
await loadTranslations(SelfServeExample.default.name);
|
await loadTranslations("SelfServeExample");
|
||||||
return new SelfServeExample.default().toSelfServeDescriptor();
|
return new SelfServeExample.default().toSelfServeDescriptor();
|
||||||
}
|
}
|
||||||
case SelfServeType.sqlx: {
|
case SelfServeType.sqlx: {
|
||||||
const SqlX = await import(/* webpackChunkName: "SqlX" */ "./SqlX/SqlX");
|
const SqlX = await import(/* webpackChunkName: "SqlX" */ "./SqlX/SqlX");
|
||||||
await loadTranslations(SqlX.default.name);
|
await loadTranslations("SqlX");
|
||||||
return new SqlX.default().toSelfServeDescriptor();
|
return new SqlX.default().toSelfServeDescriptor();
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
@@ -107,6 +109,16 @@ const handleMessage = async (event: MessageEvent): Promise<void> => {
|
|||||||
subscriptionId: inputs.subscriptionId,
|
subscriptionId: inputs.subscriptionId,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (i18n.isInitialized) {
|
||||||
|
await displaySelfServeComponent(selfServeType);
|
||||||
|
} else {
|
||||||
|
i18n.on("initialized", async () => {
|
||||||
|
await displaySelfServeComponent(selfServeType);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const displaySelfServeComponent = async (selfServeType: SelfServeType): Promise<void> => {
|
||||||
const descriptor = await getDescriptor(selfServeType);
|
const descriptor = await getDescriptor(selfServeType);
|
||||||
ReactDOM.render(renderComponent(descriptor), document.getElementById("selfServeContent"));
|
ReactDOM.render(renderComponent(descriptor), document.getElementById("selfServeContent"));
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,69 +1,24 @@
|
|||||||
import { sendMessage } from "../Common/MessageHandler";
|
|
||||||
import { configContext } from "../ConfigContext";
|
|
||||||
import { SelfServeMessageTypes } from "../Contracts/SelfServeContracts";
|
import { SelfServeMessageTypes } from "../Contracts/SelfServeContracts";
|
||||||
import { appInsights } from "../Shared/appInsights";
|
|
||||||
import { Action, ActionModifiers } from "../Shared/Telemetry/TelemetryConstants";
|
import { Action, ActionModifiers } from "../Shared/Telemetry/TelemetryConstants";
|
||||||
import { userContext } from "../UserContext";
|
import { trace, traceCancel, traceFailure, traceStart, traceSuccess } from "../Shared/Telemetry/TelemetryProcessor";
|
||||||
import { SelfServeTelemetryMessage } from "./SelfServeTypes";
|
import { SelfServeTelemetryMessage } from "./SelfServeTypes";
|
||||||
|
|
||||||
const action = Action.SelfServe;
|
export const selfServeTrace = (data: SelfServeTelemetryMessage): void => {
|
||||||
|
trace(Action.SelfServe, ActionModifiers.Mark, data, SelfServeMessageTypes.TelemetryInfo);
|
||||||
export const trace = (data: SelfServeTelemetryMessage): void => {
|
|
||||||
sendSelfServeTelemetryMessage(ActionModifiers.Mark, data);
|
|
||||||
appInsights.trackEvent({ name: Action[action] }, decorateData(data, ActionModifiers.Mark));
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const traceStart = (data: SelfServeTelemetryMessage): number => {
|
export const selfServeTraceStart = (data: SelfServeTelemetryMessage): number => {
|
||||||
const timestamp: number = Date.now();
|
return traceStart(Action.SelfServe, data, SelfServeMessageTypes.TelemetryInfo);
|
||||||
sendSelfServeTelemetryMessage(ActionModifiers.Start, data);
|
|
||||||
appInsights.startTrackEvent(Action[action]);
|
|
||||||
return timestamp;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const traceSuccess = (data: SelfServeTelemetryMessage, timestamp?: number): void => {
|
export const selfServeTraceSuccess = (data: SelfServeTelemetryMessage, timestamp?: number): void => {
|
||||||
sendSelfServeTelemetryMessage(ActionModifiers.Success, data, timestamp || Date.now());
|
traceSuccess(Action.SelfServe, data, timestamp, SelfServeMessageTypes.TelemetryInfo);
|
||||||
appInsights.stopTrackEvent(Action[action], decorateData(data, ActionModifiers.Success));
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const traceFailure = (data: SelfServeTelemetryMessage, timestamp?: number): void => {
|
export const selfServeTraceFailure = (data: SelfServeTelemetryMessage, timestamp?: number): void => {
|
||||||
sendSelfServeTelemetryMessage(ActionModifiers.Failed, data, timestamp || Date.now());
|
traceFailure(Action.SelfServe, data, timestamp, SelfServeMessageTypes.TelemetryInfo);
|
||||||
appInsights.stopTrackEvent(Action[action], decorateData(data, ActionModifiers.Failed));
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const traceCancel = (data: SelfServeTelemetryMessage, timestamp?: number): void => {
|
export const selfServeTraceCancel = (data: SelfServeTelemetryMessage, timestamp?: number): void => {
|
||||||
sendSelfServeTelemetryMessage(ActionModifiers.Cancel, data, timestamp || Date.now());
|
traceCancel(Action.SelfServe, data, timestamp, SelfServeMessageTypes.TelemetryInfo);
|
||||||
appInsights.stopTrackEvent(Action[action], decorateData(data, ActionModifiers.Cancel));
|
|
||||||
};
|
|
||||||
|
|
||||||
const sendSelfServeTelemetryMessage = (
|
|
||||||
actionModifier: string,
|
|
||||||
data: SelfServeTelemetryMessage,
|
|
||||||
timeStamp?: number
|
|
||||||
): void => {
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
||||||
const dataToSend: any = {
|
|
||||||
type: SelfServeMessageTypes.TelemetryInfo,
|
|
||||||
data: {
|
|
||||||
action: Action[action],
|
|
||||||
actionModifier: actionModifier,
|
|
||||||
data: JSON.stringify(decorateData(data)),
|
|
||||||
},
|
|
||||||
};
|
|
||||||
if (timeStamp) {
|
|
||||||
dataToSend.data.timeStamp = timeStamp;
|
|
||||||
}
|
|
||||||
sendMessage(dataToSend);
|
|
||||||
};
|
|
||||||
|
|
||||||
const decorateData = (data: SelfServeTelemetryMessage, actionModifier?: string) => {
|
|
||||||
return {
|
|
||||||
databaseAccountName: userContext.databaseAccount?.name,
|
|
||||||
defaultExperience: userContext.defaultExperience,
|
|
||||||
authType: userContext.authType,
|
|
||||||
subscriptionId: userContext.subscriptionId,
|
|
||||||
platform: configContext.platform,
|
|
||||||
env: process.env.NODE_ENV,
|
|
||||||
actionModifier,
|
|
||||||
...data,
|
|
||||||
} as { [key: string]: string };
|
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
import { TelemetryData } from "../Shared/Telemetry/TelemetryProcessor";
|
||||||
|
|
||||||
interface BaseInput {
|
interface BaseInput {
|
||||||
dataFieldName: string;
|
dataFieldName: string;
|
||||||
errorMessage?: string;
|
errorMessage?: string;
|
||||||
@@ -158,8 +160,6 @@ export interface RefreshParams {
|
|||||||
retryIntervalInMs: number;
|
retryIntervalInMs: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface SelfServeTelemetryMessage {
|
export interface SelfServeTelemetryMessage extends TelemetryData {
|
||||||
selfServeClassName: string;
|
selfServeClassName: string;
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
||||||
data?: any;
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { IsDisplayable, OnChange, RefreshOptions, Values } from "../Decorators";
|
import { IsDisplayable, OnChange, RefreshOptions, Values } from "../Decorators";
|
||||||
import { trace } from "../SelfServeTelemetryProcessor";
|
import { selfServeTrace } from "../SelfServeTelemetryProcessor";
|
||||||
import {
|
import {
|
||||||
ChoiceItem,
|
ChoiceItem,
|
||||||
Description,
|
Description,
|
||||||
@@ -177,7 +177,7 @@ export default class SqlX extends SelfServeBaseClass {
|
|||||||
currentValues: Map<string, SmartUiInput>,
|
currentValues: Map<string, SmartUiInput>,
|
||||||
baselineValues: Map<string, SmartUiInput>
|
baselineValues: Map<string, SmartUiInput>
|
||||||
): Promise<OnSaveResult> => {
|
): Promise<OnSaveResult> => {
|
||||||
trace({ selfServeClassName: "SqlX" });
|
selfServeTrace({ selfServeClassName: "SqlX" });
|
||||||
|
|
||||||
const dedicatedGatewayCurrentlyEnabled = currentValues.get("enableDedicatedGateway")?.value as boolean;
|
const dedicatedGatewayCurrentlyEnabled = currentValues.get("enableDedicatedGateway")?.value as boolean;
|
||||||
const dedicatedGatewayOriginallyEnabled = baselineValues.get("enableDedicatedGateway")?.value as boolean;
|
const dedicatedGatewayOriginallyEnabled = baselineValues.get("enableDedicatedGateway")?.value as boolean;
|
||||||
@@ -234,7 +234,7 @@ export default class SqlX extends SelfServeBaseClass {
|
|||||||
portalNotification: {
|
portalNotification: {
|
||||||
initialize: {
|
initialize: {
|
||||||
titleTKey: "CreateInitializeTitle",
|
titleTKey: "CreateInitializeTitle",
|
||||||
messageTKey: "CreateInitializeTitle",
|
messageTKey: "CreateInitializeMessage",
|
||||||
},
|
},
|
||||||
success: {
|
success: {
|
||||||
titleTKey: "CreateSuccessTitle",
|
titleTKey: "CreateSuccessTitle",
|
||||||
|
|||||||
@@ -1,15 +1,25 @@
|
|||||||
import { sendMessage } from "../../Common/MessageHandler";
|
import { sendMessage } from "../../Common/MessageHandler";
|
||||||
import { configContext } from "../../ConfigContext";
|
import { configContext } from "../../ConfigContext";
|
||||||
import { MessageTypes } from "../../Contracts/ExplorerContracts";
|
import { MessageTypes } from "../../Contracts/ExplorerContracts";
|
||||||
|
import { SelfServeMessageTypes } from "../../Contracts/SelfServeContracts";
|
||||||
import { userContext } from "../../UserContext";
|
import { userContext } from "../../UserContext";
|
||||||
import { appInsights } from "../appInsights";
|
import { startTrackEvent, stopTrackEvent, trackEvent } from "../appInsights";
|
||||||
import { Action, ActionModifiers } from "./TelemetryConstants";
|
import { Action, ActionModifiers } from "./TelemetryConstants";
|
||||||
|
|
||||||
type TelemetryData = { [key: string]: unknown };
|
// Right now, the ExplorerContracts has MessageTypes as a numeric enum (TelemetryInfo = 0) while the SelfServeContracts
|
||||||
|
// has MessageTypes as a string enum (TelemetryInfo = "TelemetryInfo"). We should move to string enums for all use cases.
|
||||||
|
type TelemetryType = MessageTypes.TelemetryInfo | SelfServeMessageTypes.TelemetryInfo;
|
||||||
|
|
||||||
export function trace(action: Action, actionModifier: string = ActionModifiers.Mark, data: TelemetryData = {}): void {
|
export type TelemetryData = { [key: string]: unknown };
|
||||||
|
|
||||||
|
export function trace(
|
||||||
|
action: Action,
|
||||||
|
actionModifier: string = ActionModifiers.Mark,
|
||||||
|
data: TelemetryData = {},
|
||||||
|
type: TelemetryType = MessageTypes.TelemetryInfo
|
||||||
|
): void {
|
||||||
sendMessage({
|
sendMessage({
|
||||||
type: MessageTypes.TelemetryInfo,
|
type: type,
|
||||||
data: {
|
data: {
|
||||||
action: Action[action],
|
action: Action[action],
|
||||||
actionModifier: actionModifier,
|
actionModifier: actionModifier,
|
||||||
@@ -17,13 +27,17 @@ export function trace(action: Action, actionModifier: string = ActionModifiers.M
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
appInsights.trackEvent({ name: Action[action] }, decorateData(data, actionModifier));
|
trackEvent({ name: Action[action] }, decorateData(data, actionModifier));
|
||||||
}
|
}
|
||||||
|
|
||||||
export function traceStart(action: Action, data?: TelemetryData): number {
|
export function traceStart(
|
||||||
|
action: Action,
|
||||||
|
data?: TelemetryData,
|
||||||
|
type: TelemetryType = MessageTypes.TelemetryInfo
|
||||||
|
): number {
|
||||||
const timestamp: number = Date.now();
|
const timestamp: number = Date.now();
|
||||||
sendMessage({
|
sendMessage({
|
||||||
type: MessageTypes.TelemetryInfo,
|
type: type,
|
||||||
data: {
|
data: {
|
||||||
action: Action[action],
|
action: Action[action],
|
||||||
actionModifier: ActionModifiers.Start,
|
actionModifier: ActionModifiers.Start,
|
||||||
@@ -32,13 +46,18 @@ export function traceStart(action: Action, data?: TelemetryData): number {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
appInsights.startTrackEvent(Action[action]);
|
startTrackEvent(Action[action]);
|
||||||
return timestamp;
|
return timestamp;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function traceSuccess(action: Action, data?: TelemetryData, timestamp?: number): void {
|
export function traceSuccess(
|
||||||
|
action: Action,
|
||||||
|
data?: TelemetryData,
|
||||||
|
timestamp?: number,
|
||||||
|
type: TelemetryType = MessageTypes.TelemetryInfo
|
||||||
|
): void {
|
||||||
sendMessage({
|
sendMessage({
|
||||||
type: MessageTypes.TelemetryInfo,
|
type: type,
|
||||||
data: {
|
data: {
|
||||||
action: Action[action],
|
action: Action[action],
|
||||||
actionModifier: ActionModifiers.Success,
|
actionModifier: ActionModifiers.Success,
|
||||||
@@ -47,12 +66,17 @@ export function traceSuccess(action: Action, data?: TelemetryData, timestamp?: n
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
appInsights.stopTrackEvent(Action[action], decorateData(data, ActionModifiers.Success));
|
stopTrackEvent(Action[action], decorateData(data, ActionModifiers.Success));
|
||||||
}
|
}
|
||||||
|
|
||||||
export function traceFailure(action: Action, data?: TelemetryData, timestamp?: number): void {
|
export function traceFailure(
|
||||||
|
action: Action,
|
||||||
|
data?: TelemetryData,
|
||||||
|
timestamp?: number,
|
||||||
|
type: TelemetryType = MessageTypes.TelemetryInfo
|
||||||
|
): void {
|
||||||
sendMessage({
|
sendMessage({
|
||||||
type: MessageTypes.TelemetryInfo,
|
type: type,
|
||||||
data: {
|
data: {
|
||||||
action: Action[action],
|
action: Action[action],
|
||||||
actionModifier: ActionModifiers.Failed,
|
actionModifier: ActionModifiers.Failed,
|
||||||
@@ -61,12 +85,17 @@ export function traceFailure(action: Action, data?: TelemetryData, timestamp?: n
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
appInsights.stopTrackEvent(Action[action], decorateData(data, ActionModifiers.Failed));
|
stopTrackEvent(Action[action], decorateData(data, ActionModifiers.Failed));
|
||||||
}
|
}
|
||||||
|
|
||||||
export function traceCancel(action: Action, data?: TelemetryData, timestamp?: number): void {
|
export function traceCancel(
|
||||||
|
action: Action,
|
||||||
|
data?: TelemetryData,
|
||||||
|
timestamp?: number,
|
||||||
|
type: TelemetryType = MessageTypes.TelemetryInfo
|
||||||
|
): void {
|
||||||
sendMessage({
|
sendMessage({
|
||||||
type: MessageTypes.TelemetryInfo,
|
type: type,
|
||||||
data: {
|
data: {
|
||||||
action: Action[action],
|
action: Action[action],
|
||||||
actionModifier: ActionModifiers.Cancel,
|
actionModifier: ActionModifiers.Cancel,
|
||||||
@@ -75,13 +104,18 @@ export function traceCancel(action: Action, data?: TelemetryData, timestamp?: nu
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
appInsights.stopTrackEvent(Action[action], decorateData(data, ActionModifiers.Cancel));
|
stopTrackEvent(Action[action], decorateData(data, ActionModifiers.Cancel));
|
||||||
}
|
}
|
||||||
|
|
||||||
export function traceOpen(action: Action, data?: TelemetryData, timestamp?: number): number {
|
export function traceOpen(
|
||||||
|
action: Action,
|
||||||
|
data?: TelemetryData,
|
||||||
|
timestamp?: number,
|
||||||
|
type: TelemetryType = MessageTypes.TelemetryInfo
|
||||||
|
): number {
|
||||||
const validTimestamp = timestamp || Date.now();
|
const validTimestamp = timestamp || Date.now();
|
||||||
sendMessage({
|
sendMessage({
|
||||||
type: MessageTypes.TelemetryInfo,
|
type: type,
|
||||||
data: {
|
data: {
|
||||||
action: Action[action],
|
action: Action[action],
|
||||||
actionModifier: ActionModifiers.Open,
|
actionModifier: ActionModifiers.Open,
|
||||||
@@ -90,14 +124,19 @@ export function traceOpen(action: Action, data?: TelemetryData, timestamp?: numb
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
appInsights.startTrackEvent(Action[action]);
|
startTrackEvent(Action[action]);
|
||||||
return validTimestamp;
|
return validTimestamp;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function traceMark(action: Action, data?: TelemetryData, timestamp?: number): number {
|
export function traceMark(
|
||||||
|
action: Action,
|
||||||
|
data?: TelemetryData,
|
||||||
|
timestamp?: number,
|
||||||
|
type: TelemetryType = MessageTypes.TelemetryInfo
|
||||||
|
): number {
|
||||||
const validTimestamp = timestamp || Date.now();
|
const validTimestamp = timestamp || Date.now();
|
||||||
sendMessage({
|
sendMessage({
|
||||||
type: MessageTypes.TelemetryInfo,
|
type: type,
|
||||||
data: {
|
data: {
|
||||||
action: Action[action],
|
action: Action[action],
|
||||||
actionModifier: ActionModifiers.Mark,
|
actionModifier: ActionModifiers.Mark,
|
||||||
@@ -106,7 +145,7 @@ export function traceMark(action: Action, data?: TelemetryData, timestamp?: numb
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
appInsights.startTrackEvent(Action[action]);
|
startTrackEvent(Action[action]);
|
||||||
return validTimestamp;
|
return validTimestamp;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,8 @@
|
|||||||
import { ApplicationInsights } from "@microsoft/applicationinsights-web";
|
import { ApplicationInsights } from "@microsoft/applicationinsights-web";
|
||||||
|
|
||||||
|
// TODO: Remove this after 06/01/21.
|
||||||
|
// This points to an old app insights instance that is difficult to access
|
||||||
|
// For now we are sending data to two instances of app insights
|
||||||
const appInsights = new ApplicationInsights({
|
const appInsights = new ApplicationInsights({
|
||||||
config: {
|
config: {
|
||||||
instrumentationKey: "fa645d97-6237-4656-9559-0ee0cb55ee49",
|
instrumentationKey: "fa645d97-6237-4656-9559-0ee0cb55ee49",
|
||||||
@@ -7,7 +10,38 @@ const appInsights = new ApplicationInsights({
|
|||||||
disableCorrelationHeaders: true,
|
disableCorrelationHeaders: true,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
appInsights.loadAppInsights();
|
|
||||||
appInsights.trackPageView(); // Manually call trackPageView to establish the current user/session/pageview
|
|
||||||
|
|
||||||
export { appInsights };
|
const appInsights2 = new ApplicationInsights({
|
||||||
|
config: {
|
||||||
|
instrumentationKey: "023d2c39-8f86-468e-bb8f-bcaebd9025c7",
|
||||||
|
disableFetchTracking: false,
|
||||||
|
disableCorrelationHeaders: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
appInsights.loadAppInsights();
|
||||||
|
appInsights.trackPageView();
|
||||||
|
appInsights2.loadAppInsights();
|
||||||
|
appInsights2.trackPageView();
|
||||||
|
|
||||||
|
const trackEvent: typeof appInsights.trackEvent = (...args) => {
|
||||||
|
appInsights.trackEvent(...args);
|
||||||
|
appInsights2.trackEvent(...args);
|
||||||
|
};
|
||||||
|
|
||||||
|
const startTrackEvent: typeof appInsights.startTrackEvent = (...args) => {
|
||||||
|
appInsights.startTrackEvent(...args);
|
||||||
|
appInsights2.startTrackEvent(...args);
|
||||||
|
};
|
||||||
|
|
||||||
|
const stopTrackEvent: typeof appInsights.stopTrackEvent = (...args) => {
|
||||||
|
appInsights.stopTrackEvent(...args);
|
||||||
|
appInsights2.stopTrackEvent(...args);
|
||||||
|
};
|
||||||
|
|
||||||
|
const trackTrace: typeof appInsights.trackTrace = (...args) => {
|
||||||
|
appInsights.trackTrace(...args);
|
||||||
|
appInsights2.trackTrace(...args);
|
||||||
|
};
|
||||||
|
|
||||||
|
export { trackEvent, startTrackEvent, stopTrackEvent, trackTrace };
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ import { DatabaseAccount } from "./Contracts/DataModels";
|
|||||||
import { SubscriptionType } from "./Contracts/SubscriptionType";
|
import { SubscriptionType } from "./Contracts/SubscriptionType";
|
||||||
import { DefaultAccountExperienceType } from "./DefaultAccountExperienceType";
|
import { DefaultAccountExperienceType } from "./DefaultAccountExperienceType";
|
||||||
import { extractFeatures, Features } from "./Platform/Hosted/extractFeatures";
|
import { extractFeatures, Features } from "./Platform/Hosted/extractFeatures";
|
||||||
|
import { CollectionCreation } from "./Shared/Constants";
|
||||||
|
|
||||||
interface UserContext {
|
interface UserContext {
|
||||||
readonly authType?: AuthType;
|
readonly authType?: AuthType;
|
||||||
@@ -24,6 +25,8 @@ interface UserContext {
|
|||||||
readonly isTryCosmosDBSubscription?: boolean;
|
readonly isTryCosmosDBSubscription?: boolean;
|
||||||
readonly portalEnv?: PortalEnv;
|
readonly portalEnv?: PortalEnv;
|
||||||
readonly features: Features;
|
readonly features: Features;
|
||||||
|
readonly addCollectionFlight: string;
|
||||||
|
readonly hasWriteAccess: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
type ApiType = "SQL" | "Mongo" | "Gremlin" | "Tables" | "Cassandra";
|
type ApiType = "SQL" | "Mongo" | "Gremlin" | "Tables" | "Cassandra";
|
||||||
@@ -33,10 +36,13 @@ const features = extractFeatures();
|
|||||||
const { enableSDKoperations: useSDKOperations } = features;
|
const { enableSDKoperations: useSDKOperations } = features;
|
||||||
|
|
||||||
const userContext: UserContext = {
|
const userContext: UserContext = {
|
||||||
|
hasWriteAccess: true,
|
||||||
isTryCosmosDBSubscription: false,
|
isTryCosmosDBSubscription: false,
|
||||||
portalEnv: "prod",
|
portalEnv: "prod",
|
||||||
features,
|
features,
|
||||||
useSDKOperations,
|
useSDKOperations,
|
||||||
|
addCollectionFlight: CollectionCreation.DefaultAddCollectionDefaultFlight,
|
||||||
|
subscriptionType: CollectionCreation.DefaultSubscriptionType,
|
||||||
};
|
};
|
||||||
|
|
||||||
function updateUserContext(newContext: Partial<UserContext>): void {
|
function updateUserContext(newContext: Partial<UserContext>): void {
|
||||||
|
|||||||
@@ -23,6 +23,7 @@ import {
|
|||||||
getDatabaseAccountKindFromExperience,
|
getDatabaseAccountKindFromExperience,
|
||||||
getDatabaseAccountPropertiesFromMetadata,
|
getDatabaseAccountPropertiesFromMetadata,
|
||||||
} from "../Platform/Hosted/HostedUtils";
|
} from "../Platform/Hosted/HostedUtils";
|
||||||
|
import { CollectionCreation } from "../Shared/Constants";
|
||||||
import { DefaultExperienceUtility } from "../Shared/DefaultExperienceUtility";
|
import { DefaultExperienceUtility } from "../Shared/DefaultExperienceUtility";
|
||||||
import { PortalEnv, updateUserContext } from "../UserContext";
|
import { PortalEnv, updateUserContext } from "../UserContext";
|
||||||
import { listKeys } from "../Utils/arm/generatedClients/2020-04-01/databaseAccounts";
|
import { listKeys } from "../Utils/arm/generatedClients/2020-04-01/databaseAccounts";
|
||||||
@@ -254,6 +255,9 @@ async function configurePortal(explorerParams: ExplorerParams): Promise<Explorer
|
|||||||
subscriptionType: inputs.subscriptionType,
|
subscriptionType: inputs.subscriptionType,
|
||||||
quotaId: inputs.quotaId,
|
quotaId: inputs.quotaId,
|
||||||
portalEnv: inputs.serverId as PortalEnv,
|
portalEnv: inputs.serverId as PortalEnv,
|
||||||
|
hasWriteAccess: inputs.hasWriteAccess ?? true,
|
||||||
|
addCollectionFlight:
|
||||||
|
inputs.addCollectionDefaultFlight || CollectionCreation.DefaultAddCollectionDefaultFlight,
|
||||||
});
|
});
|
||||||
|
|
||||||
const explorer = new Explorer(explorerParams);
|
const explorer = new Explorer(explorerParams);
|
||||||
|
|||||||
@@ -8,14 +8,15 @@ i18n
|
|||||||
.init({
|
.init({
|
||||||
fallbackLng: "en",
|
fallbackLng: "en",
|
||||||
detection: { order: ["navigator", "cookie", "localStorage", "sessionStorage", "querystring", "htmlTag"] },
|
detection: { order: ["navigator", "cookie", "localStorage", "sessionStorage", "querystring", "htmlTag"] },
|
||||||
debug: process.env.NODE_ENV === "development",
|
// temporarily setting debug to true to investigate loading issues in prod
|
||||||
|
debug: true,
|
||||||
keySeparator: ".",
|
keySeparator: ".",
|
||||||
interpolation: {
|
interpolation: {
|
||||||
formatSeparator: ",",
|
formatSeparator: ",",
|
||||||
},
|
},
|
||||||
react: {
|
react: {
|
||||||
wait: true,
|
wait: true,
|
||||||
bindI18n: "languageChanged loaded",
|
bindI18n: "languageChanged added loaded",
|
||||||
bindI18nStore: "added removed",
|
bindI18nStore: "added removed",
|
||||||
nsMode: "default",
|
nsMode: "default",
|
||||||
useSuspense: false,
|
useSuspense: false,
|
||||||
|
|||||||
Reference in New Issue
Block a user