-
Notifications
You must be signed in to change notification settings - Fork 23
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
Create Access Key #129
base: master
Are you sure you want to change the base?
Create Access Key #129
Changes from all commits
90b7fc1
32a8394
ac00b1d
3596593
1497185
a7a4b81
43ea621
0e2cd7a
08d776d
a781c51
d2db292
7abce95
b60d1b1
0e254c6
ffaa48b
a7e9a21
b9fbe6f
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,35 @@ | ||
using Keen.Core.AccessKey; | ||
using System; | ||
using System.Collections.Generic; | ||
using System.Linq; | ||
using System.Text; | ||
using System.Threading.Tasks; | ||
using Newtonsoft.Json.Linq; | ||
using Keen.Core; | ||
|
||
namespace Keen.Net.Test | ||
{ | ||
/// <summary> | ||
/// AccessKeyMock provides an implementation of IAccessKeys with a constructor that | ||
/// accepts delegates for each of the interface methods. | ||
/// The purpose of this is to allow test methods to set up a customized | ||
/// IAccessKeys for each test. | ||
/// </summary> | ||
class AccessKeysMock : IAccessKeys | ||
{ | ||
private readonly IProjectSettings _settings; | ||
private readonly Func<AccessKey, IProjectSettings, JObject> _createAccessKey; | ||
|
||
public AccessKeysMock(IProjectSettings projSettings, | ||
Func<AccessKey, IProjectSettings, JObject> createAccessKey = null) | ||
{ | ||
_settings = projSettings; | ||
_createAccessKey = createAccessKey ?? ((p, k) => new JObject()); | ||
} | ||
|
||
public Task<JObject> CreateAccessKey(AccessKey accesskey) | ||
{ | ||
return Task.Run(() => _createAccessKey(accesskey, _settings)); | ||
} | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,91 @@ | ||
using Keen.Core; | ||
using Keen.Core.AccessKey; | ||
using Keen.Core.Query; | ||
using Newtonsoft.Json.Linq; | ||
using NUnit.Framework; | ||
using System; | ||
using System.Collections.Generic; | ||
using System.Linq; | ||
using System.Text; | ||
using System.Threading.Tasks; | ||
|
||
namespace Keen.Net.Test | ||
{ | ||
[TestFixture] | ||
class AccessKeyTests : TestBase | ||
{ | ||
[Test] | ||
public void CreateAccessKey_Success() | ||
{ | ||
var settings = new ProjectSettingsProvider(projectId: "X", masterKey: SettingsEnv.MasterKey); // Replace X with respective value | ||
var client = new KeenClient(settings); | ||
|
||
if (UseMocks) | ||
client.AccessKeys = new AccessKeysMock(settings, | ||
createAccessKey: new Func<AccessKey, IProjectSettings, JObject>((e, p) => | ||
{ | ||
Assert.True(p == settings, "Incorrect Settings"); | ||
Assert.NotNull(e.Name, "Expected a name for the newly created Key"); | ||
Assert.NotNull(e.Permitted, "Expected a list of high level actions this key can perform"); | ||
Assert.NotNull(e.Options, "Expected an object containing more details about the key’s permitted and restricted functionality"); | ||
if ((p == settings) && (e.Name == "TestAccessKey") && (e.IsActive) && e.Permitted.First() == "queries" && e.Options.CachedQueries.Allowed.First() == "my_cached_query") | ||
return new JObject(); | ||
else | ||
throw new Exception("Unexpected value"); | ||
})); | ||
|
||
HashSet<string> permissions = new HashSet<string>() { "queries" }; | ||
List<QueryFilter> qFilters = new List<QueryFilter>() { new QueryFilter("customer.id", QueryFilter.FilterOperator.Equals(), "asdf12345z") }; | ||
CachedQueries cachedQueries = new CachedQueries(); | ||
cachedQueries.Allowed = new HashSet<string>() { "my_cached_query" }; | ||
Options options = new Options() | ||
{ | ||
Queries = new Core.AccessKey.Queries { Filters = qFilters }, | ||
CachedQueries = cachedQueries | ||
}; | ||
|
||
Assert.DoesNotThrow(() => client.CreateAccessKey(new AccessKey { Name = "TestAccessKey", IsActive = true, Options = options, Permitted = permissions })); | ||
} | ||
|
||
|
||
[Test] | ||
public void CreateAccessKey_With_All_Properties_Given_As_Null_Success() | ||
{ | ||
var settings = new ProjectSettingsProvider(projectId: "X", masterKey: SettingsEnv.MasterKey); // Replace X with respective value | ||
var client = new KeenClient(settings); | ||
|
||
if (UseMocks) | ||
client.AccessKeys = new AccessKeysMock(settings, | ||
createAccessKey: new Func<AccessKey, IProjectSettings, JObject>((e, p) => | ||
{ | ||
Assert.True(p == settings, "Incorrect Settings"); | ||
Assert.NotNull(e.Name, "Expected a name for the newly created Key"); | ||
Assert.NotNull(e.Permitted, "Expected a list of high level actions this key can perform"); | ||
Assert.NotNull(e.Options, "Expected an object containing more details about the key’s permitted and restricted functionality"); | ||
if ((p == settings) && (e.Name == "TestAccessKey") && (e.IsActive) && e.Permitted.First() == "queries" && e.Options.CachedQueries.Allowed.First() == "my_cached_query") | ||
return new JObject(); | ||
else | ||
throw new Exception("Unexpected value"); | ||
})); | ||
|
||
HashSet<string> permissions = new HashSet<string>() { "queries" }; | ||
List<QueryFilter> qFilters = new List<QueryFilter>() { new QueryFilter("customer.id", QueryFilter.FilterOperator.Equals(), "asdf12345z") }; | ||
CachedQueries cachedQueries = new CachedQueries() { Allowed = null, Blocked = null }; | ||
SavedQueries savedQuaries = new SavedQueries() { Allowed = null, Blocked = null, Filters = null }; | ||
Datasets datasets = new Datasets() { Allowed = null, Blocked = null, Operations = null }; | ||
Writes writes = new Writes() { Autofill = null }; | ||
cachedQueries.Allowed = new HashSet<string>() { "my_cached_query" }; | ||
Options options = new Options() | ||
{ | ||
Queries = new Core.AccessKey.Queries { Filters = qFilters }, | ||
CachedQueries = cachedQueries, | ||
SavedQueries = savedQuaries, | ||
Datasets = datasets, | ||
Writes = writes | ||
}; | ||
|
||
Assert.DoesNotThrow(() => client.CreateAccessKey(new AccessKey { Name = "TestAccessKey", IsActive = true, Options = options, Permitted = permissions })); | ||
} | ||
|
||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,83 @@ | ||
using Keen.Core.Query; | ||
using System; | ||
using System.Collections.Generic; | ||
|
||
namespace Keen.Core.AccessKey | ||
{ | ||
/// <summary> | ||
/// Model for AccessKey object | ||
/// </summary> | ||
public class AccessKey | ||
{ | ||
public string Name { get; set; } | ||
public bool IsActive { get; set; } | ||
public ISet<string> Permitted { get; set; } | ||
public string Key { get; set; } | ||
public Options Options { get; set; } | ||
} | ||
|
||
/// <summary> | ||
/// When SavedQueries are permitted, the Access Key will have access to run saved queries. | ||
/// </summary> | ||
public class SavedQueries | ||
{ | ||
public ISet<string> Blocked { get; set; } | ||
public IEnumerable<QueryFilter> Filters { get; set; } | ||
public ISet<string> Allowed { get; set; } | ||
} | ||
|
||
/// <summary> | ||
/// When Queries are permitted, the Access Key will have the ability to do ad-hoc queries. | ||
/// </summary> | ||
public class Queries | ||
{ | ||
public IEnumerable<QueryFilter> Filters { get; set; } | ||
} | ||
|
||
/// <summary> | ||
/// When Writes are permitted, the Access Key will have the ability to stream data to Keen. | ||
/// </summary> | ||
public class Writes | ||
{ | ||
public dynamic Autofill { get; set; } // "customer": { "id": "93iskds39kd93id", "name": "Ada Corp." } | ||
} | ||
|
||
/// <summary> | ||
/// When Datasets are permitted, the Access Key will have access to getting a dataset definition, retrieving cached dataset results, and listing cached datasets definitions for a project. | ||
/// </summary> | ||
public class Datasets | ||
{ | ||
public IEnumerable<string> Operations { get; set; } | ||
public IDictionary<string, AllowedDatasetIndexes> Allowed { get; set; } | ||
public ISet<string> Blocked { get; set; } | ||
} | ||
|
||
/// <summary> | ||
/// Optionals limiting of allowed datasets in the access key by index | ||
/// </summary> | ||
public class AllowedDatasetIndexes | ||
{ | ||
public Tuple<string, string> IndexBy { get; set;} | ||
} | ||
|
||
/// <summary> | ||
/// When CachedQueries are permitted, the Access Key will have access to retrieve results from cached queries. | ||
/// </summary> | ||
public class CachedQueries | ||
{ | ||
public ISet<string> Blocked { get; set; } | ||
public ISet<string> Allowed { get; set; } | ||
} | ||
|
||
/// <summary> | ||
/// An object containing more details about the key’s permitted and restricted functionality. | ||
/// </summary> | ||
public class Options | ||
{ | ||
public SavedQueries SavedQueries { get; set; } | ||
public Writes Writes { get; set; } | ||
public Datasets Datasets { get; set; } | ||
public CachedQueries CachedQueries { get; set; } | ||
public Queries Queries { get; set; } | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,100 @@ | ||
using Newtonsoft.Json; | ||
using Newtonsoft.Json.Linq; | ||
using Newtonsoft.Json.Serialization; | ||
using System; | ||
using System.Threading.Tasks; | ||
|
||
namespace Keen.Core.AccessKey | ||
{ | ||
/// <summary> | ||
/// AccessKeys implements the IAccessKeys interface which represents the Keen.IO Access Key API methods. | ||
/// </summary> | ||
public class AccessKeys : IAccessKeys | ||
{ | ||
private readonly IKeenHttpClient _keenHttpClient; | ||
private readonly string _accesKeyRelativeUrl; | ||
private readonly string _readKey; | ||
private readonly string _masterKey; | ||
|
||
internal AccessKeys(IProjectSettings prjSettings, | ||
IKeenHttpClientProvider keenHttpClientProvider) | ||
{ | ||
if (null == prjSettings) | ||
{ | ||
throw new ArgumentNullException(nameof(prjSettings), | ||
"Project Settings must be provided."); | ||
} | ||
|
||
if (null == keenHttpClientProvider) | ||
{ | ||
throw new ArgumentNullException(nameof(keenHttpClientProvider), | ||
"A KeenHttpClient provider must be provided."); | ||
} | ||
|
||
if (string.IsNullOrWhiteSpace(prjSettings.KeenUrl) || | ||
!Uri.IsWellFormedUriString(prjSettings.KeenUrl, UriKind.Absolute)) | ||
{ | ||
throw new KeenException( | ||
"A properly formatted KeenUrl must be provided via Project Settings."); | ||
} | ||
|
||
var serverBaseUrl = new Uri(prjSettings.KeenUrl); | ||
_keenHttpClient = keenHttpClientProvider.GetForUrl(serverBaseUrl); | ||
_accesKeyRelativeUrl = KeenHttpClient.GetRelativeUrl(prjSettings.ProjectId, | ||
KeenConstants.AccessKeyResource); | ||
|
||
_readKey = prjSettings.ReadKey; | ||
_masterKey = prjSettings.MasterKey; | ||
} | ||
|
||
public async Task<JObject> CreateAccessKey(AccessKey accesskey) | ||
{ | ||
if (string.IsNullOrWhiteSpace(_masterKey)) | ||
{ | ||
throw new KeenException("An API WriteKey is required to add events."); | ||
} | ||
|
||
DefaultContractResolver contractResolver = new DefaultContractResolver | ||
{ | ||
NamingStrategy = new SnakeCaseNamingStrategy() | ||
}; | ||
|
||
var content = JsonConvert.SerializeObject(accesskey, new JsonSerializerSettings | ||
{ | ||
ContractResolver = contractResolver, | ||
Formatting = Formatting.Indented | ||
}).ToSafeString(); | ||
|
||
var responseMsg = await _keenHttpClient | ||
.PostAsync(_accesKeyRelativeUrl, _masterKey, content) | ||
.ConfigureAwait(continueOnCapturedContext: false); | ||
|
||
var responseString = await responseMsg | ||
.Content | ||
.ReadAsStringAsync() | ||
.ConfigureAwait(continueOnCapturedContext: false); | ||
|
||
JObject jsonResponse = null; | ||
|
||
try | ||
{ | ||
jsonResponse = JObject.Parse(responseString); | ||
} | ||
catch (Exception) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Was this because you were seeing a 404 error or something similar and it's actually OK in this case to swallow the exception? Then will we throw below? I know we do this elsewhere, and we should really keep the inner exception around for context, but that's another issue. At least if that's the case, let's put a comment here explaining the empty |
||
{ | ||
// To avoid any flow stoppers | ||
} | ||
if (!responseMsg.IsSuccessStatusCode) | ||
{ | ||
throw new KeenException("AddEvents failed with status: " + responseMsg.StatusCode); | ||
} | ||
|
||
if (null == jsonResponse) | ||
{ | ||
throw new KeenException("AddEvents failed with empty response from server."); | ||
} | ||
|
||
return jsonResponse; | ||
} | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
using Newtonsoft.Json.Linq; | ||
using System.Threading.Tasks; | ||
|
||
namespace Keen.Core.AccessKey | ||
{ | ||
/// <summary> | ||
/// Public interface for Access Key related functionalities | ||
/// </summary> | ||
public interface IAccessKeys | ||
{ | ||
/// <summary> | ||
/// Creates an Access Key | ||
/// </summary> | ||
/// <param name="accesskey"></param> | ||
/// <returns></returns> | ||
Task<JObject> CreateAccessKey(AccessKey accesskey); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We prefer Task-returning methods to use the |
||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Don't forget to organize and remove unnecessary
using
s.