AzureSource.js 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  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, skipLog?: ?boolean): 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. skipLog,
  94. }).then(projectsResponse => {
  95. if (!Util.isResponseValid(projectsResponse)) {
  96. return []
  97. }
  98. let projects = projectsResponse.data.value.filter(p => p.type === 'Microsoft.Migrate/projects')
  99. // Load groups for each project
  100. return Promise.all(projects.map(project => {
  101. return Api.send({
  102. url: Util.buildUrl(groupsUrl({ projectName: project.name, subscriptionId, resourceGroupName })),
  103. cancelId,
  104. }).then(groupsResponse => {
  105. if (!Util.isResponseValid(groupsResponse)) {
  106. return null
  107. }
  108. return groupsResponse.data.value.map(group => { return { ...group, project } })
  109. })
  110. }))
  111. }).then(groupsResponses => {
  112. let groups = []
  113. groupsResponses.filter(r => r !== null).forEach(validGroupsReponse => {
  114. groups = groups.concat(validGroupsReponse)
  115. })
  116. // Load assessments for each group
  117. return Promise.all(groups.map(group => {
  118. // $FlowIgnore
  119. return Api.send({
  120. url: Util.buildUrl(assessmentsUrl({ subscriptionId, resourceGroupName, projectName: group.project.name, groupName: group.name })),
  121. cancelId,
  122. }).then(assessmentResponse => {
  123. if (!Util.isResponseValid(assessmentResponse)) {
  124. return null
  125. }
  126. return assessmentResponse.data.value.map(assessment => { return { ...assessment, group, project: group.project } })
  127. })
  128. }))
  129. }).then(assessementsResponses => {
  130. let assessments = []
  131. assessementsResponses.filter(r => r !== null).forEach(validAssessmentsResponse => {
  132. assessments = assessments.concat(validAssessmentsResponse)
  133. })
  134. return Util.sortAssessments(assessments)
  135. })
  136. }
  137. static getAssessmentDetails(info: Assessment): Promise<Assessment> {
  138. return Api.get(Util.buildUrl(assessmentDetailsUrl({ ...info, subscriptionId: info.connectionInfo.subscription_id }))).then(response => {
  139. return Util.validateResponse(response, { ...response.data, ...info })
  140. })
  141. }
  142. static getAssessedVms(info: Assessment): Promise<VmItem[]> {
  143. return Api.get(Util.buildUrl(assessedVmsUrl({ ...info, subscriptionId: info.connectionInfo.subscription_id }))).then(response => {
  144. if (!Util.isResponseValid(response)) {
  145. return []
  146. }
  147. let vms = response.data.value
  148. vms.sort((a, b) => {
  149. let getLabel = item => `${item.properties.datacenterContainer}/${item.properties.displayName}`
  150. return getLabel(a).localeCompare(getLabel(b))
  151. })
  152. return vms
  153. })
  154. }
  155. static getVmSizes(info: Assessment): Promise<VmSize[]> {
  156. return Api.get(Util.buildUrl(`/subscriptions/${info.connectionInfo.subscription_id}/providers/Microsoft.Compute/locations/${info.location}/vmSizes`, '2017-12-01')).then(response => {
  157. return Util.validateResponse(response, response.data.value)
  158. })
  159. }
  160. }
  161. export default AzureSource