fuel-ui/static/utils.js

352 lines
12 KiB
JavaScript

/*
* Copyright 2013 Mirantis, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may
* not use this file except in compliance with the License. You may obtain
* a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
**/
import _ from 'underscore';
import i18n from 'i18n';
import React from 'react';
import classNames from 'classnames';
import naturalSort from 'javascript-natural-sort';
import IP from 'ip';
import moment from 'moment';
import {ErrorDialog} from 'views/dialogs';
import models from 'models';
var utils = {
/*eslint-disable max-len*/
regexes: {
url: /(?:https?:\/\/([\-\w\.]+)+(:\d+)?(\/([\w\/_\-\.]*(\?[\w\/_\-\.&%]*)?(#[\w\/_\-\.&%]*)?)?)?)/,
ip: /^(([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\.){3}([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])$/,
mac: /^([0-9a-f]{1,2}[\.:-]){5}([0-9a-f]{1,2})$/,
cidr: /^(?:(?:[0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\.){3}(?:[0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\/([1-9]|[1-2]\d|3[0-2])$/,
networkName: /^[A-Za-z0-9 _]*[A-Za-z0-9][A-Za-z0-9 _]*$/
},
/*eslint-enable max-len*/
serializeTabOptions(options) {
return _.map(options, (value, key) => key + ':' + value).join(';');
},
deserializeTabOptions(serializedOptions) {
return _.fromPairs(_.map((serializedOptions || '').split(';'), (option) => option.split(':')));
},
getNodeListFromTabOptions(options) {
var nodeIds = utils.deserializeTabOptions(options.screenOptions[0]).nodes;
var ids = nodeIds ? nodeIds.split(',').map((id) => parseInt(id, 10)) : [];
var nodes = new models.Nodes(
options.cluster.get('nodes').getByIds(ids),
{fetchOptions: {cluster_id: options.cluster.id}}
);
return nodes.length === ids.length ? nodes : null;
},
renderMultilineText(text) {
if (!text) return null;
return <div>{text.split('\n').map((str, index) => <p key={index}>{str}</p>)}</div>;
},
linebreaks(text) {
return text.replace(/\n/g, '<br/>');
},
composeLink(url) {
return '<a target="_blank" href="' + url + '">' + url + '</a>';
},
urlify(text) {
return utils.linebreaks(text).replace(new RegExp(utils.regexes.url.source, 'g'),
utils.composeLink);
},
composeList(value) {
return _.isUndefined(value) ? [] : _.isArray(value) ? value : [value];
},
// FIXME(vkramskikh): moved here from healthcheck_tab to make testable
highlightTestStep(text, step) {
return text.replace(new RegExp('(^|\\s*)(' + step + '\\.[\\s\\S]*?)(\\s*\\d+\\.|$)'),
'$1<b>$2</b>$3');
},
classNames: classNames,
expandRestriction(restriction) {
var result = {
action: 'disable',
message: null
};
if (_.isString(restriction)) {
result.condition = restriction;
} else if (_.isPlainObject(restriction)) {
if (_.has(restriction, 'condition')) {
_.extend(result, restriction);
} else {
result.condition = _.keys(restriction)[0];
result.message = _.values(restriction)[0];
}
} else {
throw new Error('Invalid restriction format');
}
return result;
},
showErrorDialog(options) {
options.message = options.response ? utils.getResponseText(options.response) :
options.message || i18n('dialog.error_dialog.server_error');
ErrorDialog.show(options);
},
showBandwidth(bandwidth) {
bandwidth = parseInt(bandwidth, 10);
if (!_.isNumber(bandwidth) || _.isNaN(bandwidth)) return i18n('common.not_available');
return (bandwidth / 1000).toFixed(1) + ' Gbps';
},
showFrequency(frequency) {
frequency = parseInt(frequency, 10);
if (!_.isNumber(frequency) || _.isNaN(frequency)) return i18n('common.not_available');
var base = 1000;
var treshold = 1000;
return (frequency >= treshold ? (frequency / base).toFixed(2) + ' GHz' : frequency + ' MHz');
},
showSize(bytes, power = 0, treshold = 1024) {
bytes = parseInt(bytes, 10);
if (!_.isNumber(bytes) || _.isNaN(bytes)) return i18n('common.not_available');
bytes *= Math.pow(1024, power);
var base = 1024;
var units = ['byte', 'kb', 'mb', 'gb', 'tb'];
var i, result;
var unit = 'tb';
for (i = 0; i < units.length; i += 1) {
result = bytes / Math.pow(base, i);
if (result < treshold) {
unit = units[i];
break;
}
}
return (result ? result.toFixed(1) : result) + ' ' + i18n('common.size.' + unit,
{count: result});
},
formatNumber(n) {
return String(n).replace(/\d/g, (c, i, a) => i > 0 && c !== '.' &&
(a.length - i) % 3 === 0 ? ',' + c : c);
},
floor(n, decimals) {
return Math.floor(n * Math.pow(10, decimals)) / Math.pow(10, decimals);
},
isNaturalNumber(n) {
return _.isNumber(n) && n > 0 && n % 1 === 0;
},
validateVlan(vlan) {
if (
!_.isNull(vlan) &&
(!utils.isNaturalNumber(vlan) || vlan < 1 || vlan > 4094)
) {
return {vlan_start: i18n('cluster_page.network_tab.validation.invalid_vlan')};
}
return {};
},
validateCidr(cidr, attributeName = 'cidr') {
var error = {};
var match;
if (_.isString(cidr)) {
match = cidr.match(utils.regexes.cidr);
if (match) {
var prefix = parseInt(match[1], 10);
if (prefix < 2) {
error[attributeName] = i18n('cluster_page.network_tab.validation.large_network');
} else if (prefix > 30) {
error[attributeName] = i18n('cluster_page.network_tab.validation.small_network');
}
} else {
error[attributeName] = i18n('cluster_page.network_tab.validation.invalid_cidr');
}
} else {
error[attributeName] = i18n('cluster_page.network_tab.validation.invalid_cidr');
}
return error[attributeName] ? error : {};
},
validateGateway(gateway, cidr, attributeName = 'gateway') {
if (!utils.validateIP(gateway)) {
return {
[attributeName]: i18n('cluster_page.network_tab.validation.invalid_gateway')
};
}
if (cidr && !utils.validateIpCorrespondsToCIDR(cidr, gateway)) {
return {
[attributeName]: i18n('cluster_page.network_tab.validation.gateway_does_not_match_cidr')
};
}
return null;
},
validateIP(ip) {
return _.isString(ip) && !!ip.match(utils.regexes.ip);
},
validateIPRanges(ranges, cidr, existingRanges = [], warnings = {}) {
var ipRangesErrors = [];
var ns = 'cluster_page.network_tab.validation.';
_.defaults(warnings, {
INVALID_IP: i18n(ns + 'invalid_ip'),
DOES_NOT_MATCH_CIDR: i18n(ns + 'ip_does_not_match_cidr'),
INVALID_IP_RANGE: i18n(ns + 'invalid_ip_range'),
EMPTY_IP_RANGE: i18n(ns + 'empty_ip_range'),
IP_RANGES_INTERSECTION: i18n(ns + 'ip_ranges_intersection')
});
if (_.some(ranges, (range) => _.compact(range).length)) {
_.each(ranges, (range, index) => {
if (_.some(range)) {
var error = {};
_.each(range, (ip, ipIndex) => {
var errorKey = !ipIndex ? 'start' : 'end';
if (!utils.validateIP(ip)) {
error[errorKey] = warnings.INVALID_IP;
} else if (cidr && !utils.validateIpCorrespondsToCIDR(cidr, ip)) {
error[errorKey] = warnings.DOES_NOT_MATCH_CIDR;
}
});
if (_.isEmpty(error)) {
if (IP.toLong(range[0]) > IP.toLong(range[1])) {
error.start = error.end = warnings.INVALID_IP_RANGE;
} else if (existingRanges.length) {
var intersection = utils.checkIPRangesIntersection(range, existingRanges);
if (intersection) {
error.start = error.end = warnings.IP_RANGES_INTERSECTION +
intersection.join(' - ');
}
}
}
if (!_.isEmpty(error)) {
ipRangesErrors.push(_.extend(error, {index: index}));
}
}
});
} else {
ipRangesErrors.push({
index: 0,
start: warnings.EMPTY_IP_RANGE,
end: warnings.EMPTY_IP_RANGE
});
}
return ipRangesErrors;
},
checkIPRangesIntersection([startIP, endIP], existingRanges) {
var startIPInt = IP.toLong(startIP);
var endIPInt = IP.toLong(endIP);
return _.find(existingRanges, ([ip1, ip2]) => {
return IP.toLong(ip2) >= startIPInt && IP.toLong(ip1) <= endIPInt;
});
},
validateIpCorrespondsToCIDR(cidr, ip) {
if (!cidr) return true;
var networkData = IP.cidrSubnet(cidr);
var ipInt = IP.toLong(ip);
return ipInt >= IP.toLong(networkData.firstAddress) &&
ipInt <= IP.toLong(networkData.lastAddress);
},
validateVlanRange(vlanStart, vlanEnd, vlan) {
return vlan >= vlanStart && vlan <= vlanEnd;
},
getDefaultGatewayForCidr(cidr) {
if (!_.isEmpty(utils.validateCidr(cidr))) return '';
return IP.cidrSubnet(cidr).firstAddress;
},
getDefaultIPRangeForCidr(cidr, excludeGateway) {
if (!_.isEmpty(utils.validateCidr(cidr))) return [['', '']];
var networkData = IP.cidrSubnet(cidr);
if (excludeGateway) {
var startIPInt = IP.toLong(networkData.firstAddress);
startIPInt++;
return [[IP.fromLong(startIPInt), networkData.lastAddress]];
}
return [[networkData.firstAddress, networkData.lastAddress]];
},
sortEntryProperties(entry, sortOrder) {
sortOrder = sortOrder || ['name'];
var properties = _.keys(entry);
return _.sortBy(properties, (property) => {
var index = _.indexOf(sortOrder, property);
return index === -1 ? properties.length : index;
});
},
getResponseText(response, defaultText) {
var serverErrorMessage = defaultText || i18n('dialog.error_dialog.server_error');
var serverUnavailableMessage = i18n('dialog.error_dialog.server_unavailable');
if (response && (!response.status || response.status >= 400)) {
if (!response.status || response.status === 502) return serverUnavailableMessage;
if (response.status === 500) return serverErrorMessage;
// parsing new backend response format in responseText
response = response.responseText || response;
try {
response = JSON.parse(response);
return response.message || serverErrorMessage;
} catch (exception) {
return serverErrorMessage;
}
}
return '';
},
natsort(str1, str2, options = {}) {
var {insensitive, desc} = options;
naturalSort.insensitive = insensitive;
return naturalSort(str1, str2) * (desc ? -1 : 1);
},
multiSort(model1, model2, attributes) {
var result = utils.compare(model1, model2, attributes[0]);
if (result === 0 && attributes.length > 1) {
attributes.splice(0, 1);
result = utils.multiSort(model1, model2, attributes);
}
return result;
},
compare(model1, model2, options) {
var getValue = (model) => {
var attr = options.attr;
return _.isFunction(model[attr]) ? model[attr]() : model.get(attr);
};
var value1 = getValue(model1);
var value2 = getValue(model2);
if (_.isString(value1) && _.isString(value2)) {
return utils.natsort(value1, value2, options);
}
var result;
if (_.isNumber(value1) && _.isNumber(value2)) {
result = value1 - value2;
} else {
result = value1 === value2 || !value1 && !value2 ? 0 : !value1 ? 1 : -1;
}
return options.desc ? -result : result;
},
makePath(...args) {
return args.join('.');
},
deepOmit(object, keys) {
if (!_.isObject(object) && !_.isArray(object)) {
return object;
}
return _.reduce(_.difference(_.keys(object), keys), (result, key) => {
result[key] = this.deepOmit(object[key], keys);
return result;
}, _.isArray(object) ? [] : {});
},
formatTimestamp(timestamp, showSeconds = true) {
return moment(timestamp).format(showSeconds ? 'HH:mm:ss DD/MM/YYYY' : 'HH:mm DD/MM/YYYY');
},
parseISO8601Date: _.memoize((date) => new Date(date + 'Z').getTime()),
parseRFC2822Date(date) {
return Number(moment.utc(date, 'ddd, D MMM YYYY H:mm:ss [GMT]', true));
},
getStringHashCode(str) {
var hash = 0;
for (var i = 0; i < str.length; i++) {
hash = ((hash << 5) - hash) + str.charCodeAt(i);
hash = hash & hash; // convert to 32bit integer
}
return hash;
}
};
export default utils;