datasource.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541
  1. /* global AWS */
  2. define([
  3. 'angular',
  4. 'lodash',
  5. 'kbn',
  6. 'moment',
  7. './queryCtrl',
  8. 'aws-sdk',
  9. ],
  10. function (angular, _, kbn) {
  11. 'use strict';
  12. var module = angular.module('grafana.services');
  13. module.factory('CloudWatchDatasource', function($q, $http, templateSrv) {
  14. function CloudWatchDatasource(datasource) {
  15. this.type = 'cloudwatch';
  16. this.name = datasource.name;
  17. this.supportMetrics = true;
  18. this.proxyMode = (datasource.jsonData.access === 'proxy');
  19. this.proxyUrl = datasource.url;
  20. this.defaultRegion = datasource.jsonData.defaultRegion;
  21. this.credentials = {
  22. accessKeyId: datasource.jsonData.accessKeyId,
  23. secretAccessKey: datasource.jsonData.secretAccessKey
  24. };
  25. /* jshint -W101 */
  26. this.supportedRegion = [
  27. 'us-east-1', 'us-west-2', 'us-west-1', 'eu-west-1', 'eu-central-1', 'ap-southeast-1', 'ap-southeast-2', 'ap-northeast-1', 'sa-east-1'
  28. ];
  29. this.supportedMetrics = {
  30. 'AWS/AutoScaling': [
  31. 'GroupMinSize', 'GroupMaxSize', 'GroupDesiredCapacity', 'GroupInServiceInstances', 'GroupPendingInstances', 'GroupStandbyInstances', 'GroupTerminatingInstances', 'GroupTotalInstances'
  32. ],
  33. 'AWS/Billing': [
  34. 'EstimatedCharges'
  35. ],
  36. 'AWS/CloudFront': [
  37. 'Requests', 'BytesDownloaded', 'BytesUploaded', 'TotalErrorRate', '4xxErrorRate', '5xxErrorRate'
  38. ],
  39. 'AWS/CloudSearch': [
  40. 'SuccessfulRequests', 'SearchableDocuments', 'IndexUtilization', 'Partitions'
  41. ],
  42. 'AWS/DynamoDB': [
  43. 'ConditionalCheckFailedRequests', 'ConsumedReadCapacityUnits', 'ConsumedWriteCapacityUnits', 'OnlineIndexConsumedWriteCapacity', 'OnlineIndexPercentageProgress', 'OnlineIndexThrottleEvents', 'ProvisionedReadCapacityUnits', 'ProvisionedWriteCapacityUnits', 'ReadThrottleEvents', 'ReturnedItemCount', 'SuccessfulRequestLatency', 'SystemErrors', 'ThrottledRequests', 'UserErrors', 'WriteThrottleEvents'
  44. ],
  45. 'AWS/ElastiCache': [
  46. 'CPUUtilization', 'SwapUsage', 'FreeableMemory', 'NetworkBytesIn', 'NetworkBytesOut',
  47. 'BytesUsedForCacheItems', 'BytesReadIntoMemcached', 'BytesWrittenOutFromMemcached', 'CasBadval', 'CasHits', 'CasMisses', 'CmdFlush', 'CmdGet', 'CmdSet', 'CurrConnections', 'CurrItems', 'DecrHits', 'DecrMisses', 'DeleteHits', 'DeleteMisses', 'Evictions', 'GetHits', 'GetMisses', 'IncrHits', 'IncrMisses', 'Reclaimed',
  48. 'CurrConnections', 'Evictions', 'Reclaimed', 'NewConnections', 'BytesUsedForCache', 'CacheHits', 'CacheMisses', 'ReplicationLag', 'GetTypeCmds', 'SetTypeCmds', 'KeyBasedCmds', 'StringBasedCmds', 'HashBasedCmds', 'ListBasedCmds', 'SetBasedCmds', 'SortedSetBasedCmds', 'CurrItems'
  49. ],
  50. 'AWS/EBS': [
  51. 'VolumeReadBytes', 'VolumeWriteBytes', 'VolumeReadOps', 'VolumeWriteOps', 'VolumeTotalReadTime', 'VolumeTotalWriteTime', 'VolumeIdleTime', 'VolumeQueueLength', 'VolumeThroughputPercentage', 'VolumeConsumedReadWriteOps'
  52. ],
  53. 'AWS/EC2': [
  54. 'CPUCreditUsage', 'CPUCreditBalance', 'CPUUtilization', 'DiskReadOps', 'DiskWriteOps', 'DiskReadBytes', 'DiskWriteBytes', 'NetworkIn', 'NetworkOut', 'StatusCheckFailed', 'StatusCheckFailed_Instance', 'StatusCheckFailed_System'
  55. ],
  56. 'AWS/ELB': [
  57. 'HealthyHostCount', 'UnHealthyHostCount', 'RequestCount', 'Latency', 'HTTPCode_ELB_4XX', 'HTTPCode_ELB_5XX', 'HTTPCode_Backend_2XX', 'HTTPCode_Backend_3XX', 'HTTPCode_Backend_4XX', 'HTTPCode_Backend_5XX', 'BackendConnectionErrors', 'SurgeQueueLength', 'SpilloverCount'
  58. ],
  59. 'AWS/ElasticMapReduce': [
  60. 'CoreNodesPending', 'CoreNodesRunning', 'HBaseBackupFailed', 'HBaseMostRecentBackupDuration', 'HBaseTimeSinceLastSuccessfulBackup', 'HDFSBytesRead', 'HDFSBytesWritten', 'HDFSUtilization', 'IsIdle', 'JobsFailed', 'JobsRunning', 'LiveDataNodes', 'LiveTaskTrackers', 'MapSlotsOpen', 'MissingBlocks', 'ReduceSlotsOpen', 'RemainingMapTasks', 'RemainingMapTasksPerSlot', 'RemainingReduceTasks', 'RunningMapTasks', 'RunningReduceTasks', 'S3BytesRead', 'S3BytesWritten', 'TaskNodesPending', 'TaskNodesRunning', 'TotalLoad'
  61. ],
  62. 'AWS/Kinesis': [
  63. 'PutRecord.Bytes', 'PutRecord.Latency', 'PutRecord.Success', 'PutRecords.Bytes', 'PutRecords.Latency', 'PutRecords.Records', 'PutRecords.Success', 'IncomingBytes', 'IncomingRecords', 'GetRecords.Bytes', 'GetRecords.IteratorAgeMilliseconds', 'GetRecords.Latency', 'GetRecords.Success'
  64. ],
  65. 'AWS/ML': [
  66. 'PredictCount', 'PredictFailureCount'
  67. ],
  68. 'AWS/OpsWorks': [
  69. 'cpu_idle', 'cpu_nice', 'cpu_system', 'cpu_user', 'cpu_waitio', 'load_1', 'load_5', 'load_15', 'memory_buffers', 'memory_cached', 'memory_free', 'memory_swap', 'memory_total', 'memory_used', 'procs'
  70. ],
  71. 'AWS/Redshift': [
  72. 'CPUUtilization', 'DatabaseConnections', 'HealthStatus', 'MaintenanceMode', 'NetworkReceiveThroughput', 'NetworkTransmitThroughput', 'PercentageDiskSpaceUsed', 'ReadIOPS', 'ReadLatency', 'ReadThroughput', 'WriteIOPS', 'WriteLatency', 'WriteThroughput'
  73. ],
  74. 'AWS/RDS': [
  75. 'BinLogDiskUsage', 'CPUUtilization', 'DatabaseConnections', 'DiskQueueDepth', 'FreeableMemory', 'FreeStorageSpace', 'ReplicaLag', 'SwapUsage', 'ReadIOPS', 'WriteIOPS', 'ReadLatency', 'WriteLatency', 'ReadThroughput', 'WriteThroughput', 'NetworkReceiveThroughput', 'NetworkTransmitThroughput'
  76. ],
  77. 'AWS/Route53': [
  78. 'HealthCheckStatus', 'HealthCheckPercentageHealthy'
  79. ],
  80. 'AWS/SNS': [
  81. 'NumberOfMessagesPublished', 'PublishSize', 'NumberOfNotificationsDelivered', 'NumberOfNotificationsFailed'
  82. ],
  83. 'AWS/SQS': [
  84. 'NumberOfMessagesSent', 'SentMessageSize', 'NumberOfMessagesReceived', 'NumberOfEmptyReceives', 'NumberOfMessagesDeleted', 'ApproximateNumberOfMessagesDelayed', 'ApproximateNumberOfMessagesVisible', 'ApproximateNumberOfMessagesNotVisible'
  85. ],
  86. 'AWS/S3': [
  87. 'BucketSizeBytes', 'NumberOfObjects'
  88. ],
  89. 'AWS/SWF': [
  90. 'DecisionTaskScheduleToStartTime', 'DecisionTaskStartToCloseTime', 'DecisionTasksCompleted', 'StartedDecisionTasksTimedOutOnClose', 'WorkflowStartToCloseTime', 'WorkflowsCanceled', 'WorkflowsCompleted', 'WorkflowsContinuedAsNew', 'WorkflowsFailed', 'WorkflowsTerminated', 'WorkflowsTimedOut'
  91. ],
  92. 'AWS/StorageGateway': [
  93. 'CacheHitPercent', 'CachePercentUsed', 'CachePercentDirty', 'CloudBytesDownloaded', 'CloudDownloadLatency', 'CloudBytesUploaded', 'UploadBufferFree', 'UploadBufferPercentUsed', 'UploadBufferUsed', 'QueuedWrites', 'ReadBytes', 'ReadTime', 'TotalCacheSize', 'WriteBytes', 'WriteTime', 'WorkingStorageFree', 'WorkingStoragePercentUsed', 'WorkingStorageUsed', 'CacheHitPercent', 'CachePercentUsed', 'CachePercentDirty', 'ReadBytes', 'ReadTime', 'WriteBytes', 'WriteTime', 'QueuedWrites'
  94. ],
  95. 'AWS/WorkSpaces': [
  96. 'Available', 'Unhealthy', 'ConnectionAttempt', 'ConnectionSuccess', 'ConnectionFailure', 'SessionLaunchTime', 'InSessionLatency', 'SessionDisconnect'
  97. ],
  98. };
  99. this.supportedDimensions = {
  100. 'AWS/AutoScaling': [
  101. 'AutoScalingGroupName'
  102. ],
  103. 'AWS/Billing': [
  104. 'ServiceName', 'LinkedAccount', 'Currency'
  105. ],
  106. 'AWS/CloudFront': [
  107. 'DistributionId', 'Region'
  108. ],
  109. 'AWS/CloudSearch': [
  110. ],
  111. 'AWS/DynamoDB': [
  112. 'TableName', 'GlobalSecondaryIndexName', 'Operation'
  113. ],
  114. 'AWS/ElastiCache': [
  115. 'CacheClusterId', 'CacheNodeId'
  116. ],
  117. 'AWS/EBS': [
  118. 'VolumeId'
  119. ],
  120. 'AWS/EC2': [
  121. 'AutoScalingGroupName', 'ImageId', 'InstanceId', 'InstanceType'
  122. ],
  123. 'AWS/ELB': [
  124. 'LoadBalancerName', 'AvailabilityZone'
  125. ],
  126. 'AWS/ElasticMapReduce': [
  127. 'ClusterId', 'JobId'
  128. ],
  129. 'AWS/Kinesis': [
  130. 'StreamName'
  131. ],
  132. 'AWS/ML': [
  133. 'MLModelId', 'RequestMode'
  134. ],
  135. 'AWS/OpsWorks': [
  136. 'StackId', 'LayerId', 'InstanceId'
  137. ],
  138. 'AWS/Redshift': [
  139. 'NodeID', 'ClusterIdentifier'
  140. ],
  141. 'AWS/RDS': [
  142. 'DBInstanceIdentifier', 'DatabaseClass', 'EngineName'
  143. ],
  144. 'AWS/Route53': [
  145. 'HealthCheckId'
  146. ],
  147. 'AWS/SNS': [
  148. 'Application', 'Platform', 'TopicName'
  149. ],
  150. 'AWS/SQS': [
  151. 'QueueName'
  152. ],
  153. 'AWS/S3': [
  154. 'BucketName', 'StorageType'
  155. ],
  156. 'AWS/SWF': [
  157. 'Domain', 'ActivityTypeName', 'ActivityTypeVersion'
  158. ],
  159. 'AWS/StorageGateway': [
  160. 'GatewayId', 'GatewayName', 'VolumeId'
  161. ],
  162. 'AWS/WorkSpaces': [
  163. 'DirectoryId', 'WorkspaceId'
  164. ],
  165. };
  166. /* jshint +W101 */
  167. /* load custom metrics definitions */
  168. var self = this;
  169. $q.all(
  170. _.chain(datasource.jsonData.customMetricsAttributes)
  171. .reject(function(u) {
  172. return _.isEmpty(u);
  173. })
  174. .map(function(u) {
  175. return $http({ method: 'GET', url: u });
  176. })
  177. )
  178. .then(function(allResponse) {
  179. _.chain(allResponse)
  180. .map(function(d) {
  181. return d.data.Metrics;
  182. })
  183. .flatten()
  184. .reject(function(metric) {
  185. return metric.Namespace.indexOf('AWS/') === 0;
  186. })
  187. .map(function(metric) {
  188. metric.Dimensions = _.chain(metric.Dimensions)
  189. .map(function(d) {
  190. return d.Name;
  191. })
  192. .value().sort();
  193. return metric;
  194. })
  195. .uniq(function(metric) {
  196. return metric.Namespace + metric.MetricName + metric.Dimensions.join('');
  197. })
  198. .each(function(metric) {
  199. if (!_.has(self.supportedMetrics, metric.Namespace)) {
  200. self.supportedMetrics[metric.Namespace] = [];
  201. }
  202. self.supportedMetrics[metric.Namespace].push(metric.MetricName);
  203. if (!_.has(self.supportedDimensions, metric.Namespace)) {
  204. self.supportedDimensions[metric.Namespace] = [];
  205. }
  206. self.supportedDimensions[metric.Namespace] = _.union(self.supportedDimensions[metric.Namespace], metric.Dimensions);
  207. });
  208. });
  209. }
  210. // Called once per panel (graph)
  211. CloudWatchDatasource.prototype.query = function(options) {
  212. var start = convertToCloudWatchTime(options.range.from);
  213. var end = convertToCloudWatchTime(options.range.to);
  214. var queries = [];
  215. _.each(options.targets, _.bind(function(target) {
  216. if (!target.namespace || !target.metricName || _.isEmpty(target.statistics)) {
  217. return;
  218. }
  219. var query = {};
  220. query.region = templateSrv.replace(target.region, options.scopedVars);
  221. query.namespace = templateSrv.replace(target.namespace, options.scopedVars);
  222. query.metricName = templateSrv.replace(target.metricName, options.scopedVars);
  223. query.dimensions = _.map(_.keys(target.dimensions), function(key) {
  224. return {
  225. Name: templateSrv.replace(key, options.scopedVars),
  226. Value: templateSrv.replace(target.dimensions[key], options.scopedVars)
  227. };
  228. });
  229. query.statistics = getActivatedStatistics(target.statistics);
  230. query.period = parseInt(target.period, 10);
  231. var range = end - start;
  232. // CloudWatch limit datapoints up to 1440
  233. if (range / query.period >= 1440) {
  234. query.period = Math.floor(range / 1440 / 60) * 60;
  235. }
  236. queries.push(query);
  237. }, this));
  238. // No valid targets, return the empty result to save a round trip.
  239. if (_.isEmpty(queries)) {
  240. var d = $q.defer();
  241. d.resolve({ data: [] });
  242. return d.promise;
  243. }
  244. var allQueryPromise = _.map(queries, _.bind(function(query) {
  245. return this.performTimeSeriesQuery(query, start, end);
  246. }, this));
  247. return $q.all(allQueryPromise)
  248. .then(function(allResponse) {
  249. var result = [];
  250. _.each(allResponse, function(response, index) {
  251. var metrics = transformMetricData(response, options.targets[index]);
  252. _.each(metrics, function(m) {
  253. result.push(m);
  254. });
  255. });
  256. return { data: result };
  257. });
  258. };
  259. CloudWatchDatasource.prototype.performTimeSeriesQuery = function(query, start, end) {
  260. var cloudwatch = this.getCloudWatchClient(query.region);
  261. var params = {
  262. Namespace: query.namespace,
  263. MetricName: query.metricName,
  264. Dimensions: query.dimensions,
  265. Statistics: query.statistics,
  266. StartTime: start,
  267. EndTime: end,
  268. Period: query.period
  269. };
  270. var d = $q.defer();
  271. cloudwatch.getMetricStatistics(params, function(err, data) {
  272. if (err) {
  273. return d.reject(err);
  274. }
  275. return d.resolve(data);
  276. });
  277. return d.promise;
  278. };
  279. CloudWatchDatasource.prototype.performSuggestRegion = function() {
  280. return this.supportedRegion;
  281. };
  282. CloudWatchDatasource.prototype.performSuggestNamespace = function() {
  283. return _.keys(this.supportedMetrics);
  284. };
  285. CloudWatchDatasource.prototype.performSuggestMetrics = function(namespace) {
  286. namespace = templateSrv.replace(namespace);
  287. return this.supportedMetrics[namespace] || [];
  288. };
  289. CloudWatchDatasource.prototype.performSuggestDimensionKeys = function(namespace) {
  290. namespace = templateSrv.replace(namespace);
  291. return this.supportedDimensions[namespace] || [];
  292. };
  293. CloudWatchDatasource.prototype.performSuggestDimensionValues = function(region, namespace, metricName, dimensions) {
  294. region = templateSrv.replace(region);
  295. namespace = templateSrv.replace(namespace);
  296. metricName = templateSrv.replace(metricName);
  297. var cloudwatch = this.getCloudWatchClient(region);
  298. var params = {
  299. Namespace: namespace,
  300. MetricName: metricName
  301. };
  302. if (!_.isEmpty(dimensions)) {
  303. params.Dimensions = _.map(_.keys(dimensions), function(key) {
  304. return {
  305. Name: templateSrv.replace(key),
  306. Value: templateSrv.replace(dimensions[key])
  307. };
  308. });
  309. }
  310. var d = $q.defer();
  311. cloudwatch.listMetrics(params, function(err, data) {
  312. if (err) {
  313. return d.reject(err);
  314. }
  315. var suggestData = _.chain(data.Metrics)
  316. .map(function(metric) {
  317. return metric.Dimensions;
  318. })
  319. .reject(function(metric) {
  320. return _.isEmpty(metric);
  321. })
  322. .value();
  323. return d.resolve(suggestData);
  324. });
  325. return d.promise;
  326. };
  327. CloudWatchDatasource.prototype.getTemplateVariableNames = function() {
  328. var variables = [];
  329. templateSrv.fillVariableValuesForUrl(variables);
  330. return _.map(_.keys(variables), function(k) {
  331. return k.replace(/var-/, '$');
  332. });
  333. };
  334. CloudWatchDatasource.prototype.metricFindQuery = function(query) {
  335. var region;
  336. var namespace;
  337. var metricName;
  338. var transformSuggestData = function(suggestData) {
  339. return _.map(suggestData, function(v) {
  340. return { text: v };
  341. });
  342. };
  343. var d = $q.defer();
  344. var regionQuery = query.match(/^region\(\)/);
  345. if (regionQuery) {
  346. d.resolve(transformSuggestData(this.performSuggestRegion()));
  347. return d.promise;
  348. }
  349. var namespaceQuery = query.match(/^namespace\(\)/);
  350. if (namespaceQuery) {
  351. d.resolve(transformSuggestData(this.performSuggestNamespace()));
  352. return d.promise;
  353. }
  354. var metricNameQuery = query.match(/^metrics\(([^\)]+?)\)/);
  355. if (metricNameQuery) {
  356. namespace = templateSrv.replace(metricNameQuery[1]);
  357. d.resolve(transformSuggestData(this.performSuggestMetrics(namespace)));
  358. return d.promise;
  359. }
  360. var dimensionKeysQuery = query.match(/^dimension_keys\(([^\)]+?)\)/);
  361. if (dimensionKeysQuery) {
  362. namespace = templateSrv.replace(dimensionKeysQuery[1]);
  363. d.resolve(transformSuggestData(this.performSuggestDimensionKeys(namespace)));
  364. return d.promise;
  365. }
  366. var dimensionValuesQuery = query.match(/^dimension_values\(([^,]+?),\s?([^,]+?),\s?([^,]+?)(,\s?([^)]*))?\)/);
  367. if (dimensionValuesQuery) {
  368. region = templateSrv.replace(dimensionValuesQuery[1]);
  369. namespace = templateSrv.replace(dimensionValuesQuery[2]);
  370. metricName = templateSrv.replace(dimensionValuesQuery[3]);
  371. var dimensionPart = templateSrv.replace(dimensionValuesQuery[5]);
  372. var dimensions = {};
  373. if (!_.isEmpty(dimensionPart)) {
  374. _.each(dimensionPart.split(','), function(v) {
  375. var t = v.split('=');
  376. if (t.length !== 2) {
  377. throw new Error('Invalid query format');
  378. }
  379. dimensions[t[0]] = t[1];
  380. });
  381. }
  382. return this.performSuggestDimensionValues(region, namespace, metricName, dimensions)
  383. .then(function(suggestData) {
  384. return _.map(suggestData, function(dimensions) {
  385. var result = _.chain(dimensions)
  386. .sortBy(function(dimension) {
  387. return dimension.Name;
  388. })
  389. .map(function(dimension) {
  390. return dimension.Name + '=' + dimension.Value;
  391. })
  392. .value().join(',');
  393. return { text: result };
  394. });
  395. });
  396. }
  397. return $q.when([]);
  398. };
  399. CloudWatchDatasource.prototype.testDatasource = function() {
  400. /* use billing metrics for test */
  401. var region = 'us-east-1';
  402. var namespace = 'AWS/Billing';
  403. var metricName = 'EstimatedCharges';
  404. var dimensions = {};
  405. return this.performSuggestDimensionValues(region, namespace, metricName, dimensions).then(function () {
  406. return { status: 'success', message: 'Data source is working', title: 'Success' };
  407. });
  408. };
  409. CloudWatchDatasource.prototype.getCloudWatchClient = function(region) {
  410. if (!this.proxyMode) {
  411. return new AWS.CloudWatch({
  412. region: region,
  413. accessKeyId: this.credentials.accessKeyId,
  414. secretAccessKey: this.credentials.secretAccessKey
  415. });
  416. } else {
  417. var self = this;
  418. var generateRequestProxy = function(service, action) {
  419. return function(params, callback) {
  420. var data = {
  421. region: region,
  422. service: service,
  423. action: action,
  424. parameters: params
  425. };
  426. var options = {
  427. method: 'POST',
  428. url: self.proxyUrl,
  429. data: data
  430. };
  431. $http(options).then(function(response) {
  432. callback(null, response.data);
  433. }, function(err) {
  434. callback(err, []);
  435. });
  436. };
  437. };
  438. return {
  439. getMetricStatistics: generateRequestProxy('CloudWatch', 'GetMetricStatistics'),
  440. listMetrics: generateRequestProxy('CloudWatch', 'ListMetrics')
  441. };
  442. }
  443. };
  444. CloudWatchDatasource.prototype.getDefaultRegion = function() {
  445. return this.defaultRegion;
  446. };
  447. function transformMetricData(md, options) {
  448. var result = [];
  449. var dimensionPart = templateSrv.replace(JSON.stringify(options.dimensions));
  450. _.each(getActivatedStatistics(options.statistics), function(s) {
  451. var metricLabel = md.Label + '_' + s + dimensionPart;
  452. var dps = _.map(md.Datapoints, function(value) {
  453. return [value[s], new Date(value.Timestamp).getTime()];
  454. });
  455. dps = _.sortBy(dps, function(dp) { return dp[1]; });
  456. result.push({ target: metricLabel, datapoints: dps });
  457. });
  458. return result;
  459. }
  460. function getActivatedStatistics(statistics) {
  461. var activatedStatistics = [];
  462. _.each(statistics, function(v, k) {
  463. if (v) {
  464. activatedStatistics.push(k);
  465. }
  466. });
  467. return activatedStatistics;
  468. }
  469. function convertToCloudWatchTime(date) {
  470. return Math.round(kbn.parseDate(date).getTime() / 1000);
  471. }
  472. return CloudWatchDatasource;
  473. });
  474. });