AzureSource.js 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. /*
  2. Copyright (C) 2017 Cloudbase Solutions SRL
  3. This program is free software: you can redistribute it and/or modify
  4. it under the terms of the GNU Affero General Public License as
  5. published by the Free Software Foundation, either version 3 of the
  6. License, or (at your option) any later version.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU Affero General Public License for more details.
  11. You should have received a copy of the GNU Affero General Public License
  12. along with this program. If not, see <http://www.gnu.org/licenses/>.
  13. */
  14. // @flow
  15. import moment from 'moment'
  16. import Api from '../utils/ApiCaller'
  17. import type { Assessment, VmItem, VmSize } from '../types/Assessment'
  18. const azureUrl = 'https://management.azure.com/'
  19. const defaultApiVersion = '2017-11-11-preview'
  20. const resourceGroupsUrl = (opts: { subscriptionId: string }) => `/subscriptions/${opts.subscriptionId}/resourceGroups`
  21. const projectsUrl = ({ resourceGroupName, ...other }) => `${resourceGroupsUrl({ ...other })}/${resourceGroupName}/providers/Microsoft.Migrate/projects`
  22. const groupsUrl = ({ projectName, ...other }) => `${projectsUrl({ ...other })}/${projectName}/groups`
  23. const assessmentsUrl = ({ groupName, ...other }) => `${groupsUrl({ ...other })}/${groupName}/assessments`
  24. const assessmentDetailsUrl = ({ assessmentName, ...other }) => `${assessmentsUrl({ ...other })}/${assessmentName}`
  25. const assessedVmsUrl = ({ ...other }) => `${assessmentDetailsUrl({ ...other })}/assessedMachines`
  26. class Util {
  27. static buildUrl(baseUrl: string, apiVersion?: string): string {
  28. const url = `/proxy/${azureUrl + baseUrl}?api-version=${apiVersion || defaultApiVersion}`
  29. return url
  30. }
  31. static sortAssessments(assessments) {
  32. assessments.sort((a, b) => {
  33. return moment(b.properties.updatedTimestamp).toDate().getTime() - moment(a.properties.updatedTimestamp)
  34. })
  35. return assessments
  36. }
  37. static checkQueues(queues, requestIds, callback) {
  38. if (requestIds[0] !== requestIds[1]) {
  39. return
  40. }
  41. let doneQeues = queues.filter(q => q === 0).length
  42. if (doneQeues === queues.length) {
  43. callback()
  44. }
  45. }
  46. static isResponseValid(response): boolean {
  47. if (response && response.data && response.data.error) {
  48. const error = response.data.error
  49. console.error('%c', 'color: #D0021B', `${error.code}: ${error.message}`)
  50. return false
  51. }
  52. return true
  53. }
  54. static validateResponse(response, resolveData): Promise<any> {
  55. if (!this.isResponseValid(response)) {
  56. return Promise.reject()
  57. }
  58. if (resolveData) {
  59. return Promise.resolve(resolveData)
  60. }
  61. return Promise.resolve(response)
  62. }
  63. }
  64. class AzureSource {
  65. static authenticate(username: string, password: string): Promise<any> {
  66. return Api.send({
  67. url: '/azure-login',
  68. method: 'POST',
  69. data: { username, password },
  70. }).then(response => {
  71. let entries = Object.keys(response.data.tokenCache)[0]
  72. let accessToken = response.data.tokenCache[entries][0].accessToken
  73. Api.setDefaultHeader('Authorization', `Bearer ${accessToken}`)
  74. return response.data
  75. })
  76. }
  77. static getResourceGroups(subscriptionId: string): Promise<$PropertyType<Assessment, 'group'>[]> {
  78. return Api.get(Util.buildUrl(resourceGroupsUrl({ subscriptionId }), '2017-08-01')).then(response => {
  79. return Util.validateResponse(response, response.data.value)
  80. })
  81. }
  82. static previousReqId: string
  83. static getAssessments(subscriptionId: string, resourceGroupName: string): Promise<Assessment[]> {
  84. let cancelId = subscriptionId + resourceGroupName
  85. if (this.previousReqId) {
  86. Api.cancelRequests(this.previousReqId)
  87. }
  88. this.previousReqId = cancelId
  89. // Load Projects
  90. return Api.send({
  91. url: Util.buildUrl(projectsUrl({ resourceGroupName, subscriptionId })),
  92. cancelId,
  93. }).then(projectsResponse => {
  94. if (!Util.isResponseValid(projectsResponse)) {
  95. return []
  96. }
  97. let projects = projectsResponse.data.value.filter(p => p.type === 'Microsoft.Migrate/projects')
  98. // Load groups for each project
  99. return Promise.all(projects.map(project => {
  100. return Api.send({
  101. url: Util.buildUrl(groupsUrl({ projectName: project.name, subscriptionId, resourceGroupName })),
  102. cancelId,
  103. }).then(groupsResponse => {
  104. if (!Util.isResponseValid(groupsResponse)) {
  105. return null
  106. }
  107. return groupsResponse.data.value.map(group => { return { ...group, project } })
  108. })
  109. }))
  110. }).then(groupsResponses => {
  111. let groups = []
  112. groupsResponses.filter(r => r !== null).forEach(validGroupsReponse => {
  113. groups = groups.concat(validGroupsReponse)
  114. })
  115. // Load assessments for each group
  116. return Promise.all(groups.map(group => {
  117. // $FlowIgnore
  118. return Api.send({
  119. url: Util.buildUrl(assessmentsUrl({ subscriptionId, resourceGroupName, projectName: group.project.name, groupName: group.name })),
  120. cancelId,
  121. }).then(assessmentResponse => {
  122. if (!Util.isResponseValid(assessmentResponse)) {
  123. return null
  124. }
  125. return assessmentResponse.data.value.map(assessment => { return { ...assessment, group, project: group.project } })
  126. })
  127. }))
  128. }).then(assessementsResponses => {
  129. let assessments = []
  130. assessementsResponses.filter(r => r !== null).forEach(validAssessmentsResponse => {
  131. assessments = assessments.concat(validAssessmentsResponse)
  132. })
  133. return Util.sortAssessments(assessments)
  134. })
  135. }
  136. static getAssessmentDetails(info: Assessment): Promise<Assessment> {
  137. return Api.get(Util.buildUrl(assessmentDetailsUrl({ ...info, subscriptionId: info.connectionInfo.subscription_id }))).then(response => {
  138. return Util.validateResponse(response, { ...response.data, ...info })
  139. })
  140. }
  141. static getAssessedVms(info: Assessment): Promise<VmItem[]> {
  142. return Api.get(Util.buildUrl(assessedVmsUrl({ ...info, subscriptionId: info.connectionInfo.subscription_id }))).then(response => {
  143. if (!Util.isResponseValid(response)) {
  144. return []
  145. }
  146. let vms = response.data.value
  147. vms.sort((a, b) => {
  148. let getLabel = item => `${item.properties.datacenterContainer}/${item.properties.displayName}`
  149. return getLabel(a).localeCompare(getLabel(b))
  150. })
  151. return vms
  152. })
  153. }
  154. static getVmSizes(info: Assessment): Promise<VmSize[]> {
  155. return Api.get(Util.buildUrl(`/subscriptions/${info.connectionInfo.subscription_id}/providers/Microsoft.Compute/locations/${info.location}/vmSizes`, '2017-12-01')).then(response => {
  156. return Util.validateResponse(response, response.data.value)
  157. })
  158. }
  159. }
  160. export default AzureSource