7903512b26
* Add some tests to check the correct GET API endpoints are called * Refactor adapters 1. Add integration tests for `urlFor...` and majority `handleResponse` methods 2. Refactor out `handleResponse` a little more into single/batch/boolean methods 3. Move setting of the `Datacenter` property into the `handleResponse` method, basically the same place that the uid is being set using the dc parsed form the URL 4. Add some Errors for if you don't pass ids to certain `urlFor` methods
150 lines
5.2 KiB
JavaScript
150 lines
5.2 KiB
JavaScript
import Adapter, {
|
|
REQUEST_CREATE,
|
|
REQUEST_UPDATE,
|
|
REQUEST_DELETE,
|
|
DATACENTER_QUERY_PARAM as API_DATACENTER_KEY,
|
|
} from './application';
|
|
import isFolder from 'consul-ui/utils/isFolder';
|
|
import injectableRequestToJQueryAjaxHash from 'consul-ui/utils/injectableRequestToJQueryAjaxHash';
|
|
import { typeOf } from '@ember/utils';
|
|
import { get } from '@ember/object';
|
|
import { inject as service } from '@ember/service';
|
|
|
|
import keyToArray from 'consul-ui/utils/keyToArray';
|
|
import removeNull from 'consul-ui/utils/remove-null';
|
|
|
|
import { PRIMARY_KEY, SLUG_KEY } from 'consul-ui/models/kv';
|
|
import { FOREIGN_KEY as DATACENTER_KEY } from 'consul-ui/models/dc';
|
|
import { PUT as HTTP_PUT, DELETE as HTTP_DELETE } from 'consul-ui/utils/http/method';
|
|
import { OK as HTTP_OK } from 'consul-ui/utils/http/status';
|
|
|
|
const API_KEYS_KEY = 'keys';
|
|
const stringify = function(obj) {
|
|
if (typeOf(obj) === 'string') {
|
|
return obj;
|
|
}
|
|
return JSON.stringify(obj);
|
|
};
|
|
export default Adapter.extend({
|
|
// There is no code path that can avoid the payload of a PUT request from
|
|
// going via JSON.stringify.
|
|
// Therefore a string payload of 'foobar' will always be encoded to '"foobar"'
|
|
//
|
|
// This means we have no other choice but rewriting the entire codepath or
|
|
// overwriting the private `_requestToJQueryAjaxHash` method
|
|
//
|
|
// The `injectableRequestToJQueryAjaxHash` function makes the JSON object
|
|
// injectable, meaning we can copy letter for letter the sourcecode of
|
|
// `_requestToJQueryAjaxHash`, which means we can compare it with the original
|
|
// private method within a test (`tests/unit/utils/injectableRequestToJQueryAjaxHash.js`).
|
|
// This means, if `_requestToJQueryAjaxHash` changes between Ember versions
|
|
// we will know about it
|
|
|
|
_requestToJQueryAjaxHash: injectableRequestToJQueryAjaxHash({
|
|
stringify: stringify,
|
|
}),
|
|
decoder: service('atob'),
|
|
urlForQuery: function(query, modelName) {
|
|
if (typeof query.id === 'undefined') {
|
|
throw new Error('You must specify an id');
|
|
}
|
|
// append keys here otherwise query.keys will add an '='
|
|
return this.appendURL('kv', keyToArray(query.id), {
|
|
...{
|
|
[API_KEYS_KEY]: null,
|
|
},
|
|
...this.cleanQuery(query),
|
|
});
|
|
},
|
|
urlForQueryRecord: function(query, modelName) {
|
|
if (typeof query.id === 'undefined') {
|
|
throw new Error('You must specify an id');
|
|
}
|
|
return this.appendURL('kv', keyToArray(query.id), this.cleanQuery(query));
|
|
},
|
|
urlForCreateRecord: function(modelName, snapshot) {
|
|
return this.appendURL('kv', keyToArray(snapshot.attr(SLUG_KEY)), {
|
|
[API_DATACENTER_KEY]: snapshot.attr(DATACENTER_KEY),
|
|
});
|
|
},
|
|
urlForUpdateRecord: function(id, modelName, snapshot) {
|
|
return this.appendURL('kv', keyToArray(snapshot.attr(SLUG_KEY)), {
|
|
[API_DATACENTER_KEY]: snapshot.attr(DATACENTER_KEY),
|
|
});
|
|
},
|
|
urlForDeleteRecord: function(id, modelName, snapshot) {
|
|
const query = {
|
|
[API_DATACENTER_KEY]: snapshot.attr(DATACENTER_KEY),
|
|
};
|
|
if (isFolder(snapshot.attr(SLUG_KEY))) {
|
|
query.recurse = null;
|
|
}
|
|
return this.appendURL('kv', keyToArray(snapshot.attr(SLUG_KEY)), query);
|
|
},
|
|
slugFromURL: function(url) {
|
|
// keys don't follow the 'last part of the url' rule as they contain slashes
|
|
return decodeURIComponent(
|
|
url.pathname
|
|
.split('/')
|
|
.splice(3)
|
|
.join('/')
|
|
);
|
|
},
|
|
isQueryRecord: function(url, method) {
|
|
return !url.searchParams.has(API_KEYS_KEY);
|
|
},
|
|
handleBatchResponse: function(url, response, primary, slug) {
|
|
const dc = url.searchParams.get(API_DATACENTER_KEY) || '';
|
|
return response.map((item, i, arr) => {
|
|
return {
|
|
[DATACENTER_KEY]: dc,
|
|
[PRIMARY_KEY]: this.uidForURL(url, item),
|
|
[SLUG_KEY]: item,
|
|
};
|
|
});
|
|
},
|
|
// handleSingleResponse: function(url, response, primary, slug) {
|
|
// return this._super(url, removeNull(response[0]), PRIMARY_KEY, SLUG_KEY);
|
|
// },
|
|
handleResponse: function(status, headers, payload, requestData) {
|
|
let response = payload;
|
|
const method = requestData.method;
|
|
if (status === HTTP_OK) {
|
|
const url = this.parseURL(requestData.url);
|
|
switch (true) {
|
|
case response === true:
|
|
response = this.handleBooleanResponse(url, response, PRIMARY_KEY, SLUG_KEY);
|
|
break;
|
|
case this.isQueryRecord(url, method):
|
|
response = this.handleSingleResponse(url, removeNull(response[0]), PRIMARY_KEY, SLUG_KEY);
|
|
break;
|
|
default:
|
|
response = this.handleBatchResponse(url, response, PRIMARY_KEY, SLUG_KEY);
|
|
}
|
|
}
|
|
return this._super(status, headers, response, requestData);
|
|
},
|
|
dataForRequest: function(params) {
|
|
const data = this._super(...arguments);
|
|
let value = '';
|
|
switch (params.requestType) {
|
|
case REQUEST_UPDATE:
|
|
case REQUEST_CREATE:
|
|
value = data.kv.Value;
|
|
if (typeof value === 'string') {
|
|
return get(this, 'decoder').execute(value);
|
|
}
|
|
return null;
|
|
}
|
|
},
|
|
methodForRequest: function(params) {
|
|
switch (params.requestType) {
|
|
case REQUEST_DELETE:
|
|
return HTTP_DELETE;
|
|
case REQUEST_CREATE:
|
|
return HTTP_PUT;
|
|
}
|
|
return this._super(...arguments);
|
|
},
|
|
});
|