Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Initial metrics implementation #7

Merged
merged 10 commits into from
Oct 13, 2020
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,11 @@

<PropertyGroup>
<TargetFramework>netcoreapp3.1</TargetFramework>
<PackageId>Amazon.LambdaPowertools.Metrics</PackageId>
<Version>0.0.9</Version>
<Authors>awslabs</Authors>
<Company>Amazon</Company>
<GeneratePackageOnBuild>true</GeneratePackageOnBuild>
</PropertyGroup>

</Project>
8 changes: 0 additions & 8 deletions libraries/src/Amazon.LambdaPowertools.Metrics/Class1.cs

This file was deleted.

24 changes: 24 additions & 0 deletions libraries/src/Amazon.LambdaPowertools.Metrics/Metrics.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
using System;
using System.Collections.Generic;
using Amazon.LambdaPowertools.Metrics.Model;

namespace Amazon.LambdaPowertools.Metrics
{
public class Metrics : MetricsManager
{
public Metrics(
string metricsNamespace = null,
string serviceName = null,
Dictionary<string, string> dimensions = null,
Dictionary<string, Metric> metrics = null,
Dictionary<string, dynamic> metadata = null)
: base(metricsNamespace,
serviceName,
dimensions,
metrics,
metadata)
{

}
}
}
230 changes: 230 additions & 0 deletions libraries/src/Amazon.LambdaPowertools.Metrics/MetricsManager.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,230 @@
using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;
using System.Text.Json;
using Amazon.LambdaPowertools.Metrics.Model;

namespace Amazon.LambdaPowertools.Metrics
{

public class MetricsManager : IDisposable
{
private string _namespace;
public string Namespace
{
get { return _namespace; }
set { _namespace = value; }
}

private string _service;
public string Service
{
get { return _service; }
set { _service = value; }
}

[MaxLength(9)]
private Dictionary<string, string> _dimensions;
public Dictionary<string, string> Dimensions
{
get { return _dimensions; }
set { _dimensions = value; }
}

[MaxLength(100)]
private Dictionary<string, Metric> _metrics;
public Dictionary<string, Metric> Metrics
{
get { return _metrics; }
set { _metrics = value; }
}
t1agob marked this conversation as resolved.
Show resolved Hide resolved

private Dictionary<string, dynamic> _metadata;
private bool disposedValue;

public Dictionary<string, dynamic> Metadata
{
get { return _metadata; }
set { _metadata = value; }
}

public MetricsManager(
string metricsNamespace = null,
string serviceName = null,
Dictionary<string, string> dimensions = null,
Dictionary<string, Metric> metrics = null,
Dictionary<string, dynamic> metadata = null)
{
_namespace = !string.IsNullOrEmpty(metricsNamespace) ? metricsNamespace : Environment.GetEnvironmentVariable("POWERTOOLS_METRICS_NAMESPACE");
_service = !string.IsNullOrEmpty(serviceName) ? serviceName : Environment.GetEnvironmentVariable("POWERTOOLS_SERVICE_NAME");
t1agob marked this conversation as resolved.
Show resolved Hide resolved
_dimensions = dimensions != null ? dimensions : new Dictionary<string, string>();
_metrics = metrics != null ? metrics : new Dictionary<string, Metric>();
_metadata = metadata != null ? metadata : new Dictionary<string, dynamic>();

if(!string.IsNullOrEmpty(_service)){
_dimensions.Add("service", _service);
}
}

public void SetNamespace(string metricsNamespace)
{
_namespace = metricsNamespace;
}

public void SetService(string metricsService)
{
_service = metricsService;
}

public void AddDimension(string key, string value)
{
if (_dimensions.ContainsKey(key))
{
Console.WriteLine($"Dimension '{key}' already exists. Skipping...");
return;
}

if(_dimensions.Count == 9)
{
Console.WriteLine($"Maximum number of dimensions (9) reached. Cannot add '{key}' to dimensions.");
return;
}

_dimensions.Add(key, value);
}

public virtual void AddMetric(Metric metric)
{
if (_metrics.ContainsKey(metric.Name))
{
Console.WriteLine($"Metric '{metric}' already exists. Skipping...");
return;
}

_metrics.Add(metric.Name, metric);

if (_metrics.Count == 100)
{
Flush();
_metrics.Clear();
t1agob marked this conversation as resolved.
Show resolved Hide resolved
}
}

public void AddMetadata(string key, dynamic value)
t1agob marked this conversation as resolved.
Show resolved Hide resolved
{
if (_metadata.ContainsKey(key))
{
Console.WriteLine($"Metadata '{key}' already exists. Skipping...");
return;
}

_metadata.Add(key, value);
}

//TODO: Turn this into a private method once decision has been made on the implementation
public void Flush()
{
var metricObj = new MetricsWrapper
{
Root = new MetricsDefinition
{
Timestamp = DateTimeOffset.Now.ToUnixTimeMilliseconds(),
CloudWatchMetrics = new List<CloudWatchMetrics>{
new CloudWatchMetrics{
Namespace = Namespace,
Dimensions = new List<List<string>>{
extractDimensions(Dimensions)
},
Metrics = extractMetricDefinitionSet(Metrics)
}
}
}
};

var json = JsonSerializer.Serialize(metricObj, typeof(MetricsWrapper));

string result = AddToJson(json, _dimensions, _metadata, _metrics);

Console.WriteLine(result);

_metrics.Clear();
}

private List<string> extractDimensions(Dictionary<string, string> dimensions)
t1agob marked this conversation as resolved.
Show resolved Hide resolved
{
List<string> result = new List<string>();

result.AddRange(dimensions.Keys);

return result;
}

private List<MetricsDefinitionSet> extractMetricDefinitionSet(Dictionary<string, Metric> metrics)
t1agob marked this conversation as resolved.
Show resolved Hide resolved
{
List<MetricsDefinitionSet> metricDefintionSet = new List<MetricsDefinitionSet>();

foreach (var item in metrics)
{
metricDefintionSet.Add(new MetricsDefinitionSet
{
Name = item.Value.Name,
Unit = item.Value.Unit
});
}

return metricDefintionSet;
}

private string AddToJson(string json, Dictionary<string, string> dimensions, Dictionary<string, dynamic> metadata, Dictionary<string, Metric> metrics)
{
string result = "";
foreach (var item in dimensions)
{
result += $",\"{item.Key}\":\"{item.Value}\"";
}

foreach (var item in metadata)
{
result += $",\"{item.Key}\":\"{item.Value}\"";
}

foreach (var item in metrics.Values)
{
result += $",\"{item.Name}\": {item.Value}";
}

return $"{json.Remove(json.Length - 1)}{result}}}";
}

protected virtual void Dispose(bool disposing)
{
if (!disposedValue)
{
if (disposing)
{
Flush();
}

_dimensions = null;
_metrics = null;
_metadata = null;
_namespace=null;
_service=null;
disposedValue = true;
}
}

~MetricsManager()
{
// Do not change this code. Put cleanup code in 'Dispose(bool disposing)' method
Dispose(disposing: false);
}

public void Dispose()
{
// Do not change this code. Put cleanup code in 'Dispose(bool disposing)' method
Dispose(disposing: true);
GC.SuppressFinalize(this);
}
}
}
29 changes: 29 additions & 0 deletions libraries/src/Amazon.LambdaPowertools.Metrics/Model/Metric.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
using Amazon.LambdaPowertools.Metrics.Model;

