import Adapter, { DATACENTER_QUERY_PARAM as API_DATACENTER_KEY } from './application'; import { FOREIGN_KEY as DATACENTER_KEY } from 'consul-ui/models/dc'; import { PRIMARY_KEY, SLUG_KEY } from 'consul-ui/models/node'; import { OK as HTTP_OK } from 'consul-ui/utils/http/status'; // TODO: Looks like ID just isn't used at all // consider just using .Node for the SLUG_KEY const fillSlug = function(item) { if (item[SLUG_KEY] === '') { item[SLUG_KEY] = item['Node']; } return item; }; export default Adapter.extend({ urlForQuery: function(query, modelName) { return this.appendURL('internal/ui/nodes', [], this.cleanQuery(query)); }, urlForQueryRecord: function(query, modelName) { if (typeof query.id === 'undefined') { throw new Error('You must specify an id'); } return this.appendURL('internal/ui/node', [query.id], this.cleanQuery(query)); }, urlForRequest: function({ type, snapshot, requestType }) { switch (requestType) { case 'queryLeader': return this.urlForQueryLeader(snapshot, type.modelName); } return this._super(...arguments); }, urlForQueryLeader: function(query, modelName) { // https://www.consul.io/api/status.html#get-raft-leader return this.appendURL('status/leader', [], this.cleanQuery(query)); }, isQueryLeader: function(url, method) { return url.pathname === this.parseURL(this.urlForQueryLeader({})).pathname; }, queryLeader: function(store, modelClass, id, snapshot) { const params = { store: store, type: modelClass, id: id, snapshot: snapshot, requestType: 'queryLeader', }; // _requestFor is private... but these methods aren't, until they disappear.. const request = { method: this.methodForRequest(params), url: this.urlForRequest(params), headers: this.headersForRequest(params), data: this.dataForRequest(params), }; // TODO: private.. return this._makeRequest(request); }, handleBatchResponse: function(url, response, primary, slug) { const dc = url.searchParams.get(API_DATACENTER_KEY) || ''; return response.map((item, i, arr) => { // this can go in the serializer item = fillSlug(item); // this could be replaced by handleSingleResponse // maybe perf test first although even polyfilled searchParams should be super fast return { ...item, ...{ [DATACENTER_KEY]: dc, [PRIMARY_KEY]: this.uidForURL(url, item[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); let temp, port, address; switch (true) { case this.isQueryLeader(url, method): // This response is just an ip:port like `"10.0.0.1:8000"` // split it and make it look like a `C`onsul.`R`esponse // popping off the end for ports should cover us for IPv6 addresses // as we should always get a `address:port` or `[a:dd:re:ss]:port` combo temp = response.split(':'); port = temp.pop(); address = temp.join(':'); response = { Address: address, Port: port, }; break; case this.isQueryRecord(url, method): response = this.handleSingleResponse(url, fillSlug(response), PRIMARY_KEY, SLUG_KEY); break; default: response = this.handleBatchResponse(url, response, PRIMARY_KEY, SLUG_KEY); } } return this._super(status, headers, response, requestData); }, });