Paging params in API docs

Closes #6003
This commit is contained in:
Mark McDowall 2023-09-07 21:59:00 -07:00
parent b4ef873cc3
commit 4cf7c89282
8 changed files with 279 additions and 37 deletions

View File

@ -158,6 +158,8 @@ namespace NzbDrone.Host
{ {
{ apikeyQuery, Array.Empty<string>() } { apikeyQuery, Array.Empty<string>() }
}); });
c.DescribeAllParametersInCamelCase();
}); });
services services

View File

@ -23,9 +23,9 @@ namespace Sonarr.Api.V3.Blocklist
[HttpGet] [HttpGet]
[Produces("application/json")] [Produces("application/json")]
public PagingResource<BlocklistResource> GetBlocklist() public PagingResource<BlocklistResource> GetBlocklist([FromQuery] PagingRequestResource paging)
{ {
var pagingResource = Request.ReadPagingResourceFromRequest<BlocklistResource>(); var pagingResource = new PagingResource<BlocklistResource>(paging);
var pagingSpec = pagingResource.MapToPagingSpec<BlocklistResource, NzbDrone.Core.Blocklisting.Blocklist>("date", SortDirection.Descending); var pagingSpec = pagingResource.MapToPagingSpec<BlocklistResource, NzbDrone.Core.Blocklisting.Blocklist>("date", SortDirection.Descending);
return pagingSpec.ApplyToPage(_blocklistService.Paged, model => BlocklistResourceMapper.MapToResource(model, _formatCalculator)); return pagingSpec.ApplyToPage(_blocklistService.Paged, model => BlocklistResourceMapper.MapToResource(model, _formatCalculator));

View File

@ -61,9 +61,9 @@ namespace Sonarr.Api.V3.History
[HttpGet] [HttpGet]
[Produces("application/json")] [Produces("application/json")]
public PagingResource<HistoryResource> GetHistory(bool includeSeries, bool includeEpisode) public PagingResource<HistoryResource> GetHistory([FromQuery] PagingRequestResource paging, bool includeSeries, bool includeEpisode)
{ {
var pagingResource = Request.ReadPagingResourceFromRequest<HistoryResource>(); var pagingResource = new PagingResource<HistoryResource>(paging);
var pagingSpec = pagingResource.MapToPagingSpec<HistoryResource, EpisodeHistory>("date", SortDirection.Descending); var pagingSpec = pagingResource.MapToPagingSpec<HistoryResource, EpisodeHistory>("date", SortDirection.Descending);
var eventTypeFilter = pagingResource.Filters.FirstOrDefault(f => f.Key == "eventType"); var eventTypeFilter = pagingResource.Filters.FirstOrDefault(f => f.Key == "eventType");

View File

@ -18,9 +18,9 @@ namespace Sonarr.Api.V3.Logs
[HttpGet] [HttpGet]
[Produces("application/json")] [Produces("application/json")]
public PagingResource<LogResource> GetLogs() public PagingResource<LogResource> GetLogs([FromQuery] PagingRequestResource paging)
{ {
var pagingResource = Request.ReadPagingResourceFromRequest<LogResource>(); var pagingResource = new PagingResource<LogResource>(paging);
var pageSpec = pagingResource.MapToPagingSpec<LogResource, Log>(); var pageSpec = pagingResource.MapToPagingSpec<LogResource, Log>();
if (pageSpec.SortKey == "time") if (pageSpec.SortKey == "time")

View File

@ -29,9 +29,9 @@ namespace Sonarr.Api.V3.Wanted
[HttpGet] [HttpGet]
[Produces("application/json")] [Produces("application/json")]
public PagingResource<EpisodeResource> GetCutoffUnmetEpisodes(bool includeSeries = false, bool includeEpisodeFile = false, bool includeImages = false) public PagingResource<EpisodeResource> GetCutoffUnmetEpisodes([FromQuery] PagingRequestResource paging, bool includeSeries = false, bool includeEpisodeFile = false, bool includeImages = false)
{ {
var pagingResource = Request.ReadPagingResourceFromRequest<EpisodeResource>(); var pagingResource = new PagingResource<EpisodeResource>(paging);
var pagingSpec = new PagingSpec<Episode> var pagingSpec = new PagingSpec<Episode>
{ {
Page = pagingResource.Page, Page = pagingResource.Page,

View File

@ -25,9 +25,9 @@ namespace Sonarr.Api.V3.Wanted
[HttpGet] [HttpGet]
[Produces("application/json")] [Produces("application/json")]
public PagingResource<EpisodeResource> GetMissingEpisodes(bool includeSeries = false, bool includeImages = false) public PagingResource<EpisodeResource> GetMissingEpisodes([FromQuery] PagingRequestResource paging, bool includeSeries = false, bool includeImages = false)
{ {
var pagingResource = Request.ReadPagingResourceFromRequest<EpisodeResource>(); var pagingResource = new PagingResource<EpisodeResource>(paging);
var pagingSpec = new PagingSpec<Episode> var pagingSpec = new PagingSpec<Episode>
{ {
Page = pagingResource.Page, Page = pagingResource.Page,

View File

@ -59,25 +59,25 @@
"schema": { "schema": {
"type": "object", "type": "object",
"properties": { "properties": {
"Username": { "username": {
"type": "string" "type": "string"
}, },
"Password": { "password": {
"type": "string" "type": "string"
}, },
"RememberMe": { "rememberMe": {
"type": "string" "type": "string"
} }
} }
}, },
"encoding": { "encoding": {
"Username": { "username": {
"style": "form" "style": "form"
}, },
"Password": { "password": {
"style": "form" "style": "form"
}, },
"RememberMe": { "rememberMe": {
"style": "form" "style": "form"
} }
} }
@ -381,6 +381,50 @@
"tags": [ "tags": [
"Blocklist" "Blocklist"
], ],
"parameters": [
{
"name": "page",
"in": "query",
"schema": {
"type": "integer",
"format": "int32",
"default": 1
}
},
{
"name": "pageSize",
"in": "query",
"schema": {
"type": "integer",
"format": "int32",
"default": 10
}
},
{
"name": "sortKey",
"in": "query",
"schema": {
"type": "string"
}
},
{
"name": "sortDirection",
"in": "query",
"schema": {
"$ref": "#/components/schemas/SortDirection"
}
},
{
"name": "filters",
"in": "query",
"schema": {
"type": "array",
"items": {
"$ref": "#/components/schemas/PagingResourceFilter"
}
}
}
],
"responses": { "responses": {
"200": { "200": {
"description": "Success", "description": "Success",
@ -1050,6 +1094,48 @@
"Cutoff" "Cutoff"
], ],
"parameters": [ "parameters": [
{
"name": "page",
"in": "query",
"schema": {
"type": "integer",
"format": "int32",
"default": 1
}
},
{
"name": "pageSize",
"in": "query",
"schema": {
"type": "integer",
"format": "int32",
"default": 10
}
},
{
"name": "sortKey",
"in": "query",
"schema": {
"type": "string"
}
},
{
"name": "sortDirection",
"in": "query",
"schema": {
"$ref": "#/components/schemas/SortDirection"
}
},
{
"name": "filters",
"in": "query",
"schema": {
"type": "array",
"items": {
"$ref": "#/components/schemas/PagingResourceFilter"
}
}
},
{ {
"name": "includeSeries", "name": "includeSeries",
"in": "query", "in": "query",
@ -2220,6 +2306,48 @@
"History" "History"
], ],
"parameters": [ "parameters": [
{
"name": "page",
"in": "query",
"schema": {
"type": "integer",
"format": "int32",
"default": 1
}
},
{
"name": "pageSize",
"in": "query",
"schema": {
"type": "integer",
"format": "int32",
"default": 10
}
},
{
"name": "sortKey",
"in": "query",
"schema": {
"type": "string"
}
},
{
"name": "sortDirection",
"in": "query",
"schema": {
"$ref": "#/components/schemas/SortDirection"
}
},
{
"name": "filters",
"in": "query",
"schema": {
"type": "array",
"items": {
"$ref": "#/components/schemas/PagingResourceFilter"
}
}
},
{ {
"name": "includeSeries", "name": "includeSeries",
"in": "query", "in": "query",
@ -3627,6 +3755,50 @@
"tags": [ "tags": [
"Log" "Log"
], ],
"parameters": [
{
"name": "page",
"in": "query",
"schema": {
"type": "integer",
"format": "int32",
"default": 1
}
},
{
"name": "pageSize",
"in": "query",
"schema": {
"type": "integer",
"format": "int32",
"default": 10
}
},
{
"name": "sortKey",
"in": "query",
"schema": {
"type": "string"
}
},
{
"name": "sortDirection",
"in": "query",
"schema": {
"$ref": "#/components/schemas/SortDirection"
}
},
{
"name": "filters",
"in": "query",
"schema": {
"type": "array",
"items": {
"$ref": "#/components/schemas/PagingResourceFilter"
}
}
}
],
"responses": { "responses": {
"200": { "200": {
"description": "Success", "description": "Success",
@ -4142,6 +4314,48 @@
"Missing" "Missing"
], ],
"parameters": [ "parameters": [
{
"name": "page",
"in": "query",
"schema": {
"type": "integer",
"format": "int32",
"default": 1
}
},
{
"name": "pageSize",
"in": "query",
"schema": {
"type": "integer",
"format": "int32",
"default": 10
}
},
{
"name": "sortKey",
"in": "query",
"schema": {
"type": "string"
}
},
{
"name": "sortDirection",
"in": "query",
"schema": {
"$ref": "#/components/schemas/SortDirection"
}
},
{
"name": "filters",
"in": "query",
"schema": {
"type": "array",
"items": {
"$ref": "#/components/schemas/PagingResourceFilter"
}
}
},
{ {
"name": "includeSeries", "name": "includeSeries",
"in": "query", "in": "query",
@ -4325,21 +4539,21 @@
], ],
"parameters": [ "parameters": [
{ {
"name": "RenameEpisodes", "name": "renameEpisodes",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "boolean" "type": "boolean"
} }
}, },
{ {
"name": "ReplaceIllegalCharacters", "name": "replaceIllegalCharacters",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "boolean" "type": "boolean"
} }
}, },
{ {
"name": "ColonReplacementFormat", "name": "colonReplacementFormat",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "integer", "type": "integer",
@ -4347,7 +4561,7 @@
} }
}, },
{ {
"name": "MultiEpisodeStyle", "name": "multiEpisodeStyle",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "integer", "type": "integer",
@ -4355,91 +4569,91 @@
} }
}, },
{ {
"name": "StandardEpisodeFormat", "name": "standardEpisodeFormat",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "string" "type": "string"
} }
}, },
{ {
"name": "DailyEpisodeFormat", "name": "dailyEpisodeFormat",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "string" "type": "string"
} }
}, },
{ {
"name": "AnimeEpisodeFormat", "name": "animeEpisodeFormat",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "string" "type": "string"
} }
}, },
{ {
"name": "SeriesFolderFormat", "name": "seriesFolderFormat",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "string" "type": "string"
} }
}, },
{ {
"name": "SeasonFolderFormat", "name": "seasonFolderFormat",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "string" "type": "string"
} }
}, },
{ {
"name": "SpecialsFolderFormat", "name": "specialsFolderFormat",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "string" "type": "string"
} }
}, },
{ {
"name": "IncludeSeriesTitle", "name": "includeSeriesTitle",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "boolean" "type": "boolean"
} }
}, },
{ {
"name": "IncludeEpisodeTitle", "name": "includeEpisodeTitle",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "boolean" "type": "boolean"
} }
}, },
{ {
"name": "IncludeQuality", "name": "includeQuality",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "boolean" "type": "boolean"
} }
}, },
{ {
"name": "ReplaceSpaces", "name": "replaceSpaces",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "boolean" "type": "boolean"
} }
}, },
{ {
"name": "Separator", "name": "separator",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "string" "type": "string"
} }
}, },
{ {
"name": "NumberStyle", "name": "numberStyle",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "string" "type": "string"
} }
}, },
{ {
"name": "Id", "name": "id",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "integer", "type": "integer",
@ -4447,7 +4661,7 @@
} }
}, },
{ {
"name": "ResourceName", "name": "resourceName",
"in": "query", "in": "query",
"schema": { "schema": {
"type": "string" "type": "string"
@ -7402,7 +7616,7 @@
"type": "array", "type": "array",
"items": { "items": {
"type": "object", "type": "object",
"additionalProperties": { } "additionalProperties": {}
}, },
"nullable": true "nullable": true
} }
@ -11628,10 +11842,10 @@
}, },
"security": [ "security": [
{ {
"X-Api-Key": [ ] "X-Api-Key": []
}, },
{ {
"apikey": [ ] "apikey": []
} }
] ]
} }

