-
Notifications
You must be signed in to change notification settings - Fork 75
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
base: 2.0-.Net-Core
Are you sure you want to change the base?
Changes from all commits
77c0978
7661450
28cbcdd
944a6f0
5018170
01ed3ba
da4f07d
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,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; } | ||
} | ||
} |
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; } | ||
} | ||
} |
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; } | ||
} | ||
} |
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}"; | ||
} | ||
} | ||
} |
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; | ||
|
||
} | ||
} | ||
} |
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; | ||
} | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
using Newtonsoft.Json; | ||
|
||
namespace RedditSharp.Interfaces | ||
{ | ||
internal interface ISettableWebAgent | ||
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. What's the purpose behind having this as an interface here? 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. 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. var result = Helpers.PopulateObjects<Collection>(json, WebAgent);
foreach (var collection in result)
collection.WebAgent = WebAgent;
return result; Or pass in Or something else? 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. I'm thinking it may make more sense just to make the WebAgent settable by default instead of having a specific interface for it. 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. Why can't PopulateObjects just use the |
||
{ | ||
[JsonIgnore] | ||
IWebAgent WebAgent { set; } | ||
} | ||
} |
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."); | ||
} | ||
} | ||
} |
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.
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.
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.
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 is200
."errors" is an array of arrays... kinda weird, but the consumer can do whatever with it
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.
@CrustyJew It appears RedditHttpException has the status code built in. Perhaps eventually having an
Error
class that can be used across the board?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.
@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.
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.
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.