Merge branch 'master' into projects
[osm/UI.git] / skyquake / plugins / composer / api / composer.js
1 /*
2 *
3 * Copyright 2016 RIFT.IO Inc
4 *
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
8 *
9 * http://www.apache.org/licenses/LICENSE-2.0
10 *
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
16 *
17 */
18
19 var request = require('request');
20 var Promise = require('bluebird');
21 var rp = require('request-promise');
22 var utils = require('../../../framework/core/api_utils/utils.js');
23 var constants = require('../../../framework/core/api_utils/constants.js');
24 var _ = require('underscore');
25 var URL = require('url');
26 var uuid = require('uuid');
27 var APIVersion = '/v1';
28 var PackageFileHandler = require('./packageFileHandler.js');
29
30 var Composer = {};
31 var FileManager = {};
32 var PackageManager = {};
33 var DataCenters = {};
34 // Catalog module methods
35 Composer.get = function(req) {
36 var api_server = req.query['api_server'];
37 var results = {}
38 var projectPrefix = req.session.projectId ? "project-" : "";
39 return new Promise(function(resolve, reject) {
40 Promise.all([
41 rp({
42 uri: utils.projectContextUrl(req, utils.confdPort(api_server) + APIVersion + '/api/config/nsd-catalog/nsd?deep'),
43 method: 'GET',
44 headers: _.extend({}, constants.HTTP_HEADERS.accept.collection, {
45 'Authorization': req.session && req.session.authorization
46 }),
47 forever: constants.FOREVER_ON,
48 rejectUnauthorized: false,
49 resolveWithFullResponse: true
50 }),
51 rp({
52 uri: utils.projectContextUrl(req, utils.confdPort(api_server) + APIVersion + '/api/config/vnfd-catalog/vnfd?deep'),
53 method: 'GET',
54 headers: _.extend({}, constants.HTTP_HEADERS.accept.collection, {
55 'Authorization': req.session && req.session.authorization
56 }),
57 forever: constants.FOREVER_ON,
58 rejectUnauthorized: false,
59 resolveWithFullResponse: true
60 }),
61 rp({
62 uri: utils.projectContextUrl(req, utils.confdPort(api_server) + APIVersion + '/api/operational/ns-instance-opdata?deep'),
63 method: 'GET',
64 headers: _.extend({}, constants.HTTP_HEADERS.accept.data, {
65 'Authorization': req.session && req.session.authorization
66 }),
67 forever: constants.FOREVER_ON,
68 rejectUnauthorized: false,
69 resolveWithFullResponse: true
70 })
71 // Not enabled for now
72 // rp({
73 // uri: utils.confdPort(api_server) + '/api/config/pnfd-catalog/pnfd?deep',
74 // method: 'GET',
75 // headers: _.extend({},
76 // constants.HTTP_HEADERS.accept.collection,
77 // {
78 // 'Authorization': req.session && req.session.authorization
79 // }),
80 // forever: constants.FOREVER_ON,
81 // rejectUnauthorized: false,
82 // resolveWithFullResponse: true
83 // })
84 ]).then(function(result) {
85 var response = [{
86 "id": "GUID-1",
87 "name": "RIFT.ware™ NS Descriptors Catalog",
88 "short-name": "rift.ware-nsd-cat",
89 "description": "RIFT.ware™, an open source NFV development and deployment software platform that makes it simple to create, deploy and manage hyper-scale Virtual network functions and applications.",
90 "vendor": "RIFT.io",
91 "version": "",
92 "created-on": "",
93 "type": "nsd",
94 "meta": {
95 "icon-svg": "data:image/svg+xml,%3C%3Fxml%20version%3D%221.0%22%20encoding%3D%22iso-8859-1%22%3F%3E%0A%3C!--%20Generator%3A%20Adobe%20Illustrator%2018.0.0%2C%20SVG%20Export%20Plug-In%20.%20SVG%20Version%3A%206.00%20Build%200)%20%20--%3E%0A%3C!DOCTYPE%20svg%20PUBLIC%20%22-%2F%2FW3C%2F%2FDTD%20SVG%201.1%2F%2FEN%22%20%22http%3A%2F%2Fwww.w3.org%2FGraphics%2FSVG%2F1.1%2FDTD%2Fsvg11.dtd%22%3E%0A%3Csvg%20version%3D%221.1%22%20id%3D%22connection-icon-1%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20xmlns%3Axlink%3D%22http%3A%2F%2Fwww.w3.org%2F1999%2Fxlink%22%20x%3D%220px%22%20y%3D%220px%22%0A%09%20viewBox%3D%220%200%2050%2050%22%20style%3D%22enable-background%3Anew%200%200%2050%2050%3B%22%20xml%3Aspace%3D%22preserve%22%3E%0A%09%3Cpath%20d%3D%22M15%2030c-2.8%200-5-2.2-5-5s2.2-5%205-5%205%202.2%205%205-2.2%205-5%205zm0-8c-1.7%200-3%201.3-3%203s1.3%203%203%203%203-1.3%203-3-1.3-3-3-3z%22%2F%3E%3Cpath%20d%3D%22M35%2020c-2.8%200-5-2.2-5-5s2.2-5%205-5%205%202.2%205%205-2.2%205-5%205zm0-8c-1.7%200-3%201.3-3%203s1.3%203%203%203%203-1.3%203-3-1.3-3-3-3z%22%2F%3E%3Cpath%20d%3D%22M35%2040c-2.8%200-5-2.2-5-5s2.2-5%205-5%205%202.2%205%205-2.2%205-5%205zm0-8c-1.7%200-3%201.3-3%203s1.3%203%203%203%203-1.3%203-3-1.3-3-3-3z%22%2F%3E%3Cpath%20d%3D%22M19.007%2025.885l12.88%206.44-.895%201.788-12.88-6.44z%22%2F%3E%3Cpath%20d%3D%22M30.993%2015.885l.894%201.79-12.88%206.438-.894-1.79z%22%2F%3E%3C%2Fsvg%3E"
96 },
97 "descriptors": []
98 }, {
99 "id": "GUID-2",
100 "name": "RIFT.ware™ VNF Descriptors Catalog",
101 "short-name": "rift.ware-vnfd-cat",
102 "description": "RIFT.ware™, an open source NFV development and deployment software platform that makes it simple to create, deploy and manage hyper-scale Virtual network functions and applications.",
103 "vendor": "RIFT.io",
104 "version": "",
105 "created-on": "",
106 "type": "vnfd",
107 "meta": {
108 "icon-svg": "data:image/svg+xml,<?xml version=\"1.0\" encoding=\"utf-8\"?> <!-- Generator: Adobe Illustrator 16.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 0) --> <!DOCTYPE svg PUBLIC \"-//W3C//DTD SVG 1.1//EN\" \"http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd\"> <svg version=\"1.1\" id=\"Layer_3\" xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\" x=\"0px\" y=\"0px\" width=\"100px\" height=\"100px\" viewBox=\"0 0 100 100\" enable-background=\"new 0 0 100 100\" xml:space=\"preserve\"> <g> <path d=\"M58.852,62.447l-4.662-1.033c-0.047-3.138-0.719-6.168-1.996-9.007l3.606-2.92c0.858-0.695,0.99-1.954,0.296-2.813 l-4.521-5.584c-0.334-0.413-0.818-0.675-1.346-0.731c-0.525-0.057-1.056,0.102-1.468,0.435L45.25,43.64v0 c-2.486-1.907-5.277-3.259-8.297-4.019v-4.458c0-1.104-0.896-2-2-2H27.77c-1.104,0-2,0.896-2,2v4.461 c-3.08,0.777-5.922,2.171-8.447,4.144l-3.545-2.82c-0.415-0.33-0.94-0.479-1.472-0.422c-0.527,0.06-1.009,0.327-1.339,0.743 l-4.472,5.623c-0.688,0.864-0.544,2.123,0.32,2.81l3.642,2.896v0c-1.25,2.848-1.895,5.88-1.916,9.011l-4.666,1.078 c-1.076,0.249-1.747,1.322-1.499,2.398l1.616,7.001c0.249,1.077,1.325,1.747,2.399,1.499l4.813-1.111v0 c1.429,2.681,3.344,5.017,5.691,6.943l-2.17,4.55c-0.476,0.997-0.054,2.19,0.943,2.666l6.484,3.094 c0.271,0.129,0.566,0.195,0.861,0.195c0.226,0,0.451-0.038,0.668-0.115c0.5-0.177,0.909-0.545,1.138-1.024l2.198-4.611 c2.923,0.563,5.966,0.554,8.879-0.033l2.236,4.585c0.484,0.994,1.685,1.403,2.675,0.921l6.456-3.148 c0.992-0.484,1.405-1.682,0.921-2.674l-2.206-4.524c2.335-1.946,4.231-4.301,5.639-6.999l4.812,1.067 c1.076,0.237,2.146-0.441,2.385-1.52l1.556-7.014c0.115-0.518,0.02-1.06-0.266-1.508C59.82,62.878,59.369,62.562,58.852,62.447z M40.18,61.761c0,4.859-3.953,8.812-8.813,8.812c-4.858,0-8.811-3.953-8.811-8.812s3.952-8.812,8.811-8.812 C36.227,52.949,40.18,56.902,40.18,61.761z\"/> <path d=\"M64.268,45.324c0.746,0,1.463-0.42,1.806-1.139l1.054-2.208c1.826,0.353,3.736,0.345,5.551-0.021l1.07,2.195 c0.484,0.992,1.682,1.405,2.675,0.921l2.691-1.313c0.477-0.233,0.842-0.646,1.015-1.147c0.172-0.501,0.139-1.051-0.095-1.528 l-1.052-2.155c1.458-1.214,2.645-2.686,3.527-4.377l2.278,0.504c1.075,0.238,2.146-0.442,2.386-1.52l0.647-2.923 c0.238-1.078-0.442-2.146-1.521-2.385l-2.184-0.484c-0.028-1.962-0.449-3.857-1.248-5.632l1.673-1.355 c0.412-0.334,0.675-0.818,0.73-1.345s-0.102-1.056-0.436-1.468l-1.884-2.327c-0.697-0.859-1.957-0.99-2.813-0.295l-1.614,1.307 c-1.554-1.193-3.299-2.038-5.188-2.513v-2.039c0-1.104-0.896-2-2-2h-2.994c-1.104,0-2,0.896-2,2v2.04 c-1.927,0.486-3.703,1.358-5.28,2.592l-1.634-1.298c-0.862-0.687-2.12-0.543-2.81,0.32l-1.864,2.344 c-0.33,0.416-0.481,0.945-0.422,1.472c0.061,0.527,0.327,1.009,0.743,1.339l1.69,1.345c-0.78,1.779-1.184,3.676-1.197,5.636 l-2.189,0.505c-0.517,0.119-0.965,0.439-1.246,0.889c-0.281,0.45-0.372,0.993-0.252,1.51l0.675,2.918 c0.249,1.076,1.323,1.747,2.398,1.498l2.28-0.527c0.892,1.676,2.089,3.137,3.559,4.343l-1.035,2.17 c-0.228,0.479-0.257,1.028-0.08,1.528c0.178,0.5,0.546,0.91,1.024,1.138l2.703,1.289C63.686,45.261,63.979,45.324,64.268,45.324z M64.334,27.961c0-3.039,2.473-5.51,5.512-5.51c3.038,0,5.51,2.472,5.51,5.51c0,3.039-2.472,5.511-5.51,5.511 C66.807,33.472,64.334,31,64.334,27.961z\"/> <path d=\"M96.107,66.441l-2.182-0.484c-0.028-1.961-0.449-3.856-1.25-5.632l1.675-1.355c0.412-0.334,0.675-0.818,0.73-1.346 c0.056-0.527-0.102-1.056-0.436-1.468l-1.885-2.327c-0.695-0.859-1.956-0.99-2.813-0.295l-1.614,1.307 c-1.555-1.193-3.3-2.038-5.188-2.513v-2.039c0-1.104-0.896-2-2-2h-2.994c-1.104,0-2,0.896-2,2v2.041 c-1.929,0.486-3.706,1.358-5.282,2.592l-0.001,0l-1.631-1.298c-0.415-0.331-0.938-0.482-1.472-0.422 c-0.527,0.06-1.009,0.327-1.339,0.742l-1.863,2.343c-0.688,0.865-0.544,2.123,0.32,2.811l1.691,1.345 c-0.782,1.784-1.186,3.68-1.199,5.636l-2.188,0.505c-0.517,0.12-0.965,0.439-1.246,0.889c-0.281,0.45-0.372,0.993-0.252,1.51 l0.675,2.918c0.249,1.076,1.327,1.744,2.397,1.498l2.281-0.526c0.893,1.677,2.09,3.138,3.558,4.343h0.001l-1.035,2.168 c-0.229,0.479-0.258,1.029-0.081,1.529c0.178,0.5,0.546,0.909,1.024,1.138l2.702,1.289c0.278,0.132,0.571,0.195,0.86,0.195 c0.746,0,1.463-0.42,1.806-1.139l1.054-2.208c1.828,0.353,3.739,0.347,5.552-0.021l1.071,2.194 c0.484,0.992,1.682,1.405,2.675,0.921l2.69-1.312c0.477-0.233,0.842-0.645,1.014-1.147c0.173-0.501,0.14-1.051-0.093-1.528 l-1.052-2.155c1.459-1.215,2.645-2.688,3.525-4.377l2.278,0.505c0.52,0.116,1.061,0.02,1.508-0.266 c0.447-0.285,0.763-0.736,0.878-1.254l0.647-2.923C97.866,67.748,97.186,66.681,96.107,66.441z M85.162,66.174 c0,3.039-2.471,5.511-5.508,5.511c-3.039,0-5.512-2.472-5.512-5.511c0-3.039,2.473-5.511,5.512-5.511 C82.691,60.664,85.162,63.136,85.162,66.174z\"/> </g> </svg> "
109 },
110 "descriptors": []
111 }, {
112 "id": "GUID-3",
113 "name": "RIFT.ware™ PNF Descriptors Catalog",
114 "short-name": "rift.ware-pnfd-cat",
115 "description": "RIFT.ware™, an open source NFV development and deployment software platform that makes it simple to create, deploy and manage hyper-scale Virtual network functions and applications.",
116 "vendor": "RIFT.io",
117 "version": "",
118 "created-on": "",
119 "type": "pnfd",
120 "meta": {
121 "icon-svg": "data:image/svg+xml,<?xml version=\"1.0\" encoding=\"utf-8\"?> <!-- Generator: Adobe Illustrator 16.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 0) --> <!DOCTYPE svg PUBLIC \"-//W3C//DTD SVG 1.1//EN\" \"http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd\"> <svg version=\"1.1\" id=\"Layer_4\" xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\" x=\"0px\" y=\"0px\" width=\"100px\" height=\"100px\" viewBox=\"0 0 100 100\" enable-background=\"new 0 0 100 100\" xml:space=\"preserve\"> <path d=\"M86.334,47.444V35.759H13.666v11.686h3.561v5.111h-3.561v11.686h72.668V52.556h-4.108v-5.111H86.334z M26.628,59.454h-5.051 v-4.941h5.051V59.454z M26.628,52.404h-5.051v-4.941h5.051V52.404z M26.628,45.486h-5.051v-4.941h5.051V45.486z M34.094,59.454 h-5.051v-4.941h5.051V59.454z M34.094,52.404h-5.051v-4.941h5.051V52.404z M34.094,45.486h-5.051v-4.941h5.051V45.486z M41.452,59.454h-5.051v-4.941h5.051V59.454z M41.452,52.404h-5.051v-4.941h5.051V52.404z M41.452,45.486h-5.051v-4.941h5.051 V45.486z M48.733,59.454h-5.051v-4.941h5.051V59.454z M48.733,52.404h-5.051v-4.941h5.051V52.404z M48.733,45.486h-5.051v-4.941 h5.051V45.486z M56.2,59.454h-5.051v-4.941H56.2V59.454z M56.2,52.404h-5.051v-4.941H56.2V52.404z M56.2,45.486h-5.051v-4.941H56.2 V45.486z M63.558,59.454h-5.05v-4.941h5.05V59.454z M63.558,52.404h-5.05v-4.941h5.05V52.404z M63.558,45.486h-5.05v-4.941h5.05 V45.486z M74.858,59.312h-6.521v-3.013h6.521V59.312z M71.572,50.854c-2.875,0-5.204-2.33-5.204-5.203s2.329-5.203,5.204-5.203 s5.204,2.33,5.204,5.203S74.446,50.854,71.572,50.854z M74.858,45.618c0,1.801-1.46,3.261-3.261,3.261 c-1.8,0-3.261-1.46-3.261-3.261s1.46-3.26,3.261-3.26C73.398,42.358,74.858,43.817,74.858,45.618z\"/> </svg>"
122 },
123 "descriptors": []
124 }];
125 if (result[0].body) {
126 response[0].descriptors = JSON.parse(result[0].body).collection[projectPrefix + 'nsd:nsd'];
127 if (result[2].body) {
128 var data = JSON.parse(result[2].body);
129 if (data && data["nsr:ns-instance-opdata"] && data["nsr:ns-instance-opdata"]["rw-nsr:nsd-ref-count"]) {
130 var nsdRefCountCollection = data["nsr:ns-instance-opdata"]["rw-nsr:nsd-ref-count"];
131 response[0].descriptors.map(function(nsd) {
132 if (!nsd["meta"]) {
133 nsd["meta"] = {};
134 }
135 if (typeof nsd['meta'] == 'string') {
136 nsd['meta'] = JSON.parse(nsd['meta']);
137 }
138 nsd["meta"]["instance-ref-count"] = _.findWhere(nsdRefCountCollection, {
139 "nsd-id-ref": nsd.id
140 })["instance-ref-count"];
141 });
142 }
143 }
144 };
145 if (result[1].body) {
146 response[1].descriptors = JSON.parse(result[1].body).collection[projectPrefix + 'vnfd:vnfd'];
147 };
148 // if (result[2].body) {
149 // response[2].descriptors = JSON.parse(result[2].body).collection[projectPrefix + 'pnfd:pnfd'];
150 // };
151 resolve({
152 statusCode: response.statusCode || 200,
153 data: JSON.stringify(response)
154 });
155 }).catch(function(error) {
156 // Todo: Need better logic than all or nothing.
157 // Right now even if one of the southbound APIs fails - all fail
158 var res = {};
159 console.log('Problem with Composer.get', error);
160 res.statusCode = error.statusCode || 500;
161 res.errorMessage = {
162 error: 'Failed to get catalogs' + error
163 };
164 reject(res);
165 });
166 });
167 };
168 Composer.delete = function(req) {
169 var api_server = req.query['api_server'];
170 var catalogType = req.params.catalogType;
171 var id = req.params.id;
172 console.log('Deleting', catalogType, id, 'from', api_server);
173 return new Promise(function(resolve, reject) {
174 request({
175 uri: utils.projectContextUrl(req, utils.confdPort(api_server) + APIVersion + '/api/config/' + catalogType + '-catalog/' + catalogType + '/' + id),
176 method: 'DELETE',
177 headers: _.extend({}, constants.HTTP_HEADERS.accept.data, {
178 'Authorization': req.session && req.session.authorization
179 }),
180 forever: constants.FOREVER_ON,
181 rejectUnauthorized: false,
182 }, function(error, response, body) {
183 if (utils.validateResponse('Composer.delete', error, response, body, resolve, reject)) {
184 resolve({
185 statusCode: response.statusCode
186 });
187 }
188 });
189 });
190 };
191 Composer.getVNFD = function(req) {
192 var api_server = req.query['api_server'];
193 var vnfdID = req.body.data;
194 var authorization = req.session && req.session.authorization;
195 var VNFDs = [];
196 if (typeof(vnfdID) == "object" && vnfdID.constructor.name == "Array") {
197 vnfdID.map(function(id) {
198 VNFDs.push(requestVNFD(id));
199 });
200 } else {
201 VNFDs.push(requestVNFD(vnfdID));
202 }
203 return new Promise(function(resolve, reject) {
204 Promise.all(VNFDs).then(function(data) {
205 resolve(data)
206 }).catch(function(error) {
207 // Todo: Need better logic than all or nothing.
208 // Right now even if one of the southbound APIs fails - all fail
209 var res = {};
210 console.log('Problem with Composer.getVNFD', error);
211 res.statusCode = 404;
212 res.errorMessage = {
213 error: 'Failed to get VNFDs' + error
214 };
215 reject(res);
216 });
217 });
218
219 function requestVNFD(id) {
220 return new Promise(function(resolve, reject) {
221 var url = utils.confdPort(api_server) + APIVersion + '/api/config/vnfd-catalog/vnfd' + (id ? '/' + id : '') + '?deep';
222 request({
223 uri: utils.projectContextUrl(req, url),
224 method: 'GET',
225 headers: _.extend({}, constants.HTTP_HEADERS.accept.data, {
226 'Authorization': authorization
227 }),
228 forever: constants.FOREVER_ON,
229 rejectUnauthorized: false,
230 }, function(error, response, body) {
231 if (utils.validateResponse('Composer.create', error, response, body, resolve, reject)) {
232 var data;
233 //Is this still needed?
234 try {
235 data = JSON.parse(response.body)
236 } catch (e) {
237 reject({
238 statusCode: response ? response.statusCode : 400,
239 errorMessage: 'Issue parsing VNFD ' + id + 'from ' + utils.confdPort(api_server) + '/api/config/vnfd-catalog/vnfd/' + id + '?deep'
240 });
241 }
242 resolve(data);
243 }
244 });
245 });
246 }
247 };
248 Composer.create = function(req) {
249 var api_server = req.query['api_server'];
250 var catalogType = req.params.catalogType;
251 var data = req.body;
252 console.log('Creating', catalogType, 'on', api_server);
253 var jsonData = {};
254 jsonData[catalogType] = [];
255 jsonData[catalogType].push(data);
256 return new Promise(function(resolve, reject) {
257 var requestHeaders = {};
258 _.extend(requestHeaders, constants.HTTP_HEADERS.accept.data, constants.HTTP_HEADERS.content_type.data, {
259 'Authorization': req.session && req.session.authorization
260 });
261 request({
262 uri: utils.projectContextUrl(req, utils.confdPort(api_server) + '/api/config/' + catalogType + '-catalog'),
263 method: 'POST',
264 headers: requestHeaders,
265 forever: constants.FOREVER_ON,
266 rejectUnauthorized: false,
267 json: jsonData
268 }, function(error, response, body) {
269 if (utils.validateResponse('Composer.create', error, response, body, resolve, reject)) {
270 resolve({
271 statusCode: response.statusCode
272 });
273 }
274 });
275 });
276 };
277 Composer.updateSave = function(req) {
278 var api_server = req.query['api_server'];
279 var catalogType = req.params.catalogType;
280 var id = req.params.id;
281 var data = req.body;
282 console.log('Updating', catalogType, 'id', id, 'on', api_server);
283 var jsonData = {};
284 jsonData[catalogType] = {};
285 jsonData[catalogType] = data;
286 return new Promise(function(resolve, reject) {
287 var requestHeaders = {};
288 _.extend(requestHeaders, constants.HTTP_HEADERS.accept.data, constants.HTTP_HEADERS.content_type.data, {
289 'Authorization': req.session && req.session.authorization
290 });
291 request({
292 uri: utils.projectContextUrl(req, utils.confdPort(api_server) + APIVersion + '/api/config/' + catalogType + '-catalog' + '/' + catalogType + '/' + id),
293 method: 'PUT',
294 headers: requestHeaders,
295 forever: constants.FOREVER_ON,
296 rejectUnauthorized: false,
297 json: jsonData
298 }, function(error, response, body) {
299 if (utils.validateResponse('Composer.update', error, response, body, resolve, reject)) {
300 resolve({
301 statusCode: response.statusCode
302 });
303 }
304 });
305 });
306 }
307
308 PackageManager.upload = function(req) {
309 console.log(' Uploading file', req.file.originalname, 'as', req.file.filename);
310 var api_server = req.query['api_server'];
311 // dev_download_server is for testing purposes.
312 // It is the direct IP address of the Node server where the
313 // package will be hosted.
314 var download_host = req.query['dev_download_server'];
315
316 if (!download_host) {
317 download_host = req.protocol + '://' + req.get('host');//req.api_server + ':' + utils.getPortForProtocol(req.protocol);
318 }
319
320 var input = {
321 'external-url': download_host + '/composer/upload/' + req.file.filename,
322 'package-type': 'VNFD',
323 'package-id': uuid()
324 }
325
326 var uri = utils.projectContextUrl(req, utils.confdPort(api_server) + '/api/operations/package-create');
327
328 input = utils.addProjectContextToRPCPayload(req, uri, input);
329
330 return new Promise(function(resolve, reject) {
331 Promise.all([
332 rp({
333 uri: uri,
334 method: 'POST',
335 headers: _.extend({}, constants.HTTP_HEADERS.accept.collection, {
336 'Authorization': req.session && req.session.authorization
337 }),
338 forever: constants.FOREVER_ON,
339 rejectUnauthorized: false,
340 resolveWithFullResponse: true,
341 json: true,
342 body: {
343 input: input
344 }
345 })
346 ]).then(function(result) {
347 var data = {};
348 data['transaction_id'] = result[0].body['output']['transaction-id'];
349
350 // Add a status checker on the transaction and then to delete the file later
351 PackageFileHandler.checkCreatePackageStatusAndHandleFile(req, data['transaction_id'], false);
352
353 // Return status to composer UI to update the status.
354 resolve({
355 statusCode: constants.HTTP_RESPONSE_CODES.SUCCESS.OK,
356 data: data
357 });
358 }).catch(function(error) {
359 var res = {};
360 console.log('Problem with PackageManager.upload', error);
361 res.statusCode = error.statusCode || 500;
362 res.errorMessage = {
363 error: 'Failed to upload package ' + req.file.originalname + '. Error: ' + error
364 };
365 reject(res);
366 });
367 });
368 };
369
370 PackageManager.update = function(req) {
371 console.log(' Updating file', req.file.originalname, 'as', req.file.filename);
372 var api_server = req.query['api_server'];
373 // dev_download_server is for testing purposes.
374 // It is the direct IP address of the Node server where the
375 // package will be hosted.
376 var download_host = req.query['dev_download_server'];
377
378 if (!download_host) {
379 download_host = req.protocol + '://' + req.get('host');//api_server + ':' + utils.getPortForProtocol(req.protocol);
380 }
381 var input = {
382 'external-url': download_host + '/composer/update/' + req.file.filename,
383 'package-type': 'VNFD',
384 'package-id': uuid()
385 };
386
387 var uri = utils.projectContextUrl(req, utils.confdPort(api_server) + '/api/operations/package-update');
388
389 input = utils.addProjectContextToRPCPayload(req, uri, input);
390
391 return new Promise(function(resolve, reject) {
392 Promise.all([
393 rp({
394 uri: uri,
395 method: 'POST',
396 headers: _.extend({}, constants.HTTP_HEADERS.accept.collection, {
397 'Authorization': req.session && req.session.authorization
398 }),
399 forever: constants.FOREVER_ON,
400 rejectUnauthorized: false,
401 resolveWithFullResponse: true,
402 json: true,
403 body: {
404 input: input
405 }
406 })
407 ]).then(function(result) {
408 var data = {};
409 data['transaction_id'] = result[0].body['output']['transaction-id'];
410
411 // Add a status checker on the transaction and then to delete the file later
412 PackageFileHandler.checkCreatePackageStatusAndHandleFile(req, data['transaction_id'], true);
413
414 // Return status to composer UI to update the status.
415 resolve({
416 statusCode: constants.HTTP_RESPONSE_CODES.SUCCESS.OK,
417 data: data
418 });
419 }).catch(function(error) {
420 var res = {};
421 console.log('Problem with Composer.upload', error);
422 res.statusCode = error.statusCode || 500;
423 res.errorMessage = {
424 error: 'Failed to upload package ' + req.file.originalname + '. Error: ' + error
425 };
426 reject(res);
427 });
428 });
429 };
430
431 PackageManager.export = function(req) {
432 // /api/operations/package-export
433 var api_server = req.query['api_server'];
434 var uri = utils.projectContextUrl(req, utils.confdPort(api_server) + '/api/operations/package-export');
435 var input = req.body;
436 input = utils.addProjectContextToRPCPayload(req, uri, input);
437 return new Promise(function(resolve, reject) {
438 Promise.all([
439 rp({
440 uri: utils.confdPort(api_server) + '/api/operations/package-export',
441 method: 'POST',
442 headers: _.extend({}, constants.HTTP_HEADERS.accept.collection, {
443 'Authorization': req.session && req.session.authorization
444 }),
445 forever: constants.FOREVER_ON,
446 rejectUnauthorized: false,
447 resolveWithFullResponse: true,
448 json: true,
449 body: { "input": input }
450 })
451 ]).then(function(result) {
452 var data = {};
453 resolve({
454 statusCode: constants.HTTP_RESPONSE_CODES.SUCCESS.OK,
455 data: result[0].body
456 });
457 }).catch(function(error) {
458 var res = {};
459 console.log('Problem with PackageManager.export', error);
460 res.statusCode = error.statusCode || 500;
461 res.errorMessage = {
462 error: error
463 };
464 reject(res);
465 });
466 });
467 }
468
469 PackageManager.copy = function(req) {
470 // /api/operations/package-copy
471 var api_server = req.query['api_server'];
472 var uri = utils.projectContextUrl(req, utils.confdPort(api_server) + '/api/operations/package-copy');
473 var input = req.body;
474 input = utils.addProjectContextToRPCPayload(req, uri, input);
475
476 return new Promise(function(resolve, reject) {
477 Promise.all([
478 rp({
479 uri: uri,
480 method: 'POST',
481 headers: _.extend({}, constants.HTTP_HEADERS.accept.collection, {
482 'Authorization': req.session && req.session.authorization
483 }),
484 forever: constants.FOREVER_ON,
485 rejectUnauthorized: false,
486 resolveWithFullResponse: true,
487 json: true,
488 body: { "input": input}
489 })
490 ]).then(function(result) {
491 var data = {};
492 resolve({
493 statusCode: constants.HTTP_RESPONSE_CODES.SUCCESS.OK,
494 data: result[0].body
495 });
496 }).catch(function(error) {
497 var res = {};
498 console.log('Problem with PackageManager.copy', error);
499 res.statusCode = error.statusCode || 500;
500 res.errorMessage = {
501 error: error
502 };
503 reject(res);
504 });
505 });
506 }
507
508 /**
509 * This methods retrieves the status of package operations. It takes an optional
510 * transaction id (id) this if present will return only that status otherwise
511 * an array of status' will be response.
512 */
513 PackageManager.getJobStatus = function(req) {
514 var api_server = req.query["api_server"];
515 var uri = utils.confdPort(api_server);
516 var id = req.params['id'];
517 var url = utils.projectContextUrl(req, uri + '/api/operational/copy-jobs' + (id ? '/job/' + id : ''));
518 return new Promise(function(resolve, reject) {
519 request({
520 url: url,
521 method: 'GET',
522 headers: _.extend({}, constants.HTTP_HEADERS.accept.data, {
523 'Authorization': req.session && req.session.authorization
524 }),
525 forever: constants.FOREVER_ON,
526 rejectUnauthorized: false
527 }, function(error, response, body) {
528 if (utils.validateResponse('PackageManager.getJobStatus', error, response, body, resolve, reject)) {
529 var returnData;
530 if (id) {
531 returnData = JSON.parse(response.body)['rw-pkg-mgmt:job'];
532 } else {
533 var data = JSON.parse(response.body)['rw-pkg-mgmt:copy-jobs'];
534 returnData = (data && data.job) || [];
535 }
536 resolve({
537 statusCode: response.statusCode,
538 data: returnData
539 })
540 };
541 })
542 })
543 }
544
545 function makeAssetTypeParamName (type) {
546 return type.toLowerCase() + '-file-type';
547 }
548 FileManager.addFile = function(req) {
549 console.log(' Uploading file', req.file.originalname, 'as', req.file.filename);
550 var api_server = req.query['api_server'];
551 var download_host = req.query['dev_download_server'];
552 var package_id = req.query['package_id'];
553 var package_type = req.query['package_type'].toUpperCase();
554 var package_path = req.query['package_path'];
555 if (!download_host) {
556 download_host = req.protocol + '://' + req.get('host');//api_server + ':' + utils.getPortForProtocol(req.protocol);
557 }
558 var input = {
559 'external-url': download_host + '/composer/upload/' + req.query['package_id'] + '/' + req.file.filename,
560 'package-type': package_type,
561 'package-id': package_id,
562 'package-path': package_path ? package_path + '/' + req.file.filename : req.file.filename
563 };
564
565 var uri = utils.projectContextUrl(req, utils.confdPort(api_server) + '/api/operations/package-file-add');
566
567 input = utils.addProjectContextToRPCPayload(req, uri, input);
568
569 var assetType = req.query['asset_type'].toUpperCase();
570 input[makeAssetTypeParamName(package_type)] = assetType;
571 return new Promise(function(resolve, reject) {
572 Promise.all([
573 rp({
574 uri: uri,
575 method: 'POST',
576 headers: _.extend({}, constants.HTTP_HEADERS.accept.collection, {
577 'Authorization': req.session && req.session.authorization
578 }),
579 forever: constants.FOREVER_ON,
580 rejectUnauthorized: false,
581 resolveWithFullResponse: true,
582 json: true,
583 body: {
584 input: input
585 }
586 })
587 ]).then(function(result) {
588 var data = {};
589 data['transaction_id'] = result[0].body['output']['task-id'];
590 resolve({
591 statusCode: constants.HTTP_RESPONSE_CODES.SUCCESS.OK,
592 data: data
593 });
594 }).catch(function(error) {
595 var res = {};
596 console.log('Problem with Composer.upload', error);
597 res.statusCode = error.statusCode || 500;
598 res.errorMessage = {
599 error: 'Failed to upload package ' + req.file.originalname + '. Error: ' + error
600 };
601 reject(res);
602 });
603 });
604 }
605
606 FileManager.get = function(req) {
607 var api_server = req.query['api_server'];
608 var type = req.query['package_type'] && req.query['package_type'].toUpperCase();
609 var id = req.query['package_id'];
610 var downloadUrl = req.query['url'];
611 var path = req.query['package_path'];
612 var assetType = req.query['asset_type'];
613 var input = {
614 "package-type": type,
615 "package-id": id
616 }
617 var payload = {input: input};
618 if(req.method == 'GET') {
619 if(downloadUrl && path) {
620 payload.input['external-url'] = downloadUrl;
621 payload.input['package-path'] = path;
622 payload.input[makeAssetTypeParamName(type)] = assetType;
623 return download(payload);
624 } else {
625 return list(payload);
626 }
627 }
628 if(req.method == 'DELETE') {
629 payload.input['package-path'] = path;
630 payload.input[makeAssetTypeParamName(type)] = assetType;
631 return deleteFile(payload)
632 }
633
634 function deleteFile(payload) {
635 var uri = utils.projectContextUrl(req, utils.confdPort(api_server) + '/api/operations/rw-pkg-mgmt:package-file-delete');
636 payload.input = utils.addProjectContextToRPCPayload(req, uri, payload.input);
637 return new Promise(function(resolve, reject) {
638 rp({
639 uri: uri,
640 method: 'POST',
641 headers: _.extend({}, constants.HTTP_HEADERS.accept.collection, {
642 'Authorization': req.session && req.session.authorization
643 }),
644 json: payload,
645 forever: constants.FOREVER_ON,
646 rejectUnauthorized: false,
647 resolveWithFullResponse: true
648 }).then(function(data) {
649 if (utils.validateResponse('FileManager.delete', data.error, data, data.body, resolve, reject)) {
650 resolve({
651 statusCode: data.statusCode,
652 data: data.body
653 });
654 }
655 })
656 })
657 }
658 function download(payload) {
659 var uri = utils.projectContextUrl(req, utils.confdPort(api_server) + '/api/operations/rw-pkg-mgmt:package-file-add');
660 payload.input = utils.addProjectContextToRPCPayload(req, uri, payload.input);
661 return new Promise(function(resolve, reject) {
662 rp({
663 uri: uri,
664 method: 'POST',
665 headers: _.extend({}, constants.HTTP_HEADERS.accept.collection, {
666 'Authorization': req.session && req.session.authorization
667 }),
668 json: payload,
669 forever: constants.FOREVER_ON,
670 rejectUnauthorized: false,
671 resolveWithFullResponse: true
672 }).then(function(data) {
673 if (utils.validateResponse('FileManager.get', data.error, data, data.body, resolve, reject)) {
674 resolve({
675 statusCode: data.statusCode,
676 data: data.body
677 });
678 }
679 })
680 })
681 }
682 function list(payload) {
683 var uri = utils.projectContextUrl(req, utils.confdPort(api_server) + '/api/operations/get-package-endpoint');
684 payload.input = utils.addProjectContextToRPCPayload(req, uri, payload.input);
685 return new Promise(function(resolve, reject) {
686 rp({
687 uri: uri,
688 method: 'POST',
689 headers: _.extend({}, constants.HTTP_HEADERS.accept.collection, {
690 'Authorization': req.session && req.session.authorization
691 }),
692 json: payload,
693 forever: constants.FOREVER_ON,
694 rejectUnauthorized: false,
695 resolveWithFullResponse: true
696 }).then(function(data) {
697 if (utils.validateResponse('FileManager.get', data.error, data, data.body, resolve, reject)) {
698 var endpoint = null;
699 var parsedEndpoint = null;
700 try {
701 endpoint = data.body.output.endpoint
702 } catch(e) {
703 console.log('Something went wrong with the FileManager.get data that was returned');
704 reject({});
705 }
706 parsedEndpoint = URL.parse(endpoint);
707 rp({
708 uri: utils.projectContextUrl(req, api_server + ':' + parsedEndpoint.port + parsedEndpoint.path),
709 method: 'GET',
710 headers: _.extend({}, constants.HTTP_HEADERS.accept.collection, {
711 'Authorization': req.session && req.session.authorization
712 }),
713 forever: constants.FOREVER_ON,
714 rejectUnauthorized: false,
715 resolveWithFullResponse: true
716 }).then(function(data) {
717 if (utils.validateResponse('FileManager.get', data.error, data, data.body, resolve, reject)) {
718 resolve({
719 statusCode: data.statusCode,
720 data: data.body
721 });
722 }
723 }).catch(function(err) {
724 console.log('Failed to retrieve FileManager.list')
725 resolve(err);
726 })
727 }
728 })
729 })
730 }
731 }
732 FileManager.job = function(req) {
733 var api_server = req.query["api_server"];
734 var uri = utils.confdPort(api_server);
735 var url = '/api/operational/download-jobs';
736 var id = req.params['id'];
737 return new Promise(function(resolve, reject) {
738 request({
739 url: utils.projectContextUrl(req, uri + url + '?deep'),
740 method: 'GET',
741 headers: _.extend({}, constants.HTTP_HEADERS.accept.data, {
742 'Authorization': req.session && req.session.authorization
743 }),
744 forever: constants.FOREVER_ON,
745 rejectUnauthorized: false,
746 }, function(error, response, body) {
747 if (utils.validateResponse('restconfAPI.streams', error, response, body, resolve, reject)) {
748 var data = JSON.parse(response.body)['rw-pkg-mgmt:download-jobs'];
749 var returnData = [];
750 data && data.job.map(function(d) {
751 if(d['package-id'] == id) {
752 returnData.push(d)
753 }
754 })
755 resolve({
756 statusCode: response.statusCode,
757 data: returnData
758 })
759 };
760 })
761 })
762 }
763 module.exports = {
764 Composer:Composer,
765 FileManager: FileManager,
766 PackageManager: PackageManager
767 };