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

Add collection support #201

Open
wants to merge 7 commits into
base: 2.0-.Net-Core
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ TestResults/
*.mdf
*.psess
*.vsp
.idea/

/**/project.lock.json

Expand All @@ -18,4 +19,4 @@ StyleCop.Cache
/.vs
/packages
/RedditSharpTests/secrets.json
/RedditSharpTests/private.config
/RedditSharpTests/private.json
2 changes: 1 addition & 1 deletion CONTRIBUTING.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
# Development

Use a private.config file to pull in config for tests. The required format is as follows
Use a private.json file to pull in config for tests. The required format is as follows

```json
{
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
namespace RedditSharp.Data.Collections
{
internal class AddOrRemovePostsFromCollectionParams
{
/// <summary>
/// UUID of a collection
/// </summary>
[RedditAPIName("collection_id")]
internal string CollectionId { get; set; }

/// <summary>
/// Full name of link, e.g. t3_xyz
/// </summary>
[RedditAPIName("link_fullname")]
internal string LinkFullName { get; set; }
}
}
23 changes: 23 additions & 0 deletions RedditSharp/Data/Collections/CollectionCreationParams.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
namespace RedditSharp.Data.Collections
{
internal class CollectionCreationParams
{
/// <summary>
/// Title of the submission. Maximum 300 characters.
/// </summary>
[RedditAPIName("title")]
internal string Title { get; set; }

/// <summary>
/// Description of the collection. Maximum of 500 characters.
/// </summary>
[RedditAPIName("description")]
internal string Description { get; set; }

/// <summary>
/// Name of the subreddit to which you are submitting.
/// </summary>
[RedditAPIName("sr_fullname")]
internal string Subreddit { get; set; }
}
}
20 changes: 20 additions & 0 deletions RedditSharp/Data/Collections/GetCollectionParams.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
namespace RedditSharp.Data.Collections
{
internal class CollectionBaseParams
{
/// <summary>
/// UUID of a collection
/// </summary>
[RedditAPIName("collection_id")]
internal string CollectionId { get; set; }
}

internal class GetCollectionParams : CollectionBaseParams
{
/// <summary>
/// Should include all the links
/// </summary>
[RedditAPIName("include_links")]
internal bool IncludeLinks { get; set; }
}
}
15 changes: 15 additions & 0 deletions RedditSharp/Data/Urls.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
namespace RedditSharp.Data
{
internal static class Urls
{
internal static class Collections
{
internal const string AddPost = "/api/v1/collections/add_post_to_collection";
internal static string Get(string collectionId, bool includeLinks) => $"/api/v1/collections/collection.json?collection_id={collectionId}&include_links={includeLinks}";
internal const string CreateCollectionUrl = "/api/v1/collections/create_collection";
internal const string Delete = "/api/v1/collections/delete_collection";
internal const string RemovePost = "/api/v1/collections/remove_post_in_collection";
internal static string SubredditCollectionsUrl(string fullName) => $"/api/v1/collections/subreddit_collections.json?sr_fullname={fullName}";
}
}
}
14 changes: 14 additions & 0 deletions RedditSharp/Exceptions/RedditException.cs
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
using System;
using Newtonsoft.Json.Linq;

namespace RedditSharp
{
Expand All @@ -8,6 +9,8 @@ namespace RedditSharp
[Serializable]
public class RedditException : Exception
{
public JArray Errors { get; }

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does this actually return the errors from Reddit properly? That's been something I've been meaning to fix up. It also needs an HTTP status code preferably.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, it does, but only for the new endpoints I've implemented. But yeah, I agree the code in ExecuteRequestAsync needs to be cleaned up to expose the errors instead of swallowing an exception.

Example response when trying to delete a collection that doesn't exist: {"json": {"errors": [["INVALID_COLLECTION_ID", "That collection doesn't exist", "collection_id"]]}} but the status code is 200.

"errors" is an array of arrays... kinda weird, but the consumer can do whatever with it

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@CrustyJew It appears RedditHttpException has the status code built in. Perhaps eventually having an Error class that can be used across the board?

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@chasedog @jkrejcha I think it would be better to get these rolled together so you have one exception type and can determine if it is an HTTP status code issue or a reddit result issue.

Copy link
Contributor

@jkrejcha jkrejcha Aug 29, 2019

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The great thing about the reddit API is that it seems to be a combination of both, either one, none, or sometimes a completely different error code dependent on whether you're accessing via the site or via the JSON API. There does need to be handling of errors of some sort. RedditHttpException should probably inherit from RedditException IMO but it's hard to say.

/// <summary>
/// Initializes a new instance of the RedditException class.
/// </summary>
Expand All @@ -26,6 +29,17 @@ public RedditException(string message)

}

/// <summary>
/// Initializes a new instance of the RedditException class with a specified error message and a JArray of errors
/// </summary>
/// <param name="message">The message that describes the error.</param>
/// <param name="errors">List of errors.</param>
public RedditException(string message, JArray errors)
: base(message)
{
Errors = errors;
}

/// <summary>
/// Initializes a new instance of the RedditException class with a specified error message and
/// a referenced inner exception that is the cause of this exception.
Expand Down
33 changes: 33 additions & 0 deletions RedditSharp/Extensions/JTokenExtensions/JTokenExtensions.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
using Newtonsoft.Json.Linq;

namespace RedditSharp.Extensions.JTokenExtensions
{
public static class JTokenExtensions
{
public static void ThrowIfHasErrors(this JToken json, string message)
{
if (json["errors"].IsNonEmptyArray(out var errors))
{
throw new RedditException($"{message} {errors}", errors);
}
}

public static bool IsNonEmptyArray(this JToken json, out JArray array)
{
var isArray = _IsArray(json, out array);
return isArray && array.Count > 0;
}

private static bool _IsArray(JToken json, out JArray array)
{
if (json != null && json.Type == JTokenType.Array)
{
array = (JArray)json;
return true;
}
array = default;
return false;

}
}
}
30 changes: 30 additions & 0 deletions RedditSharp/Helpers/Helpers.PopulateObjects.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
using System;
using System.Collections.Generic;
using System.Linq;
using Newtonsoft.Json.Linq;
using RedditSharp.Interfaces;

namespace RedditSharp
{
public partial class Helpers
{
internal static List<T> PopulateObjects<T>(JToken json, IWebAgent webAgent)
where T : ISettableWebAgent, new()
{
if (json.Type != JTokenType.Array)
throw new ArgumentException("must be of type array", nameof(json));

var objects = new List<T>();

for (var i = 0; i < json.Count(); i++)
{
var item = new T();
PopulateObject(json[i], item);
item.WebAgent = webAgent;
objects.Add(item);
}

return objects;
}
}
}
10 changes: 10 additions & 0 deletions RedditSharp/Interfaces/ISettableWebAgent.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
using Newtonsoft.Json;

namespace RedditSharp.Interfaces
{
internal interface ISettableWebAgent
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What's the purpose behind having this as an interface here?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's for PopulateObjects to have a way to set the web agent for each object. So when you fetch the collections, you can do collection operations (e.g. AddPostAsync) directly to the object instead of passing in the web agent, for example.
Do you prefer the web agents to be set after populating the objects here:

var result = Helpers.PopulateObjects<Collection>(json, WebAgent);
foreach (var collection in result)
    collection.WebAgent = WebAgent;
return result;

Or pass in Action<T> (e.g. x => x.WebAgent = WebAgent) that gets called after the item is populated?

Or something else?

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm thinking it may make more sense just to make the WebAgent settable by default instead of having a specific interface for it.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why can't PopulateObjects just use the Thing's static Parse() method instead of creating a new interface?

{
[JsonIgnore]
IWebAgent WebAgent { set; }
}
}
21 changes: 21 additions & 0 deletions RedditSharp/Reddit.cs
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,9 @@
using System.Linq.Expressions;
using System.Security.Authentication;
using System.Threading.Tasks;
using RedditSharp.Data;
using RedditSharp.Data.Collections;
using RedditSharp.Extensions.JTokenExtensions;
using DefaultWebAgent = RedditSharp.WebAgent;

namespace RedditSharp
Expand Down Expand Up @@ -477,5 +480,23 @@ public Listing<T> GetListing<T>(string url, int maxLimit = -1, int limitPerReque
{
return new Listing<T>(this.WebAgent, url, maxLimit, limitPerRequest);
}

public async Task<Collection> GetCollectionAsync(string collectionId, bool includePostsContent = true)
{
var json = await WebAgent.Get(Urls.Collections.Get(collectionId, includePostsContent));
json.ThrowIfHasErrors("Could not retrieve the collection.");
return new Collection(json, WebAgent);
}

/// <summary>
/// Deletes the specified collection. Must be a mod of the subreddit to delete.
/// </summary>
/// <param name="collectionId"></param>
/// <returns></returns>
public async Task DeleteCollectionAsync(string collectionId)
{
var json = await WebAgent.Post(Urls.Collections.Delete, new CollectionBaseParams { CollectionId = collectionId });
json.ThrowIfHasErrors("Could not delete the collection.");
}
}
}
4 changes: 2 additions & 2 deletions RedditSharp/RedditSharp.csproj
Original file line number Diff line number Diff line change
Expand Up @@ -12,12 +12,12 @@
</PropertyGroup>

<ItemGroup>
<PackageReference Include="Microsoft.Extensions.Caching.Memory" Version="1.1.1" />
<PackageReference Include="Microsoft.Extensions.Caching.Memory" Version="1.1.2" />
<PackageReference Include="Newtonsoft.Json" Version="10.0.3" />
<PackageReference Include="System.Net.Http" Version="4.3.3" />
<PackageReference Include="System.Net.Security" Version="4.3.2" />
<PackageReference Include="System.Reflection.TypeExtensions" Version="4.4.0" />
<PackageReference Include="System.Runtime.Serialization.Formatters" Version="4.3.0" />
<PackageReference Include="System.Runtime.Serialization.Formatters" Version="4.3.0" />
<PackageReference Include="System.Threading.Thread" Version="4.3.0" />
<PackageReference Include="System.Reactive.Linq" Version="3.1.1" />
<PackageReference Include="System.Interactive.Async" Version="3.1.1" />
Expand Down
109 changes: 109 additions & 0 deletions RedditSharp/Things/Collection.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,109 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using Newtonsoft.Json;
using Newtonsoft.Json.Linq;
using RedditSharp.Data;
using RedditSharp.Data.Collections;
using RedditSharp.Extensions.JTokenExtensions;
using RedditSharp.Interfaces;

namespace RedditSharp.Things
{
public class Collection : ISettableWebAgent
{
[JsonProperty("subreddit_id")]
public string SubredditId { get; internal set; }

[JsonProperty("description")]
public string Description { get; internal set; }

[JsonProperty("author_name")]
public string AuthorName { get; internal set; }

[JsonProperty("collection_id")]
public string CollectionId { get; internal set; }

[JsonProperty("display_layout")]
public string DisplayLayout { get; internal set; }

[JsonProperty("permalink")]
public string Permalink { get; internal set; }

[JsonProperty("link_ids")]
public string[] LinkIds { get; internal set; }

[JsonProperty("title")]
public string Title { get; internal set; }

[JsonProperty("created_at_utc"), JsonConverter(typeof(UnixTimestampConverter))]
public DateTime CreatedAtUtc { get; internal set; }

[JsonProperty("author_id")]
public string AuthorId { get; internal set; }

[JsonProperty("last_update_utc"), JsonConverter(typeof(UnixTimestampConverter))]
public DateTime LastUpdateUtc { get; internal set; }

public Post[] Posts { get; }

public IWebAgent WebAgent { private get; set; }

public Collection()
{
}

public Collection(JToken json, IWebAgent agent)
{
WebAgent = agent;

Helpers.PopulateObject(json, this);

var posts = new List<Post>();
var children = json.SelectToken("sorted_links.data.children");
if (children != null && children.Type == JTokenType.Array)
{
posts.AddRange(children.Select(item => new Post(WebAgent, item)));
}

Posts = posts.ToArray();
}

/// <summary>
/// Adds a post to the collection
/// </summary>
/// <param name="linkFullName">Full name of link, e.g. t3_xyz</param>
public async Task AddPostAsync(string linkFullName)
{
var data = new AddOrRemovePostsFromCollectionParams
{
CollectionId = CollectionId,
LinkFullName = linkFullName,
};
var json = await WebAgent.Post(Urls.Collections.AddPost, data);
json.ThrowIfHasErrors("Could not add post to collection.");
}

/// <summary>
/// Removes a post from the collection
/// </summary>
/// <param name="linkFullName">Full name of link, e.g. t3_xyz</param>
public async Task RemovePostAsync(string linkFullName)
{
var data = new AddOrRemovePostsFromCollectionParams
{
CollectionId = CollectionId,
LinkFullName = linkFullName,
};
var json = await WebAgent.Post(Urls.Collections.RemovePost, data);
json.ThrowIfHasErrors("Could not remove post from collection.");
}

public async Task DeleteAsync()
{
var json = await WebAgent.Post(Urls.Collections.Delete, null);
json.ThrowIfHasErrors("Could not remove collection.");
}
}
}
Loading