diff --git a/Octokit.Tests.Integration/Clients/RepositoryContentsClientTests.cs b/Octokit.Tests.Integration/Clients/RepositoryContentsClientTests.cs index 2cb44e7ff8..91c8fd915f 100644 --- a/Octokit.Tests.Integration/Clients/RepositoryContentsClientTests.cs +++ b/Octokit.Tests.Integration/Clients/RepositoryContentsClientTests.cs @@ -107,7 +107,54 @@ await fixture.DeleteFile( new DeleteFileRequest("Deleted file", fileSha)); await Assert.ThrowsAsync( - async () => await fixture.GetAllContents(repository.Owner.Login, repository.Name, "somefile.txt")); + () => fixture.GetAllContents(repository.Owner.Login, repository.Name, "somefile.txt")); + } + } + + [IntegrationTest] + public async Task CrudTestWithNamedBranch() + { + var client = Helper.GetAuthenticatedClient(); + var fixture = client.Repository.Content; + var repoName = Helper.MakeNameWithTimestamp("source-repo"); + var branchName = "other-branch"; + + using (var context = await client.CreateRepositoryContext(new NewRepository(repoName) { AutoInit = true })) + { + var repository = context.Repository; + + var master = await client.Git.Reference.Get(Helper.UserName, repository.Name, "heads/master"); + await client.Git.Reference.Create(Helper.UserName, repository.Name, new NewReference("refs/heads/" + branchName, master.Object.Sha)); + var file = await fixture.CreateFile( + repository.Owner.Login, + repository.Name, + "somefile.txt", + new CreateFileRequest("Test commit", "Some Content", branchName)); + Assert.Equal("somefile.txt", file.Content.Name); + + var contents = await fixture.GetAllContentsByRef(repository.Owner.Login, repository.Name, "somefile.txt", branchName); + string fileSha = contents.First().Sha; + Assert.Equal("Some Content", contents.First().Content); + + var update = await fixture.UpdateFile( + repository.Owner.Login, + repository.Name, + "somefile.txt", + new UpdateFileRequest("Updating file", "New Content", fileSha, branchName)); + Assert.Equal("somefile.txt", update.Content.Name); + + contents = await fixture.GetAllContentsByRef(repository.Owner.Login, repository.Name, "somefile.txt", branchName); + Assert.Equal("New Content", contents.First().Content); + fileSha = contents.First().Sha; + + await fixture.DeleteFile( + repository.Owner.Login, + repository.Name, + "somefile.txt", + new DeleteFileRequest("Deleted file", fileSha, branchName)); + + await Assert.ThrowsAsync( + () => fixture.GetAllContents(repository.Owner.Login, repository.Name, "somefile.txt")); } } diff --git a/Octokit.Tests/Clients/RepositoryContentsClientTests.cs b/Octokit.Tests/Clients/RepositoryContentsClientTests.cs index 46324cbe19..85707ee9f7 100644 --- a/Octokit.Tests/Clients/RepositoryContentsClientTests.cs +++ b/Octokit.Tests/Clients/RepositoryContentsClientTests.cs @@ -89,5 +89,135 @@ public async Task ReturnsContents() Assert.Equal(1, contents.Count); } } + + public class TheCreateFileMethod + { + [Fact] + public void RequestsCorrectUrl() + { + var connection = Substitute.For(); + var client = new RepositoryContentsClient(connection); + + string expectedUri = "repos/org/repo/contents/path/to/file"; + client.CreateFile("org", "repo", "path/to/file", new CreateFileRequest("message", "myfilecontents", "mybranch")); + + connection.Received().Put(Arg.Is(u => u.ToString() == expectedUri), Arg.Any()); + } + + [Fact] + public void PassesRequestObject() + { + var connection = Substitute.For(); + var client = new RepositoryContentsClient(connection); + + client.CreateFile("org", "repo", "path/to/file", new CreateFileRequest("message", "myfilecontents", "mybranch")); + + connection.Received().Put( + Arg.Any(), + Arg.Is(a => + a.Message == "message" + && a.Content == "myfilecontents" + && a.Branch == "mybranch")); + } + + [Fact] + public async Task EnsuresNonNullArguments() + { + var connection = Substitute.For(); + var client = new RepositoryContentsClient(connection); + + await Assert.ThrowsAsync(() => client.CreateFile(null, "repo", "path/to/file", new CreateFileRequest("message", "myfilecontents", "mybranch"))); + await Assert.ThrowsAsync(() => client.CreateFile("org", null, "path/to/file", new CreateFileRequest("message", "myfilecontents", "mybranch"))); + await Assert.ThrowsAsync(() => client.CreateFile("org", "repo", null, new CreateFileRequest("message", "myfilecontents", "mybranch"))); + await Assert.ThrowsAsync(() => client.CreateFile("org", "repo", "path/to/file", null)); + } + } + + public class TheDeleteFileMethod + { + [Fact] + public void RequestsCorrectUrl() + { + var connection = Substitute.For(); + var client = new RepositoryContentsClient(connection); + + string expectedUri = "repos/org/repo/contents/path/to/file"; + client.DeleteFile("org", "repo", "path/to/file", new DeleteFileRequest("message", "1234abc", "mybranch")); + + connection.Received().Delete(Arg.Is(u => u.ToString() == expectedUri), Arg.Any()); + } + + [Fact] + public void PassesRequestObject() + { + var connection = Substitute.For(); + var client = new RepositoryContentsClient(connection); + + client.DeleteFile("org", "repo", "path/to/file", new DeleteFileRequest("message", "1234abc", "mybranch")); + + connection.Received().Delete( + Arg.Any(), + Arg.Is(a => + a.Message == "message" + && a.Sha == "1234abc" + && a.Branch == "mybranch")); + } + + [Fact] + public async Task EnsuresNonNullArguments() + { + var connection = Substitute.For(); + var client = new RepositoryContentsClient(connection); + + await Assert.ThrowsAsync(() => client.DeleteFile(null, "repo", "path/to/file", new DeleteFileRequest("message", "1234abc", "mybranch"))); + await Assert.ThrowsAsync(() => client.DeleteFile("org", null, "path/to/file", new DeleteFileRequest("message", "1234abc", "mybranch"))); + await Assert.ThrowsAsync(() => client.DeleteFile("org", "repo", null, new DeleteFileRequest("message", "1234abc", "mybranch"))); + await Assert.ThrowsAsync(() => client.DeleteFile("org", "repo", "path/to/file", null)); + } + } + + public class TheUpdateFileMethod + { + [Fact] + public void RequestsCorrectUrl() + { + var connection = Substitute.For(); + var client = new RepositoryContentsClient(connection); + + string expectedUri = "repos/org/repo/contents/path/to/file"; + client.UpdateFile("org", "repo", "path/to/file", new UpdateFileRequest("message", "myfilecontents", "1234abc", "mybranch")); + + connection.Received().Put(Arg.Is(u => u.ToString() == expectedUri), Arg.Any()); + } + + [Fact] + public void PassesRequestObject() + { + var connection = Substitute.For(); + var client = new RepositoryContentsClient(connection); + + client.UpdateFile("org", "repo", "path/to/file", new UpdateFileRequest("message", "myfilecontents", "1234abc", "mybranch")); + + connection.Received().Put( + Arg.Any(), + Arg.Is(a => + a.Message == "message" + && a.Content == "myfilecontents" + && a.Sha == "1234abc" + && a.Branch == "mybranch")); + } + + [Fact] + public async Task EnsuresNonNullArguments() + { + var connection = Substitute.For(); + var client = new RepositoryContentsClient(connection); + + await Assert.ThrowsAsync(() => client.UpdateFile(null, "repo", "path/to/file", new UpdateFileRequest("message", "myfilecontents", "1234abc", "mybranch"))); + await Assert.ThrowsAsync(() => client.UpdateFile("org", null, "path/to/file", new UpdateFileRequest("message", "myfilecontents", "1234abc", "mybranch"))); + await Assert.ThrowsAsync(() => client.UpdateFile("org", "repo", null, new UpdateFileRequest("message", "myfilecontents", "1234abc", "mybranch"))); + await Assert.ThrowsAsync(() => client.UpdateFile("org", "repo", "path/to/file", null)); + } + } } } \ No newline at end of file diff --git a/Octokit/Models/Request/CreateFileRequest.cs b/Octokit/Models/Request/CreateFileRequest.cs index 275d6fbf30..b9529e3bff 100644 --- a/Octokit/Models/Request/CreateFileRequest.cs +++ b/Octokit/Models/Request/CreateFileRequest.cs @@ -21,6 +21,18 @@ protected ContentRequest(string message) Message = message; } + /// + /// Initializes a new instance of the class. + /// + /// The message. + /// The branch the request is for. + protected ContentRequest(string message, string branch): this(message) + { + Ensure.ArgumentNotNullOrEmptyString(branch, "branch"); + + Branch = branch; + } + /// /// The commit message. This is required. /// @@ -54,6 +66,19 @@ public class DeleteFileRequest : ContentRequest /// The message. /// The sha. public DeleteFileRequest(string message, string sha) : base(message) + { + Ensure.ArgumentNotNullOrEmptyString(sha, "content"); + + Sha = sha; + } + + /// + /// Initializes a new instance of the class. + /// + /// The message. + /// The sha. + /// The branch the request is for. + public DeleteFileRequest(string message, string sha, string branch) : base(message, branch) { Ensure.ArgumentNotNullOrEmptyString(sha, "sha"); @@ -81,8 +106,8 @@ public class CreateFileRequest : ContentRequest /// /// Creates an instance of a . /// - /// - /// + /// The message. + /// The content. public CreateFileRequest(string message, string content) : base(message) { Ensure.ArgumentNotNull(content, "content"); @@ -90,6 +115,18 @@ public CreateFileRequest(string message, string content) : base(message) Content = content; } + /// + /// Initializes a new instance of the class. + /// + /// The message. + /// The content. + /// The branch the request is for. + public CreateFileRequest(string message, string content, string branch) : base(message, branch) + { + Ensure.ArgumentNotNullOrEmptyString(content, "content"); + + Content = content; + } /// /// The contents of the file to create. This is required. /// @@ -111,6 +148,12 @@ internal virtual string DebuggerDisplay [DebuggerDisplay("{DebuggerDisplay,nq}")] public class UpdateFileRequest : CreateFileRequest { + /// + /// Creates an instance of a . + /// + /// The message. + /// The content. + /// The sha. public UpdateFileRequest(string message, string content, string sha) : base(message, content) { @@ -119,6 +162,21 @@ public UpdateFileRequest(string message, string content, string sha) Sha = sha; } + /// + /// Creates an instance of a . + /// + /// The message. + /// The content. + /// The sha. + /// The branch the request is for. + public UpdateFileRequest(string message, string content, string sha, string branch) + : base(message, content, branch) + { + Ensure.ArgumentNotNullOrEmptyString(sha, "sha"); + + Sha = sha; + } + /// /// The blob SHA of the file being replaced. ///