namespace Amazon.LambdaPowertools.Metrics
{
public class Metric
{
private string _name;
public string Name
{
get { return _name; }
set { _name = value; }
}

private string _unit;
public string Unit
{
get { return _unit; }
set { _unit = value; }
}

private double _value;
public double Value
{
get { return _value; }
set { _value = value; }
}

}
}
32 changes: 32 additions & 0 deletions libraries/src/Amazon.LambdaPowertools.Metrics/Model/MetricUnit.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
namespace Amazon.LambdaPowertools.Metrics.Model
{
public class MetricUnit
{
public static readonly string Seconds = "Seconds";
public static readonly string Microseconds = "Microseconds";
public static readonly string Milliseconds = "Milliseconds";
public static readonly string Bytes = "Bytes";
public static readonly string Kilobytes = "Kilobytes";
public static readonly string Megabytes = "Megabytes";
public static readonly string Gigabytes = "Gigabytes";
public static readonly string Terabytes = "Terabytes";
public static readonly string Bits = "Bits";
public static readonly string Kilobits = "Kilobits";
public static readonly string Megabits = "Megabits";
public static readonly string Gigabits = "Gigabits";
public static readonly string Terabits = "Terabits";
public static readonly string Percent = "Percent";
public static readonly string Count = "Count";
public static readonly string BytesPerSecond = "Bytes/Second";
public static readonly string KilobytesPerSecond = "Kilobytes/Second";
public static readonly string MegabytesPerSecond = "Megabytes/Second";
public static readonly string GigabytesPerSecond = "Gigabytes/Second";
public static readonly string TerabytesPerSecond = "Terabytes/Second";
public static readonly string BitsPerSecond = "Bits/Second";
public static readonly string KilobitsPerSecond = "Kilobits/Second";
public static readonly string MegabitsPerSecond = "Megabits/Second";
public static readonly string GigabitsPerSecond = "Gigabits/Second";
public static readonly string TerabitsPerSecond = "Terabits/Second";
public static readonly string CountPerSecond = "Count/Second";
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
using System.Collections.Generic;
using System.Text.Json.Serialization;
using Amazon.LambdaPowertools.Metrics.Model;

namespace Amazon.LambdaPowertools.Metrics
{

public class MetricsWrapper
{
[JsonPropertyName("_aws")]
public MetricsDefinition Root { get; set; }
}

public class MetricsDefinition
{
public double Timestamp { get; set; }
public List<CloudWatchMetrics> CloudWatchMetrics { get; set; }
}

public class CloudWatchMetrics
{
public string Namespace { get; set; }
public List<List<string>> Dimensions { get; set; }
public List<MetricsDefinitionSet> Metrics { get; set; }
}

public class MetricsDefinitionSet
{
public string Name { get; set; }
public string Unit { get; set; }
}
}
Loading