View File

@ -1,8 +1,20 @@
using System.Collections.Generic; using System.Collections.Generic;
using System.ComponentModel;
using NzbDrone.Core.Datastore; using NzbDrone.Core.Datastore;
namespace Sonarr.Http namespace Sonarr.Http
{ {
public class PagingRequestResource
{
[DefaultValue(1)]
public int? Page { get; set; }
[DefaultValue(10)]
public int? PageSize { get; set; }
public string SortKey { get; set; }
public SortDirection? SortDirection { get; set; }
public List<PagingResourceFilter> Filters { get; set; }
}
public class PagingResource<TResource> public class PagingResource<TResource>
{ {
public int Page { get; set; } public int Page { get; set; }
@ -10,8 +22,22 @@ namespace Sonarr.Http
public string SortKey { get; set; } public string SortKey { get; set; }
public SortDirection SortDirection { get; set; } public SortDirection SortDirection { get; set; }
public List<PagingResourceFilter> Filters { get; set; } public List<PagingResourceFilter> Filters { get; set; }
public int TotalRecords { get; set; } public int TotalRecords { get; set; }
public List<TResource> Records { get; set; } public List<TResource> Records { get; set; }
public PagingResource()
{
}
public PagingResource(PagingRequestResource requestResource)
{
Page = requestResource.Page ?? 1;
PageSize = requestResource.PageSize ?? 10;
SortKey = requestResource.SortKey;
SortDirection = requestResource.SortDirection ?? SortDirection.Descending;
Filters = requestResource.Filters ?? new List<PagingResourceFilter>();
}
} }
public static class PagingResourceMapper public static class PagingResourceMapper