diff --git a/Octokit.Reactive/Clients/IObservableUserAdministrationClient.cs b/Octokit.Reactive/Clients/IObservableUserAdministrationClient.cs
index d893f91d8c..844a0e1598 100644
--- a/Octokit.Reactive/Clients/IObservableUserAdministrationClient.cs
+++ b/Octokit.Reactive/Clients/IObservableUserAdministrationClient.cs
@@ -5,7 +5,7 @@
namespace Octokit.Reactive
{
///
- /// A client for GitHub's User Administration API.
+ /// A client for GitHub's User Administration API (GitHub Enterprise)
///
///
/// See the Administration API documentation for more details.
@@ -13,43 +13,126 @@ namespace Octokit.Reactive
public interface IObservableUserAdministrationClient
{
///
- /// Promotes ordinary user to a site administrator.
+ /// Create a new user (must be Site Admin user).
///
///
- /// https://developer.github.com/v3/users/administration/#promote-an-ordinary-user-to-a-site-administrator
+ /// See the API documentation
+ /// for more information.
+ ///
+ /// The object describing the user to create
+ /// The created object
+ IObservable Create(NewUser newUser);
+
+ ///
+ /// Rename an existing user (must be Site Admin user).
+ ///
+ ///
+ /// See the API documentation
+ /// for more information.
+ /// Note that this queues a request to rename a user, rather than execute it straight away
+ ///
+ /// The username to rename
+ /// The request, specifying the new login
+ /// A object indicating the queued task message and Url to the user
+ IObservable Rename(string login, UserRename userRename);
+
+ ///
+ /// Create an impersonation OAuth token (must be Site Admin user).
+ ///
+ ///
+ /// See the API documentation
+ /// for more information.
+ ///
+ /// The user to impersonate
+ /// The request specifying the required scopes
+ /// An object containing the impersonation token
+ IObservable CreateImpersonationToken(string login, NewImpersonationToken newImpersonationToken);
+
+ ///
+ /// Deletes an impersonation OAuth token (must be Site Admin user).
+ ///
+ ///
+ /// See the API documentation
+ /// for more information.
+ ///
+ /// The user to remove impersonation token from
+ ///
+ IObservable DeleteImpersonationToken(string login);
+
+ ///
+ /// Promotes ordinary user to a site administrator (must be Site Admin user).
+ ///
+ ///
+ /// See the API documentation
+ /// for more information.
///
/// The user to promote to administrator.
///
IObservable Promote(string login);
///
- /// Demotes a site administrator to an ordinary user.
+ /// Demotes a site administrator to an ordinary user (must be Site Admin user).
///
///
- /// https://developer.github.com/v3/users/administration/#demote-a-site-administrator-to-an-ordinary-user
+ /// See the API documentation
+ /// for more information.
///
/// The user to demote from administrator.
///
IObservable Demote(string login);
///
- /// Suspends a user.
+ /// Suspends a user (must be Site Admin user).
///
///
- /// https://developer.github.com/v3/users/administration/#suspend-a-user
+ /// See the API documentation
+ /// for more information.
///
/// The user to suspend.
///
IObservable Suspend(string login);
///
- /// Unsuspends a user.
+ /// Unsuspends a user (must be Site Admin user).
///
///
- /// https://developer.github.com/v3/users/administration/#unsuspend-a-user
+ /// See the API documentation
+ /// for more information.
///
/// The user to unsuspend.
///
IObservable Unsuspend(string login);
+
+ ///
+ /// List all public keys (must be Site Admin user).
+ ///
+ ///
+ /// See the API documentation
+ /// for more information.
+ ///
+ ///
+ IObservable ListAllPublicKeys();
+
+ ///
+ /// Delete a user (must be Site Admin user).
+ ///
+ ///
+ /// See the API documentation
+ /// for more information.
+ ///
+ /// The user to delete
+ ///
+ IObservable Delete(string login);
+
+ ///
+ /// Delete a public key (must be Site Admin user).
+ ///
+ ///
+ /// See the API documentation
+ /// for more information.
+ ///
+ /// The key to delete
+ ///
+ IObservable DeletePublicKey(int keyId);
}
}
diff --git a/Octokit.Reactive/Clients/ObservableUserAdministrationClient.cs b/Octokit.Reactive/Clients/ObservableUserAdministrationClient.cs
index d76d7e9dad..bc0bf672e7 100644
--- a/Octokit.Reactive/Clients/ObservableUserAdministrationClient.cs
+++ b/Octokit.Reactive/Clients/ObservableUserAdministrationClient.cs
@@ -7,9 +7,16 @@
namespace Octokit.Reactive
{
+ ///
+ /// A client for GitHub's User Administration API (GitHub Enterprise)
+ ///
+ ///
+ /// See the Administration API documentation for more details.
+ ///
public class ObservableUserAdministrationClient : IObservableUserAdministrationClient
{
readonly IUserAdministrationClient _client;
+ readonly IConnection _connection;
///
/// Initializes a new instance of the class.
@@ -20,13 +27,74 @@ public ObservableUserAdministrationClient(IGitHubClient client)
Ensure.ArgumentNotNull(client, "client");
_client = client.User.Administration;
+ _connection = client.Connection;
}
///
- /// Promotes ordinary user to a site administrator.
+ /// Create a new user (must be Site Admin user).
///
///
- /// https://developer.github.com/v3/users/administration/#promote-an-ordinary-user-to-a-site-administrator
+ /// See the API documentation
+ /// for more information.
+ ///
+ /// The object describing the user to create
+ /// The created object
+ public IObservable Create(NewUser newUser)
+ {
+ return _client.Create(newUser).ToObservable();
+ }
+
+ ///
+ /// Rename an existing user (must be Site Admin user).
+ ///
+ ///
+ /// See the API documentation
+ /// for more information.
+ /// Note that this queues a request to rename a user, rather than execute it straight away
+ ///
+ /// The username to rename
+ /// The request, specifying the new login
+ /// A object indicating the queued task message and Url to the user
+ public IObservable Rename(string login, UserRename userRename)
+ {
+ return _client.Rename(login, userRename).ToObservable();
+ }
+
+ ///
+ /// Create an impersonation OAuth token (must be Site Admin user).
+ ///
+ ///
+ /// See the API documentation
+ /// for more information.
+ ///
+ /// The user to impersonate
+ /// The request specifying the required scopes
+ /// An object containing the impersonation token
+ public IObservable CreateImpersonationToken(string login, NewImpersonationToken newImpersonationToken)
+ {
+ return _client.CreateImpersonationToken(login, newImpersonationToken).ToObservable();
+ }
+
+ ///
+ /// Deletes an impersonation OAuth token (must be Site Admin user).
+ ///
+ ///
+ /// See the API documentation
+ /// for more information.
+ ///
+ /// The user to remove impersonation token from
+ ///
+ public IObservable DeleteImpersonationToken(string login)
+ {
+ return _client.DeleteImpersonationToken(login).ToObservable();
+ }
+
+ ///
+ /// Promotes ordinary user to a site administrator (must be Site Admin user).
+ ///
+ ///
+ /// See the API documentation
+ /// for more information.
///
/// The user to promote to administrator.
///
@@ -36,10 +104,11 @@ public IObservable Promote(string login)
}
///
- /// Demotes a site administrator to an ordinary user.
+ /// Demotes a site administrator to an ordinary user (must be Site Admin user).
///
///
- /// https://developer.github.com/v3/users/administration/#demote-a-site-administrator-to-an-ordinary-user
+ /// See the API documentation
+ /// for more information.
///
/// The user to demote from administrator.
///
@@ -49,10 +118,11 @@ public IObservable Demote(string login)
}
///
- /// Suspends a user.
+ /// Suspends a user (must be Site Admin user).
///
///
- /// https://developer.github.com/v3/users/administration/#suspend-a-user
+ /// See the API documentation
+ /// for more information.
///
/// The user to suspend.
///
@@ -62,10 +132,11 @@ public IObservable Suspend(string login)
}
///
- /// Unsuspends a user.
+ /// Unsuspends a user (must be Site Admin user).
///
///
- /// https://developer.github.com/v3/users/administration/#unsuspend-a-user
+ /// See the API documentation
+ /// for more information.
///
/// The user to unsuspend.
///
@@ -73,5 +144,46 @@ public IObservable Unsuspend(string login)
{
return _client.Unsuspend(login).ToObservable();
}
+
+ ///
+ /// List all public keys (must be Site Admin user).
+ ///
+ ///
+ /// See the API documentation
+ /// for more information.
+ ///
+ ///
+ public IObservable ListAllPublicKeys()
+ {
+ return _connection.GetAndFlattenAllPages(ApiUrls.UserAdministrationPublicKeys());
+ }
+
+ ///
+ /// Delete a user (must be Site Admin user).
+ ///
+ ///
+ /// See the API documentation
+ /// for more information.
+ ///
+ /// The user to delete
+ ///
+ public IObservable Delete(string login)
+ {
+ return _client.Delete(login).ToObservable();
+ }
+
+ ///
+ /// Delete a public key (must be Site Admin user).
+ ///
+ ///
+ /// See the API documentation
+ /// for more information.
+ ///
+ /// The key to delete
+ ///
+ public IObservable DeletePublicKey(int keyId)
+ {
+ return _client.DeletePublicKey(keyId).ToObservable();
+ }
}
}
diff --git a/Octokit.Tests.Integration/Helpers/ObservableGithubClientExtensions.cs b/Octokit.Tests.Integration/Helpers/ObservableGithubClientExtensions.cs
index 8a5919ade6..030795b0d0 100644
--- a/Octokit.Tests.Integration/Helpers/ObservableGithubClientExtensions.cs
+++ b/Octokit.Tests.Integration/Helpers/ObservableGithubClientExtensions.cs
@@ -27,5 +27,19 @@ internal async static Task CreateRepositoryContext(this IObse
return new RepositoryContext(repo);
}
+
+ internal async static Task CreateEnterpriseTeamContext(this IObservableGitHubClient client, string organization, NewTeam newTeam)
+ {
+ var team = await client.Organization.Team.Create(organization, newTeam);
+
+ return new EnterpriseTeamContext(team);
+ }
+
+ internal async static Task CreateEnterpriseUserContext(this IObservableGitHubClient client, NewUser newUser)
+ {
+ var user = await client.User.Administration.Create(newUser);
+
+ return new EnterpriseUserContext(user);
+ }
}
}
\ No newline at end of file
diff --git a/Octokit.Tests.Integration/Octokit.Tests.Integration.csproj b/Octokit.Tests.Integration/Octokit.Tests.Integration.csproj
index 9e3f07e736..3900738aca 100644
--- a/Octokit.Tests.Integration/Octokit.Tests.Integration.csproj
+++ b/Octokit.Tests.Integration/Octokit.Tests.Integration.csproj
@@ -145,6 +145,7 @@
+
diff --git a/Octokit.Tests.Integration/Reactive/Enterprise/ObservableEnterpriseLdapClientTests.cs b/Octokit.Tests.Integration/Reactive/Enterprise/ObservableEnterpriseLdapClientTests.cs
index 0880616259..dde932d8c9 100644
--- a/Octokit.Tests.Integration/Reactive/Enterprise/ObservableEnterpriseLdapClientTests.cs
+++ b/Octokit.Tests.Integration/Reactive/Enterprise/ObservableEnterpriseLdapClientTests.cs
@@ -20,11 +20,10 @@ public class ObservableEnterpriseLdapClientTests : IDisposable
public ObservableEnterpriseLdapClientTests()
{
- var gitHub = EnterpriseHelper.GetAuthenticatedClient();
- _github = new ObservableGitHubClient(gitHub);
+ _github = new ObservableGitHubClient(EnterpriseHelper.GetAuthenticatedClient());
NewTeam newTeam = new NewTeam(Helper.MakeNameWithTimestamp("test-team")) { Description = "Test Team" };
- _context = gitHub.CreateEnterpriseTeamContext(EnterpriseHelper.Organization, newTeam).Result;
+ _context = _github.CreateEnterpriseTeamContext(EnterpriseHelper.Organization, newTeam).Result;
}
[GitHubEnterpriseTest]
diff --git a/Octokit.Tests.Integration/Reactive/ObservableUserAdministrationClientTests.cs b/Octokit.Tests.Integration/Reactive/ObservableUserAdministrationClientTests.cs
new file mode 100644
index 0000000000..efaea9a0bd
--- /dev/null
+++ b/Octokit.Tests.Integration/Reactive/ObservableUserAdministrationClientTests.cs
@@ -0,0 +1,201 @@
+using System;
+using System.Linq;
+using System.Reactive.Linq;
+using System.Threading.Tasks;
+using Octokit.Reactive;
+using Octokit.Tests.Integration.Helpers;
+using Xunit;
+
+namespace Octokit.Tests.Integration.Clients
+{
+ public class ObservableUserAdministrationClientTests
+ {
+ readonly IObservableGitHubClient _github;
+
+ public ObservableUserAdministrationClientTests()
+ {
+ _github = new ObservableGitHubClient(EnterpriseHelper.GetAuthenticatedClient());
+ }
+
+ private NewUser GenerateNewUserDetails()
+ {
+ string username = Helper.MakeNameWithTimestamp("user");
+ string email = string.Concat(username, "@company.com");
+ return new NewUser(username, email);
+ }
+
+ [GitHubEnterpriseTest]
+ public async Task CanCreateAndDelete()
+ {
+ User checkUser = null;
+
+ // Create a new user
+ var newUser = GenerateNewUserDetails();
+
+ var observable = _github.User.Administration.Create(newUser);
+ var user = await observable;
+
+ // Check returned object (cant check email as it isn't public)
+ Assert.NotNull(user);
+ Assert.Equal(user.Login, newUser.Login);
+
+ // Get user to check they exist
+ checkUser = await _github.User.Get(newUser.Login);
+ Assert.Equal(checkUser.Login, newUser.Login);
+
+ // Delete the user
+ await _github.User.Administration.Delete(newUser.Login);
+
+ // Ensure user doesnt exist
+ try
+ {
+ checkUser = await _github.User.Get(newUser.Login);
+ if (checkUser != null)
+ {
+ throw new Exception("User still exists!");
+ }
+ }
+ catch (ApiException e)
+ {
+ if (e.StatusCode != System.Net.HttpStatusCode.NotFound)
+ {
+ throw;
+ }
+ }
+ }
+
+ [GitHubEnterpriseTest]
+ public async Task CanRename()
+ {
+ string renamedUsername = Helper.MakeNameWithTimestamp("user-renamed");
+ // Create a disposable user for the test
+ using (var context = _github.CreateEnterpriseUserContext(GenerateNewUserDetails()).Result)
+ {
+ var observable = _github.User.Administration.Rename(
+ context.UserLogin,
+ new UserRename(renamedUsername));
+ var response = await observable;
+
+ Assert.NotNull(response);
+ Assert.StartsWith("Job queued to rename user", response.Message);
+ Assert.EndsWith(context.UserId.ToString(), response.Url);
+ }
+
+ // Remove user if it was already renamed
+ EnterpriseHelper.DeleteUser(renamedUsername);
+ }
+
+ [GitHubEnterpriseTest]
+ public async Task CanAddAndDeleteImpersonationToken()
+ {
+ // Create a disposable user for the test
+ using (var context = _github.CreateEnterpriseUserContext(GenerateNewUserDetails()).Result)
+ {
+ // Create Impersonation token
+ var observable = _github.User.Administration.CreateImpersonationToken(
+ context.UserLogin,
+ new NewImpersonationToken(new string[] { "public_repo" }));
+ var token = await observable;
+
+ Assert.NotNull(token);
+ Assert.True(
+ token.Scopes.Count() == 1 &&
+ token.Scopes.All(s => s == "public_repo"));
+
+ // Delete Impersonation token
+ await _github.User.Administration.DeleteImpersonationToken(context.UserLogin);
+ }
+ }
+
+ [GitHubEnterpriseTest]
+ public async Task CanPromoteAndDemote()
+ {
+ User checkUser = null;
+
+ // Create a disposable user for the test
+ using (var context = _github.CreateEnterpriseUserContext(GenerateNewUserDetails()).Result)
+ {
+ // Ensure user is not site admin
+ checkUser = await _github.User.Get(context.UserLogin);
+ Assert.False(checkUser.SiteAdmin);
+
+ // Promote to site admin
+ await _github.User.Administration.Promote(context.UserLogin);
+
+ // Ensure user is site admin
+ checkUser = await _github.User.Get(context.UserLogin);
+ Assert.True(checkUser.SiteAdmin);
+
+ // Demote user
+ await _github.User.Administration.Demote(context.UserLogin);
+
+ // Ensure user is not site admin
+ checkUser = await _github.User.Get(context.UserLogin);
+ Assert.False(checkUser.SiteAdmin);
+ }
+ }
+
+ [GitHubEnterpriseTest]
+ public async Task CanSuspendAndUnsuspend()
+ {
+ User checkUser = null;
+
+ // Create a disposable user for the test
+ using (var context = _github.CreateEnterpriseUserContext(GenerateNewUserDetails()).Result)
+ {
+ // Ensure user is not suspended
+ checkUser = await _github.User.Get(context.UserLogin);
+ Assert.False(checkUser.Suspended);
+
+ // Suspend user
+ await _github.User.Administration.Suspend(context.UserLogin);
+
+ // Ensure user is suspended
+ checkUser = await _github.User.Get(context.UserLogin);
+ Assert.True(checkUser.Suspended);
+
+ // Unsuspend user
+ await _github.User.Administration.Unsuspend(context.UserLogin);
+
+ // Ensure user is not suspended
+ checkUser = await _github.User.Get(context.UserLogin);
+ Assert.False(checkUser.Suspended);
+ }
+ }
+
+ [GitHubEnterpriseTest(Skip = "Currently no way to add keys, so cant test listing keys")]
+ public async Task CanListAllPublicKeys()
+ {
+ // Create a disposable user for the test
+ using (var context = _github.CreateEnterpriseUserContext(GenerateNewUserDetails()).Result)
+ {
+ // Ensure user has a key
+ //var key = await _github.User.Keys.Create(new NewPublicKey("title", "key"));
+
+ // Get public keys
+ var observable = _github.User.Administration.ListAllPublicKeys();
+ var keys = await (observable.ToList());
+
+ Assert.NotNull(keys);
+ Assert.True(keys.Count > 0);
+
+ // Delete key
+ //await _github.User.Administration.DeletePublicKey(key.Id);
+ }
+ }
+
+ [GitHubEnterpriseTest(Skip = "Currently no way to add keys, so cant test deleting keys")]
+ public async Task CanDeletePublicKey()
+ {
+ // Create a disposable user for the test
+ using (var context = _github.CreateEnterpriseUserContext(GenerateNewUserDetails()).Result)
+ {
+ // Ensure user has a key
+ //var key = await _github.User.Keys.Create(new NewPublicKey("title", "key"));
+
+ // Delete key
+ //await _github.User.Administration.DeletePublicKey(key.Id);
+ }
+ }
+ }
+}
diff --git a/Octokit.Tests/Reactive/ObservableUserAdministrationClientTests.cs b/Octokit.Tests/Reactive/ObservableUserAdministrationClientTests.cs
index d605e71843..d247ddb30f 100644
--- a/Octokit.Tests/Reactive/ObservableUserAdministrationClientTests.cs
+++ b/Octokit.Tests/Reactive/ObservableUserAdministrationClientTests.cs
@@ -1,4 +1,5 @@
using System;
+using System.Linq;
using NSubstitute;
using Octokit.Reactive;
using Xunit;
@@ -7,10 +8,75 @@ namespace Octokit.Tests.Reactive
{
public class ObservableUserAdministrationClientTests
{
+ public class TheCreateMethod
+ {
+ [Fact]
+ public void CallsIntoClient()
+ {
+ var gitHubClient = Substitute.For();
+ var client = new ObservableUserAdministrationClient(gitHubClient);
+
+ client.Create(new NewUser("auser", "email@company.com"));
+
+ gitHubClient.User.Administration.Received().Create(
+ Arg.Is(a =>
+ a.Login == "auser" &&
+ a.Email == "email@company.com"));
+ }
+ }
+
+ public class TheRenameMethod
+ {
+ [Fact]
+ public void CallsIntoClient()
+ {
+ var gitHubClient = Substitute.For();
+ var client = new ObservableUserAdministrationClient(gitHubClient);
+
+ client.Rename("auser", new UserRename("renamed-user"));
+
+ gitHubClient.User.Administration.Received().Rename(
+ "auser",
+ Arg.Is(a =>
+ a.Login == "renamed-user"));
+ }
+ }
+
+ public class TheCreateImpersonationTokenMethod
+ {
+ [Fact]
+ public void CallsIntoClient()
+ {
+ var gitHubClient = Substitute.For();
+ var client = new ObservableUserAdministrationClient(gitHubClient);
+
+ client.CreateImpersonationToken("auser", new NewImpersonationToken(new string[] { "public_repo" }));
+
+ gitHubClient.User.Administration.Received().CreateImpersonationToken(
+ "auser",
+ Arg.Is(a =>
+ a.Scopes.ToList()[0] == "public_repo"));
+ }
+ }
+
+ public class TheDeleteImpersonationTokenMethod
+ {
+ [Fact]
+ public void CallsIntoClient()
+ {
+ var gitHubClient = Substitute.For();
+ var client = new ObservableUserAdministrationClient(gitHubClient);
+
+ client.DeleteImpersonationToken("auser");
+
+ gitHubClient.User.Administration.Received().DeleteImpersonationToken("auser");
+ }
+ }
+
public class ThePromoteMethod
{
[Fact]
- public void GetsFromClientPromtePromote()
+ public void CallsIntoClient()
{
var gitHubClient = Substitute.For();
var client = new ObservableUserAdministrationClient(gitHubClient);
@@ -24,7 +90,7 @@ public void GetsFromClientPromtePromote()
public class TheDemoteMethod
{
[Fact]
- public void GetsFromClientDemoteDemote()
+ public void CallsIntoClient()
{
var gitHubClient = Substitute.For();
var client = new ObservableUserAdministrationClient(gitHubClient);
@@ -38,7 +104,7 @@ public void GetsFromClientDemoteDemote()
public class TheSuspendMethod
{
[Fact]
- public void GetsFromClientSuspendSuspend()
+ public void CallsIntoClient()
{
var gitHubClient = Substitute.For();
var client = new ObservableUserAdministrationClient(gitHubClient);
@@ -52,7 +118,7 @@ public void GetsFromClientSuspendSuspend()
public class TheUnsuspendMethod
{
[Fact]
- public void GetsFromClientUnsuspendUnsuspend()
+ public void CallsIntoClient()
{
var gitHubClient = Substitute.For();
var client = new ObservableUserAdministrationClient(gitHubClient);
@@ -63,6 +129,54 @@ public void GetsFromClientUnsuspendUnsuspend()
}
}
+ public class TheListAllPublicKeysMethod
+ {
+ [Fact]
+ public void RequestsTheCorrectUrl()
+ {
+ var gitHubClient = Substitute.For();
+ var client = new ObservableUserAdministrationClient(gitHubClient);
+
+ var expectedUri = "admin/keys";
+
+ client.ListAllPublicKeys();
+
+ gitHubClient.Connection.Received().Get>(
+ Arg.Is(a =>
+ a.ToString() == expectedUri),
+ null,
+ null);
+ }
+ }
+
+ public class TheDeleteMethod
+ {
+ [Fact]
+ public void CallsIntoClient()
+ {
+ var gitHubClient = Substitute.For();
+ var client = new ObservableUserAdministrationClient(gitHubClient);
+
+ client.Delete("auser");
+
+ gitHubClient.User.Administration.Received().Delete("auser");
+ }
+ }
+
+ public class TheDeletePublicKeyMethod
+ {
+ [Fact]
+ public void CallsIntoClient()
+ {
+ var gitHubClient = Substitute.For();
+ var client = new ObservableUserAdministrationClient(gitHubClient);
+
+ client.DeletePublicKey(1);
+
+ gitHubClient.User.Administration.Received().DeletePublicKey(1);
+ }
+ }
+
public class TheCtor
{
[Fact]