linkerd2/web/app/js/components/util/ApiHelpers.jsx

238 lines
5.7 KiB
JavaScript

import 'whatwg-fetch';
import { Link } from 'react-router-dom';
import PropTypes from 'prop-types';
import React from 'react';
import _each from 'lodash/each';
import _isEmpty from 'lodash/isEmpty';
import _isNil from 'lodash/isNil';
import _map from 'lodash/map';
const checkFetchOk = resp => {
if (resp.ok) {
return resp;
}
return resp.json().then(error => {
throw {
status: resp.status,
url: resp.url,
statusText: resp.statusText,
error: error.error
};
});
};
// makeCancelable from @istarkov
// https://reactjs.org/blog/2015/12/16/ismounted-antipattern.html
const makeCancelable = (promise, onSuccess) => {
let hasCanceled_ = false;
const wrappedPromise = new Promise((resolve, reject) => {
return promise.then(
result => hasCanceled_ ? reject({ isCanceled: true }) : resolve(result),
error => hasCanceled_ ? reject({ isCanceled: true }) : reject(error)
);
})
.then(checkFetchOk)
.then(onSuccess);
return {
promise: wrappedPromise,
cancel: () => {
hasCanceled_ = true;
},
status: () => {
return hasCanceled_;
}
};
};
export const apiErrorPropType = PropTypes.shape({
status: PropTypes.number,
url: PropTypes.string,
statusText: PropTypes.string,
error: PropTypes.string
});
const ApiHelpers = (pathPrefix, defaultMetricsWindow = '1m') => {
let metricsWindow = defaultMetricsWindow;
const podsPath = `/api/pods`;
const servicesPath = `/api/services`;
const edgesPath = `/api/edges`;
const validMetricsWindows = {
"10s": "10 minutes",
"1m": "1 minute",
"10m": "10 minutes",
"1h": "1 hour"
};
// for getting json api results
const apiFetch = path => {
if (!_isEmpty(pathPrefix)) {
path = `${pathPrefix}${path}`;
}
return makeCancelable(fetch(path), r => r.json());
};
// for getting non-json results
const prefixedUrl = path => {
if (!_isEmpty(pathPrefix)) {
path = `${pathPrefix}${path}`;
}
return path;
};
const fetchMetrics = path => {
if (path.indexOf("window") === -1) {
if (path.indexOf("?") === -1) {
path = `${path}?window=${getMetricsWindow()}`;
} else {
path = `${path}&window=${getMetricsWindow()}`;
}
}
return apiFetch(path);
};
const fetchPods = namespace => {
if (!_isNil(namespace)) {
return apiFetch(podsPath + "?namespace=" + namespace);
}
return apiFetch(podsPath);
};
const fetchServices = namespace => {
if (!_isNil(namespace)) {
return apiFetch(servicesPath + "?namespace=" + namespace);
}
return apiFetch(servicesPath);
};
const fetchEdges = (namespace, resourceType) => {
return apiFetch(edgesPath + "?namespace=" + namespace + "&resource_type=" + resourceType);
};
const getMetricsWindow = () => metricsWindow;
const getMetricsWindowDisplayText = () => validMetricsWindows[metricsWindow];
const setMetricsWindow = window => {
if (!validMetricsWindows[window]) { return; }
metricsWindow = window;
};
const urlsForResource = (type, namespace, includeTcp) => {
// Traffic Performance Summary. This retrieves stats for the given resource.
let resourceUrl = '/api/tps-reports?resource_type=' + type;
if (_isEmpty(namespace)) {
resourceUrl += '&all_namespaces=true';
} else {
resourceUrl += '&namespace=' + namespace;
}
if (includeTcp) {
resourceUrl += '&tcp_stats=true';
}
return resourceUrl;
};
// maintain a list of a component's requests,
// convenient for providing a cancel() functionality
let currentRequests = [];
const setCurrentRequests = cancelablePromises => {
currentRequests = cancelablePromises;
};
const getCurrentPromises = () => {
return _map(currentRequests, r => r.promise);
};
const cancelCurrentRequests = () => {
_each(currentRequests, promise => {
promise.cancel();
});
};
// prefix all links in the app with `pathPrefix`
class PrefixedLink extends React.Component {
static defaultProps = {
targetBlank: false,
}
static propTypes = {
children: PropTypes.node.isRequired,
targetBlank: PropTypes.bool,
to: PropTypes.string.isRequired,
}
render() {
let url = prefixLink(this.props.to);
return (
<Link
to={url}
{...(this.props.targetBlank ? { target: '_blank' } : {})}>
{this.props.children}
</Link>
);
}
}
const prefixLink = to => `${pathPrefix}${to}`;
const generateResourceURL = r => {
if (r.type === "namespace") {
return "/namespaces/" + (r.namespace || r.name);
}
return "/namespaces/" + r.namespace + "/" + r.type + "s/" + r.name;
};
// a prefixed link to a Resource Detail page
const ResourceLink = ({ resource, linkText }) => {
return (
<PrefixedLink to={generateResourceURL(resource)}>
{linkText || resource.type + "/" + resource.name}
</PrefixedLink>
);
};
ResourceLink.propTypes = {
linkText: PropTypes.string,
resource: PropTypes.shape({
name: PropTypes.string,
namespace: PropTypes.string,
type: PropTypes.string,
})
};
ResourceLink.defaultProps = {
resource: {},
linkText: ""
};
return {
fetch: apiFetch,
prefixedUrl,
fetchMetrics,
fetchPods,
fetchServices,
fetchEdges,
getMetricsWindow,
setMetricsWindow,
getValidMetricsWindows: () => Object.keys(validMetricsWindows),
getMetricsWindowDisplayText,
urlsForResource,
PrefixedLink,
prefixLink,
ResourceLink,
setCurrentRequests,
getCurrentPromises,
generateResourceURL,
cancelCurrentRequests,
// DO NOT USE makeCancelable, use fetch, this is only exposed for testing
makeCancelable
};
};
export default ApiHelpers;