🎉 first push

master
Marc CHEVALDONNE 2 years ago
parent 8c7637772c
commit c4dea1860f

@ -1,2 +1,226 @@
# OpenLibraryWS_Wrapper
# OpenLibraryWS_wrapper
## Summary
This project proposes a Web Service wrapping the [OpenLibrary](https://openlibrary.org/). This is an archive of books and editions and it proposes a web api. This work here is a wrapper in order to simplify the use of this api, with less routes, and also to be interchangeable with another web service that will be issued shortly, owning its proper database of books.
Its purpose is mainly and only to be used by my students in order to practice a little bit _continuous integration_ and _continuous deployment_.
## Table of Contents
[Documentation](#documentation)
[Getting Started](#getting-started)
[Where are we now?](#where-are-we-now)
[Usage](#usage)
[Running the tests](#running-the-tests)
[Known issues and limitations](#known-issues-and-limitations)
[Built with](#built-with)
[Authors](#authors)
[Acknowledgements](#acknowledgements)
## Documentation
There is no other documentation than this ```ReadMe```, at least for now.
But here are some useful informations.
### Package diagram
```mermaid
flowchart LR
DTOAbstractLayer -.-> LibraryDTO
JSonReader -.-> LibraryDTO
OpenLibraryClient -.-> DTOAbstractLayer
OpenLibraryClient -.-> JSonReader
StubbedDTO -.-> DTOAbstractLayer
StubbedDTO -.-> JSonReader
OpenLibraryWrapper -.-> OpenLibraryClient
OpenLibraryWrapper -.-> StubbedDTO
```
This work contains some projects:
- ```LibraryDTO```: contains all **DTO** (_Data Transfer Object_) used by the web service(s) of this project
- ```DTOAbstractLayer```: defines a unique interface (```IDtoManager```) for all objects that can be used to make requests
- ```JSonReader```: contains some tools simplifying the parsing of json request results coming from ```OpenLibrary API```
- ```StubbedDTO```: is only used for testing the preceding packages. It contains some json files copied from ```OpenLibrary API``` request results
- ```OpenLibraryClient```: this is the wrapper of ```OpenLibrary API```. It implements ```IDtoManager``` and uses ```OpenLibrary API``` requests to get books and authors.
- ```OpenLibraryWrapper```: defines our Web Service and its routes.
### Routes of ```OpenLibraryWrapper``` web service:
There are very few routes:
- **get books by title**: allows to get books containing a particular string in their titles.
```
book/getbooksbytitle?title=ne&index=0&count=5
book/getbooksbytitle?title=ne&index=0&count=5&sort=old
```
- **get books by author**: allows to get books containing a particular string in the author names or alternate names.
```
book/getbooksbyauthor?name=al&index=0&count=5
book/getbooksbyauthor?name=al&index=0&count=5&sort=old
```
- **get books by author id**: allows to get books of a particular author by giving its id.
```
book/getbooksbyauthorid?id=OL1846639A&index=0&count=5
book/getbooksbyauthorid?id=OL1846639A&index=0&count=5&sort=old
```
- **get authors by name**: allows to get authors whose name (or alternate names) contains a particular string.
```
book/getauthorsbyname?name=al&index=0&count=5
book/getauthorsbyname?name=al&index=0&count=5&sort=name
```
- **get book by isbn**: allows to get a book by giving its isbn.
```
book/getBookByIsbn/9782330033118
```
- **get book by id**: allows to get a book by giving its id.
```
book/getBookById/OL25910297M
```
- **get author by id**: allows to get an author by giving its id.
```
book/getAuthorById/OL1846639A
```
### Class diagram
For what it's worth...
You will probably not need it...
Nevertheless, it shows how **DTO** classes are working with each other.
```mermaid
classDiagram
direction LR
class BookDTO {
+Id : string
+Title : string
+Publishers : List~string~
+PublishDate : DateTime
+ISBN13 : string
+Series : List~string~
+NbPages : int
+Format : string
+ImageSmall : string
+ImageMedium : string
+ImageLarge : string
}
class Languages {
<<enum>>
Unknown,
French,
}
class WorkDTO {
+Id : string
+Description : string
+Title : string
+Subjects : List~string~
}
class ContributorDTO {
+Name : string
+Role : string
}
class AuthorDTO {
+Id : string
+Name : string
+ImageSmall : string
+ImageMedium : string
+ImageLarge : string
+Bio : string
+AlternateNames : List~string~
+BirthDate : DateTime?
+DeathDate : DateTime?
}
class LinkDTO {
+Title : string
+Url : string
}
class RatingsDTO {
+Average : float
+Count : int
}
BookDTO --> "1" Languages : Language
BookDTO --> "*" ContributorDTO : Contributors
AuthorDTO --> "*" LinkDTO : Links
WorkDTO --> "*" AuthorDTO : Authors
WorkDTO --> "1" RatingsDTO : Ratings
BookDTO --> "*" AuthorDTO : Authors
BookDTO --> "*" WorkDTO : Works
```
## Getting Started
If you want to test this project locally, simply ```git clone``` this project, and open the solution ```Sources/OpenLibraryWS_Wrapper.sln```.
### Prerequisites
- Visual Studio 2019 or Visual Studio for Mac
- .NET 7.0 or higher
### Setup
Just ```git clone``` and build the solution.
## Where are we now?
Well, some parts are missing for pedagogical purposes...
- There is no CI/CD file. You will have to prepare it all by yourself.
- There is no files allowing to generate documentation (doxygen documentation for the code or swagger for the web api routes)
- There is no ```Dockerfile```.
Moreover, a lot of stuff could be enhanced, but I do not have time for this:
- there are too few unit tests
- there are too few comments in code
- the second version of the web service with its own database is not ready yet (but soon hopefully).
## Running the tests
You can run some unit tests but there are few. The unit tests project is ```OpenLibraryWrapper_UT```.
Run them in Visual Studio or using the command ```dotnet test```.
## Known issues and limitations
- CI/CD is not set yet.
- Documentation is not deployed.
## Built with
.NET and Visual Studio for Mac
## Next steps
This project should be enhanced with _Continuous Integration_ and _Continuous Deployment_ pipelines.
Here are the different steps that should be added:
1. **build** job:
All the projects of the solution should be built and published.
In order to write this job, one could find useful information in:
- the [**code first documentation about CI build jobs**](https://codefirst.iut.uca.fr/documentation/CodeFirst/docusaurus/GuidesTutorials/docs/CI-CD/CI/build/)
- the [**dotnet official documentation**](https://learn.microsoft.com/en-us/dotnet/core/tools/dotnet) about .NET command lines.
To build and publish a .NET solution, you usually need to ```restore```, ```build``` and ```publish```.
2. **unit tests** job:
All the unit tests projects of the solution should be run.
In order to write this job, one could find useful information in:
- the [**code first documentation about CI unit tests jobs**](https://codefirst.iut.uca.fr/documentation/CodeFirst/docusaurus/GuidesTutorials/docs/CI-CD/CI/tests/)
- the [**dotnet official documentation**](https://learn.microsoft.com/en-us/dotnet/core/tools/dotnet) about .NET command lines.
To run .NET unit tests projects, you usually need to ```restore``` and ```test```.
3. **continuous integration** job:
A code analysis should be run and exported to **Sonarqube**.
One can find useful information in:
- the [**code first documentation about CI unit tests jobs**](https://codefirst.iut.uca.fr/documentation/CodeFirst/docusaurus/GuidesTutorials/docs/CI-CD/CI/codeInspection/)
- the [**SonarQube documentation**](https://docs.sonarsource.com/sonarqube/latest/analyzing-source-code/scanners/sonarscanner-for-dotnet/) for .NET
4. **documentation** jobs:
Two different documentations should be published:
- **doxygen**: although very few comments are present in the source code, a doxygen documentation should be deployed.
It could look like this:
<img src="screens/doxygen1.png" width="300px"/>
And the footer could look like this:
<img src="screens/doxygen2.png" width="300px"/>
One could find useful information in:
- the [**code first documentation about doxygen jobs**](https://codefirst.iut.uca.fr/documentation/CodeFirst/docusaurus/GuidesTutorials/docs/CI-CD/DocJobs/doxygen/)
- the [**official doxygen documentation**](https://www.doxygen.nl/)
- **swagger**: the routes of the web service should be documented through a Swagger documentation. But nothing has been prepared in Web Service project. First, one should modify the .NET project to add Swagger documentation generation, and next modify the documentation job.
It could look like this:
<img src="screens/swagger.png" width="300px"/>
And the footer could look like this:
One could find useful information in:
- the [**documentation about how to integrate Swagger in a .NET project**](https://learn.microsoft.com/en-us/aspnet/core/tutorials/getting-started-with-swashbuckle?view=aspnetcore-7.0&tabs=visual-studio).
## Authors
Marc Chevaldonné
## Acknowledgements
Camille Petitalot and Cédric Bouhours

@ -0,0 +1,12 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net7.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
</PropertyGroup>
<ItemGroup>
<ProjectReference Include="..\LibraryDTO\LibraryDTO.csproj" />
</ItemGroup>
</Project>

@ -0,0 +1,97 @@
using LibraryDTO;
namespace DtoAbstractLayer;
/// <summary>
/// abstract layer for requests on Books Library
/// </summary>
public interface IDtoManager
{
/// <summary>
/// get a book by specifying its id
/// </summary>
/// <param name="id">id of the Book to get</param>
/// <returns>a Book with this id (or null if id is unknown)</returns>
Task<BookDTO> GetBookById(string id);
/// <summary>
/// get a book by specifying its isbn
/// </summary>
/// <param name="isbn">isbn of the Book to get</param>
/// <returns>a Book with this isbn (or null if isbn is unknown)</returns>
Task<BookDTO> GetBookByISBN(string isbn);
/// <summary>
/// get books containing a substring in their titles
/// </summary>
/// <param name="title">the substring to look for in book titles</param>
/// <param name="index">index of the page of resulting books</param>
/// <param name="count">number of resulting books per page</param>
/// <param name="sort">sort criterium (not mandatory):
/// <ul>
/// <li>```title```: sort books by titles in alphabetical order,</li>
/// <li>```title_reverse```: sort books by titles in reverse alphabetical order,</li>
/// <li>```new```: sort books by publishing dates, beginning with the most recents,</li>
/// <li>```old```: sort books by publishing dates, beginning with the oldest</li>
/// </ul>
/// </param>
/// <returns>max <i>count</i> books</returns>
Task<Tuple<long, IEnumerable<BookDTO>>> GetBooksByTitle(string title, int index, int count, string sort = "");
/// <summary>
/// get books of a particular author by giving the author id
/// </summary>
/// <param name="authorId">the id of the author</param>
/// <param name="index">index of the page of resulting books</param>
/// <param name="count">number of resulting books per page</param>
/// <param name="sort">sort criterium (not mandatory):
/// <ul>
/// <li>```title```: sort books by titles in alphabetical order,</li>
/// <li>```title_reverse```: sort books by titles in reverse alphabetical order,</li>
/// <li>```new```: sort books by publishing dates, beginning with the most recents,</li>
/// <li>```old```: sort books by publishing dates, beginning with the oldest</li>
/// </ul>
/// </param>
/// <returns>max <i>count</i> books</returns>
Task<Tuple<long, IEnumerable<BookDTO>>> GetBooksByAuthorId(string authorId, int index, int count, string sort = "");
/// <summary>
/// get books of authors whose name (or alternate names) contains a particular substring
/// </summary>
/// <param name="author">name to look for in author names or alternate names</param>
/// <param name="index">index of the page of resulting books</param>
/// <param name="count">number of resulting books per page</param>
/// <param name="sort">sort criterium (not mandatory):
/// <ul>
/// <li>```title```: sort books by titles in alphabetical order,</li>
/// <li>```title_reverse```: sort books by titles in reverse alphabetical order,</li>
/// <li>```new```: sort books by publishing dates, beginning with the most recents,</li>
/// <li>```old```: sort books by publishing dates, beginning with the oldest</li>
/// </ul>
/// </param>
/// <returns>max <i>count</i> books</returns>
Task<Tuple<long, IEnumerable<BookDTO>>> GetBooksByAuthor(string author, int index, int count, string sort = "");
/// <summary>
/// get an author by specifying its id
/// </summary>
/// <param name="id">id of the Author to get</param>
/// <returns>an author with this id (or null if id is unknown)</returns>
Task<AuthorDTO> GetAuthorById(string id);
/// <summary>
/// get authors containing a substring in their names (or alternate names)
/// </summary>
/// <param name="substring">the substring to look for in author names (or alternate names)</param>
/// <param name="index">index of the page of resulting authors</param>
/// <param name="count">number of resulting authors per page</param>
/// <param name="sort">sort criterium (not mandatory):
/// <ul>
/// <li>```name```: sort authors by names in alphabetical order,</li>
/// <li>```name_reverse```: sort authors by names in reverse alphabetical order,</li>
/// </ul>
/// </param>
/// <returns>max <i>count</i> authors</returns>
Task<Tuple<long, IEnumerable<AuthorDTO>>> GetAuthorsByName(string substring, int index, int count, string sort = "");
}

@ -0,0 +1,53 @@
using System;
using LibraryDTO;
using Newtonsoft.Json.Linq;
using System.Globalization;
namespace JsonReader
{
public static class AuthorJsonReader
{
public static AuthorDTO ReadAuthor(string json)
{
JObject o = JObject.Parse(json);
string bioTokenAsString = null;
if(o.TryGetValue("bio", out JToken? bioToken))
{
if(bioToken.Type == JTokenType.String)
{
bioTokenAsString = (string)bioToken;
}
else
{
var bioTokenValue = o["bio"]?["value"];
bioTokenAsString = (string)bioTokenValue;
}
}
AuthorDTO author = new AuthorDTO
{
Id = (string)o["key"],
Name = (string)o["name"],
Bio = bioTokenAsString,
BirthDate = o.TryGetValue("birth_date", out JToken? bd) ? DateTime.ParseExact((string)bd, "d MMMM yyyy", CultureInfo.InvariantCulture) : null,
DeathDate = o.TryGetValue("death_date", out JToken? dd) ? DateTime.ParseExact((string)dd, "d MMMM yyyy", CultureInfo.InvariantCulture) : null,
Links = o.TryGetValue("links", out JToken? links) ? links.Select(l => new LinkDTO { Title = (string)l["title"], Url = (string)l["url"] }).ToList() : new List<LinkDTO>(),
AlternateNames = o.TryGetValue("alternate_names", out JToken? altNames) ? altNames.Select(alt => (string)alt).ToList() : new List<string?>()
};
return author;
}
public static Tuple<long, IEnumerable<AuthorDTO>> GetAuthorsByName(string json)
{
JObject o = JObject.Parse(json);
long numFound = (long)o["numFound"];
var authors = o["docs"].Select(doc => new AuthorDTO
{
Id = $"/authors/{(string)doc["key"]}",
Name = (string)doc["name"],
});
return Tuple.Create(numFound, authors);
}
}
}

@ -0,0 +1,82 @@
using System.Globalization;
using LibraryDTO;
using Newtonsoft.Json.Linq;
namespace JsonReader;
public static class BookJsonReader
{
static Dictionary<string, Languages> languages = new Dictionary<string, Languages>()
{
[@"/languages/fre"] = Languages.French,
[@"/languages/eng"] = Languages.English,
["fre"] = Languages.French,
["eng"] = Languages.English,
[""] = Languages.Unknown
};
public static BookDTO ReadBook(string json)
{
JObject o = JObject.Parse(json);
var l = o["languages"]?.FirstOrDefault("");
Languages lang = l != null ? languages[(string)l["key"]] : Languages.Unknown;
Tuple<string, CultureInfo> pubDateFormat = lang switch
{
Languages.French => Tuple.Create("d MMMM yyyy", CultureInfo.GetCultureInfo("fr-FR")),
Languages.Unknown => Tuple.Create("MMM dd, yyyy", CultureInfo.InvariantCulture)
};
BookDTO book = new BookDTO
{
Id = (string)o["key"],
Title = (string)o["title"],
Publishers = o["publishers"].Select(p => (string)p).ToList(),
PublishDate = DateTime.TryParseExact((string)o["publish_date"], pubDateFormat.Item1, pubDateFormat.Item2, DateTimeStyles.None, out DateTime date) ? date : new DateTime((int)o["publish_date"], 12, 31),
ISBN13 = (string)o["isbn_13"][0],
NbPages = (int)o["number_of_pages"],
Language = lang,
Format = o.TryGetValue("physical_format", out JToken? f) ? (string)f : null,
Works = o["works"].Select(w => new WorkDTO { Id = (string)w["key"] }).ToList(),
Contributors = o.TryGetValue("contributors", out JToken? contr) ? contr.Select(c => new ContributorDTO { Name = (string)c["name"], Role = (string)c["role"] }).ToList() : new List<ContributorDTO>(),
Authors = o["authors"]?.Select(a => new AuthorDTO { Id = (string)a["key"] }).ToList()
};
if(book.Authors == null)
{
book.Authors = new List<AuthorDTO>();
}
return book;
}
public static Tuple<long, IEnumerable<BookDTO>> GetBooksByAuthor(string json)
{
JObject o = JObject.Parse(json);
long numFound = (long)o["numFound"];
var books = o["docs"].Select(doc => new BookDTO
{
Id = (string)(doc["seed"].First()),
Title = (string)doc["title"],
ISBN13 = (string)(doc["isbn"].First()),
Authors = doc["seed"].Where(s => ((string)s).StartsWith("/authors/"))
.Select(s => new AuthorDTO { Id = (string)s }).ToList(),
Language = languages.GetValueOrDefault((string)(doc["language"].First()))
});
return Tuple.Create(numFound, books);
}
public static Tuple<long, IEnumerable<BookDTO>> GetBooksByTitle(string json)
{
JObject o = JObject.Parse(json);
long numFound = (long)o["numFound"];
var books = o["docs"].Select(doc => new BookDTO
{
Id = (string)(doc["seed"].First()),
Title = (string)doc["title"],
ISBN13 = (string)(doc["isbn"].First()),
Authors = doc["seed"].Where(s => ((string)s).StartsWith("/authors/"))
.Select(s => new AuthorDTO { Id = (string)s }).ToList(),
Language = languages.GetValueOrDefault((string)(doc["language"].First()))
});
return Tuple.Create(numFound, books);
}
}

@ -0,0 +1,15 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net7.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="Newtonsoft.Json" Version="13.0.3" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\LibraryDTO\LibraryDTO.csproj" />
</ItemGroup>
</Project>

@ -0,0 +1,39 @@
using System;
using LibraryDTO;
using Newtonsoft.Json.Linq;
using System.Globalization;
namespace JsonReader
{
public static class WorkJsonReader
{
public static WorkDTO ReadWork(string json, string ratingsJson)
{
JObject o = JObject.Parse(json);
JObject r = JObject.Parse(ratingsJson);
var ratingsDto = new RatingsDTO();
if(r["summary"]["average"].Type != JTokenType.Float)
{
ratingsDto.Average = -1;
ratingsDto.Count = 0;
}
else
{
ratingsDto.Average = (float)r["summary"]["average"];
ratingsDto.Count = (int)r["summary"]["count"];
}
WorkDTO work = new WorkDTO
{
Id = (string)o["key"],
Title = (string)o["title"],
Authors = o["authors"].Select(a => new AuthorDTO { Id = (string)a["author"]["key"] }).ToList(),
Description = o.TryGetValue("description", out JToken? descr) ? (string)descr : null,
Subjects = o.TryGetValue("subjects", out JToken? subjects) ? subjects.Select(s => (string)s).ToList() : new List<string>(),
Ratings = ratingsDto
};
return work;
}
}
}

@ -0,0 +1,18 @@
using System;
namespace LibraryDTO
{
public class AuthorDTO
{
public string Id { get; set; }
public string Name { get; set; }
public string ImageSmall => $"https://covers.openlibrary.org/a/olid/{Id.Substring(Id.LastIndexOf("/"))}-S.jpg";
public string ImageMedium => $"https://covers.openlibrary.org/a/olid/{Id.Substring(Id.LastIndexOf("/"))}-M.jpg";
public string ImageLarge => $"https://covers.openlibrary.org/a/olid/{Id.Substring(Id.LastIndexOf("/"))}-L.jpg";
public string Bio { get; set; }
public List<string> AlternateNames { get; set; } = new List<string>();
public List<LinkDTO> Links { get; set; }
public DateTime? BirthDate { get; set; }
public DateTime? DeathDate { get; set; }
}
}

@ -0,0 +1,23 @@
using System;
namespace LibraryDTO
{
public class BookDTO
{
public string Id { get; set; }
public string Title { get; set; }
public List<string> Publishers { get; set; } = new List<string>();
public DateTime PublishDate { get; set; }
public string ISBN13 { get; set; }
public List<string> Series { get; set; } = new List<string>();
public int NbPages { get; set; }
public string Format { get; set; }
public Languages Language { get; set; }
public List<ContributorDTO> Contributors { get; set; }
public string ImageSmall => $"https://covers.openlibrary.org/b/isbn/{ISBN13}-S.jpg";
public string ImageMedium => $"https://covers.openlibrary.org/b/isbn/{ISBN13}-M.jpg";
public string ImageLarge => $"https://covers.openlibrary.org/b/isbn/{ISBN13}-L.jpg";
public List<WorkDTO> Works { get; set; } = new List<WorkDTO>();
public List<AuthorDTO> Authors { get; set; } = new List<AuthorDTO>();
}
}

@ -0,0 +1,10 @@
using System;
namespace LibraryDTO
{
public class ContributorDTO
{
public string Name { get; set; }
public string Role { get; set; }
}
}

@ -0,0 +1,11 @@
using System;
namespace LibraryDTO
{
public enum Languages
{
Unknown,
French,
English
}
}

@ -0,0 +1,16 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net7.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
</PropertyGroup>
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' ">
<WarningLevel>4</WarningLevel>
</PropertyGroup>
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Release|AnyCPU' ">
<DebugSymbols>true</DebugSymbols>
<WarningLevel>4</WarningLevel>
</PropertyGroup>
</Project>

@ -0,0 +1,10 @@
using System;
namespace LibraryDTO
{
public class LinkDTO
{
public string Title { get; set; }
public string Url { get; set; }
}
}

@ -0,0 +1,10 @@
using System;
namespace LibraryDTO
{
public class RatingsDTO
{
public float Average { get; set; }
public int Count { get; set; }
}
}

@ -0,0 +1,14 @@
using System;
namespace LibraryDTO
{
public class WorkDTO
{
public string Id { get; set; }
public string Description { get; set; }
public string Title { get; set; }
public List<string> Subjects { get; set; } = new List<string>();
public List<AuthorDTO> Authors { get; set; } = new List<AuthorDTO>();
public RatingsDTO Ratings { get; set; }
}
}

@ -0,0 +1,104 @@
using System;
using System.Data.SqlTypes;
using System.Net;
using System.Text.Json;
using DtoAbstractLayer;
using JsonReader;
using LibraryDTO;
using static System.Net.WebRequestMethods;
namespace OpenLibraryClient;
public class OpenLibClientAPI : IDtoManager
{
const string BasePath = @"https://openlibrary.org/";
const string SearchAuthorPrefix = @"search/authors.json?q=";
const string SearchBookTitlePrefix = @"search.json?title=";
const string SearchBookByAuthorPrefix = @"search.json?author=";
const string AuthorPrefix = @"authors/";
const string BookPrefix = @"books/";
const string IsbnPrefix = @"isbn/";
HttpClient client = new HttpClient();
JsonSerializerOptions SerializerOptions { get; set; } = new JsonSerializerOptions
{
PropertyNamingPolicy = JsonNamingPolicy.CamelCase,
WriteIndented = true
};
private async Task<T> GetElement<T>(string route, Func<string,T> deserializer)
{
try
{
T result = default(T);
Uri uri = new Uri (route, UriKind.RelativeOrAbsolute);
HttpResponseMessage response = await client.GetAsync (uri);
if (response.IsSuccessStatusCode)
{
string content = await response.Content.ReadAsStringAsync ();
result = deserializer(content);
}
return result;
}
catch(Exception exc)
{
throw new WebException($"The route {route} seems to be invalid");
}
}
public async Task<AuthorDTO> GetAuthorById(string id)
{
string route = $"{BasePath}{AuthorPrefix}{id}.json";
return await GetElement<AuthorDTO>(route, json => AuthorJsonReader.ReadAuthor(json));
}
public async Task<Tuple<long, IEnumerable<AuthorDTO>>> GetAuthorsByName(string substring, int index, int count, string sort = "")
{
string searchedString = substring.Trim().Replace(" ", "+");
string route = $"{BasePath}{SearchAuthorPrefix}{searchedString}"
.AddPagination(index, count)
.AddSort(sort);
return await GetElement<Tuple<long, IEnumerable<AuthorDTO>>>(route, json => AuthorJsonReader.GetAuthorsByName(json));
}
public async Task<BookDTO> GetBookById(string id)
{
string route = $"{BasePath}{BookPrefix}{id}.json";
return await GetElement<BookDTO>(route, json => BookJsonReader.ReadBook(json));
}
public async Task<BookDTO> GetBookByISBN(string isbn)
{
string route = $"{BasePath}{IsbnPrefix}{isbn}.json";
return await GetElement<BookDTO>(route, json => BookJsonReader.ReadBook(json));
}
public async Task<Tuple<long, IEnumerable<BookDTO>>> GetBooksByAuthor(string author, int index, int count, string sort = "")
{
string searchedString = author.Trim().Replace(" ", "+");
string route = $"{BasePath}{SearchBookByAuthorPrefix}{searchedString}"
.AddPagination(index, count)
.AddSort(sort);
return await GetElement<Tuple<long, IEnumerable<BookDTO>>>(route, json => BookJsonReader.GetBooksByAuthor(json));
}
public async Task<Tuple<long, IEnumerable<BookDTO>>> GetBooksByAuthorId(string authorId, int index, int count, string sort = "")
{
string searchedString = authorId.Trim().Replace(" ", "+");
string route = $"{BasePath}{SearchAuthorPrefix}{searchedString}"
.AddPagination(index, count)
.AddSort(sort);
return await GetElement<Tuple<long, IEnumerable<BookDTO>>>(route, json => BookJsonReader.GetBooksByAuthor(json));
}
public async Task<Tuple<long, IEnumerable<BookDTO>>> GetBooksByTitle(string title, int index, int count, string sort = "")
{
string searchedString = title.Trim().Replace(" ", "+");
string route = $"{BasePath}{SearchBookTitlePrefix}{searchedString}"
.AddPagination(index, count)
.AddSort(sort);
return await GetElement<Tuple<long, IEnumerable<BookDTO>>>(route, json => BookJsonReader.GetBooksByTitle(json));
}
}

@ -0,0 +1,14 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net7.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
</PropertyGroup>
<ItemGroup>
<ProjectReference Include="..\LibraryDTO\LibraryDTO.csproj" />
<ProjectReference Include="..\DtoAbstractLayer\DtoAbstractLayer.csproj" />
<ProjectReference Include="..\JsonReader\JsonReader.csproj" />
</ItemGroup>
</Project>

@ -0,0 +1,33 @@
using System;
namespace OpenLibraryClient
{
public static class RouteExtensions
{
public static string AddPagination(this string route, int index, int count)
{
if(index <= -1 || count<0)
{
return route;
}
string delimiter = route.Contains("?") ? "&" : "?";
return $"{route}{delimiter}limit={count}&page={index+1}";
}
public static string AddSort(this string route, string sort)
{
string sortCriterium = sort switch
{
"new" => "new",
"old" => "old",
"random" => "random",
"key" => "key",
_ => null
};
if(sortCriterium == null) return route;
string delimiter = route.Contains("?") ? "&" : "?";
return $"{route}{delimiter}sort={sortCriterium}";
}
}
}

@ -0,0 +1,68 @@

Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio Version 16
VisualStudioVersion = 25.0.1705.7
MinimumVisualStudioVersion = 10.0.40219.1
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "OpenLibraryWrapper", "OpenLibraryWrapper\OpenLibraryWrapper.csproj", "{EF0DED5C-7559-4D43-A30B-AE916FCDA078}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Tests", "Tests", "{0D260EDC-4F17-4BA4-9306-B6FEC6E5E394}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "OpenLibraryWrapper_UT", "OpenLibraryWrapper_UT\OpenLibraryWrapper_UT.csproj", "{B12BEDA7-EC41-417B-9C28-113A8ED87F35}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "LibraryDTO", "LibraryDTO\LibraryDTO.csproj", "{F59D46BA-6734-464E-8AC4-759BEFB87973}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StubbedDTO", "StubbedDTO\StubbedDTO.csproj", "{3CB3D741-DF5C-4C4A-82C2-5BFC212211AD}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "DtoAbstractLayer", "DtoAbstractLayer\DtoAbstractLayer.csproj", "{43E49F05-E257-441B-B5E4-C82DBE4DDF1A}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "JsonReader", "JsonReader\JsonReader.csproj", "{4EDEABC4-5AEC-4F7B-804C-3E3BA6EE29DF}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{5C6B8D83-705A-4BBD-BBA0-A86ACB370B33}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "OpenLibraryClient", "OpenLibraryClient\OpenLibraryClient.csproj", "{3A429457-D882-44E3-B65E-107554C2E91F}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
Release|Any CPU = Release|Any CPU
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
{EF0DED5C-7559-4D43-A30B-AE916FCDA078}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{EF0DED5C-7559-4D43-A30B-AE916FCDA078}.Debug|Any CPU.Build.0 = Debug|Any CPU
{EF0DED5C-7559-4D43-A30B-AE916FCDA078}.Release|Any CPU.ActiveCfg = Release|Any CPU
{EF0DED5C-7559-4D43-A30B-AE916FCDA078}.Release|Any CPU.Build.0 = Release|Any CPU
{B12BEDA7-EC41-417B-9C28-113A8ED87F35}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{B12BEDA7-EC41-417B-9C28-113A8ED87F35}.Debug|Any CPU.Build.0 = Debug|Any CPU
{B12BEDA7-EC41-417B-9C28-113A8ED87F35}.Release|Any CPU.ActiveCfg = Release|Any CPU
{B12BEDA7-EC41-417B-9C28-113A8ED87F35}.Release|Any CPU.Build.0 = Release|Any CPU
{F59D46BA-6734-464E-8AC4-759BEFB87973}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{F59D46BA-6734-464E-8AC4-759BEFB87973}.Debug|Any CPU.Build.0 = Debug|Any CPU
{F59D46BA-6734-464E-8AC4-759BEFB87973}.Release|Any CPU.ActiveCfg = Release|Any CPU
{F59D46BA-6734-464E-8AC4-759BEFB87973}.Release|Any CPU.Build.0 = Release|Any CPU
{3CB3D741-DF5C-4C4A-82C2-5BFC212211AD}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{3CB3D741-DF5C-4C4A-82C2-5BFC212211AD}.Debug|Any CPU.Build.0 = Debug|Any CPU
{3CB3D741-DF5C-4C4A-82C2-5BFC212211AD}.Release|Any CPU.ActiveCfg = Release|Any CPU
{3CB3D741-DF5C-4C4A-82C2-5BFC212211AD}.Release|Any CPU.Build.0 = Release|Any CPU
{43E49F05-E257-441B-B5E4-C82DBE4DDF1A}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{43E49F05-E257-441B-B5E4-C82DBE4DDF1A}.Debug|Any CPU.Build.0 = Debug|Any CPU
{43E49F05-E257-441B-B5E4-C82DBE4DDF1A}.Release|Any CPU.ActiveCfg = Release|Any CPU
{43E49F05-E257-441B-B5E4-C82DBE4DDF1A}.Release|Any CPU.Build.0 = Release|Any CPU
{4EDEABC4-5AEC-4F7B-804C-3E3BA6EE29DF}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{4EDEABC4-5AEC-4F7B-804C-3E3BA6EE29DF}.Debug|Any CPU.Build.0 = Debug|Any CPU
{4EDEABC4-5AEC-4F7B-804C-3E3BA6EE29DF}.Release|Any CPU.ActiveCfg = Release|Any CPU
{4EDEABC4-5AEC-4F7B-804C-3E3BA6EE29DF}.Release|Any CPU.Build.0 = Release|Any CPU
{3A429457-D882-44E3-B65E-107554C2E91F}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{3A429457-D882-44E3-B65E-107554C2E91F}.Debug|Any CPU.Build.0 = Debug|Any CPU
{3A429457-D882-44E3-B65E-107554C2E91F}.Release|Any CPU.ActiveCfg = Release|Any CPU
{3A429457-D882-44E3-B65E-107554C2E91F}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
EndGlobalSection
GlobalSection(ExtensibilityGlobals) = postSolution
SolutionGuid = {2E40DC1C-BE57-48E8-B7C2-B9CFF589DB5B}
EndGlobalSection
GlobalSection(NestedProjects) = preSolution
{B12BEDA7-EC41-417B-9C28-113A8ED87F35} = {0D260EDC-4F17-4BA4-9306-B6FEC6E5E394}
EndGlobalSection
EndGlobal

@ -0,0 +1,274 @@
using System;
using DtoAbstractLayer;
using LibraryDTO;
using Microsoft.AspNetCore.Mvc;
using StubbedDTO;
namespace OpenLibraryWrapper.Controllers
{
[ApiController]
[Route("[controller]")]
public class BookController : ControllerBase
{
private readonly ILogger<BookController> _logger;
private IDtoManager DtoManager;
public BookController(ILogger<BookController> logger, IDtoManager dtoManager)
{
_logger = logger;
DtoManager = dtoManager;
}
/// <summary>
/// Gets books of the collection matching a particular title
/// </summary>
/// <param name="title">part of a title to look for in book titles (case is ignored)</param>
/// <param name="index">index of the page</param>
/// <param name="count">number of elements per page</param>
/// <param name="sort">sort criterium of the resulting books:
/// <ul>
/// <li>```title```: sort books by titles in alphabetical order,</li>
/// <li>```title_reverse```: sort books by titles in reverse alphabetical order,</li>
/// <li>```new```: sort books by publishing dates, beginning with the most recents,</li>
/// <li>```old```: sort books by publishing dates, beginning with the oldest</li>
/// </ul>
///
/// </param>
/// <returns>a collection of count (or less) books</returns>
/// <remarks>
/// Sample requests:
///
/// book/getbooksbytitle?title=ne&amp;index=0&amp;count=5
/// book/getbooksbytitle?title=ne&amp;index=0&amp;count=5&amp;sort=old
///
/// </remarks>
/// <response code="200">Returns count books at page index</response>
/// <response code="404">no books within this range</response>
[HttpGet("getBooksByTitle")]
[ProducesResponseType(typeof(Tuple<long, IEnumerable<BookDTO>>), StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status404NotFound)]
public async Task<IActionResult> GetBooksByTitle([FromQuery] string title, [FromQuery] int index, [FromQuery] int count, [FromQuery] string sort = "")
{
_logger.LogDebug("Get books by title");
var booksDto = (await DtoManager.GetBooksByTitle(title, index, count, sort));
_logger.LogInformation($"{booksDto.Item1} books found");
if(booksDto.Item1 == 0)
{
return NotFound();
}
return Ok(booksDto);
}
/// <summary>
/// Gets books of the collection whose of one the authors is matching a particular name
/// </summary>
/// <param name="name">part of a author name to look for in book authors (case is ignored)</param>
/// <param name="index">index of the page</param>
/// <param name="count">number of elements per page</param>
/// <param name="sort">sort criterium of the resulting books:
/// <ul>
/// <li>```title```: sort books by titles in alphabetical order,</li>
/// <li>```title_reverse```: sort books by titles in reverse alphabetical order,</li>
/// <li>```new```: sort books by publishing dates, beginning with the most recents,</li>
/// <li>```old```: sort books by publishing dates, beginning with the oldest</li>
/// </ul>
///
/// </param>
/// <returns>a collection of count (or less) books</returns>
/// <remarks>
/// Sample requests:
///
/// book/getbooksbyauthor?name=al&amp;index=0&amp;count=5
/// book/getbooksbyauthor?name=al&amp;index=0&amp;count=5&amp;sort=old
///
/// <b>Note:</b>
/// <i>name is also looked for in alternate names of the authors</i>
/// </remarks>
/// <response code="200">Returns count books at page index</response>
/// <response code="404">no books within this range</response>
[HttpGet("getBooksByAuthor")]
[ProducesResponseType(typeof(Tuple<long, IEnumerable<BookDTO>>), StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status404NotFound)]
public async Task<IActionResult> GetBooksByAuthor([FromQuery] string name, [FromQuery] int index, [FromQuery] int count, [FromQuery] string sort = "")
{
_logger.LogDebug("Get books by author");
var booksDto = (await DtoManager.GetBooksByAuthor(name, index, count, sort));
_logger.LogInformation($"{booksDto.Item1} books found");
if(booksDto.Item1 == 0)
{
return NotFound();
}
return Ok(booksDto);
}
/// <summary>
/// Gets books of the collection of a particular author
/// </summary>
/// <param name="id">id of the author</param>
/// <param name="index">index of the page</param>
/// <param name="count">number of elements per page</param>
/// <param name="sort">sort criterium of the resulting books:
/// <ul>
/// <li>```title```: sort books by titles in alphabetical order,</li>
/// <li>```title_reverse```: sort books by titles in reverse alphabetical order,</li>
/// <li>```new```: sort books by publishing dates, beginning with the most recents,</li>
/// <li>```old```: sort books by publishing dates, beginning with the oldest</li>
/// </ul>
///
/// </param>
/// <returns>a collection of count (or less) books</returns>
/// <remarks>
/// Sample requests:
///
/// book/getbooksbyauthorid?id=OL1846639A&amp;index=0&amp;count=5
/// book/getbooksbyauthorid?id=OL1846639A&amp;index=0&amp;count=5&amp;sort=old
///
/// </remarks>
/// <response code="200">Returns count books at page index</response>
/// <response code="404">no books within this range</response>
[HttpGet("getBooksByAuthorId")]
[ProducesResponseType(typeof(Tuple<long, IEnumerable<BookDTO>>), StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status404NotFound)]
public async Task<IActionResult> GetBooksByAuthorId([FromQuery] string id, [FromQuery] int index, [FromQuery] int count, [FromQuery] string sort = "")
{
_logger.LogDebug("Get books by author id");
var booksDto = (await DtoManager.GetBooksByAuthorId(id, index, count, sort));
_logger.LogInformation($"{booksDto.Item1} books found");
if(booksDto.Item1 == 0)
{
return NotFound();
}
return Ok(booksDto);
}
/// <summary>
/// Gets authors of the collection matching a particular name
/// </summary>
/// <param name="name">name to look for in author names</param>
/// <param name="index">index of the page</param>
/// <param name="count">number of elements per page</param>
/// <param name="sort">sort criterium of the resulting authors:
/// <ul>
/// <li>```name```: sort authors by names in alphabetical order,</li>
/// <li>```name_reverse```: sort authors by names in reverse alphabetical order,</li>
/// </ul>
///
/// </param>
/// <returns>a collection of count (or less) authors</returns>
/// <remarks>
/// Sample requests:
///
/// book/getauthorsbyname?name=al&amp;index=0&amp;count=5
/// book/getauthorsbyname?name=al&amp;index=0&amp;count=5&amp;sort=name
///
/// </remarks>
/// <response code="200">Returns count authors at page index</response>
/// <response code="404">no authors within this range</response>
[HttpGet("getAuthorsByName")]
[ProducesResponseType(typeof(Tuple<long, IEnumerable<AuthorDTO>>), StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status404NotFound)]
public async Task<IActionResult> GetAuthorsByName([FromQuery] string name, [FromQuery] int index, [FromQuery] int count, [FromQuery] string sort = "")
{
_logger.LogDebug("Get authors by name");
var authorsDto = (await DtoManager.GetAuthorsByName(name, index, count, sort));
_logger.LogInformation($"{authorsDto.Item1} authors found");
if(authorsDto.Item1 == 0)
{
return NotFound();
}
return Ok(authorsDto);
}
/// <summary>
/// Gets book by isbn
/// </summary>
/// <param name="isbn">isbn of the book to get</param>
/// <returns>the book with the seeked isbn (or null)</returns>
/// <remarks>
/// Sample requests:
///
/// book/getBookByIsbn/9782330033118
///
/// </remarks>
/// <response code="200">Returns the book with this isbn</response>
/// <response code="404">no book found</response>
[HttpGet("getBookByIsbn/{isbn?}")]
[ProducesResponseType(typeof(BookDTO), StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status404NotFound)]
public async Task<IActionResult> GetBookByIsbn(string isbn)
{
_logger.LogDebug("Get book by isbn");
var bookDto = (await DtoManager.GetBookByISBN(isbn));
if(bookDto == null)
{
_logger.LogInformation($"{isbn} not found");
return NotFound();
}
_logger.LogInformation($"{bookDto.Title} found");
return Ok(bookDto);
}
/// <summary>
/// Gets book by id
/// </summary>
/// <param name="id">id of the book to get</param>
/// <returns>the book with the seeked id (or null)</returns>
/// <remarks>
/// Sample requests:
///
/// book/getBookById/OL25910297M
///
/// </remarks>
/// <response code="200">Returns the book with this id</response>
/// <response code="404">no book found</response>
[HttpGet("getBookById/{id?}")]
[ProducesResponseType(typeof(BookDTO), StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status404NotFound)]
public async Task<IActionResult> GetBookById(string id)
{
_logger.LogDebug("Get book by ID");
var bookDto = (await DtoManager.GetBookById(id));
if(bookDto == null)
{
_logger.LogInformation($"{id} not found");
return NotFound();
}
_logger.LogInformation($"{bookDto.Title} found");
return Ok(bookDto);
}
/// <summary>
/// Gets author by id
/// </summary>
/// <param name="id">id of the author to get</param>
/// <returns>the author with the seeked id (or null)</returns>
/// <remarks>
/// Sample requests:
///
/// book/getAuthorById/OL1846639A
///
/// </remarks>
/// <response code="200">Returns the author with this id</response>
/// <response code="404">no author found</response>
[HttpGet("getAuthorById/{id?}")]
[ProducesResponseType(typeof(AuthorDTO), StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status404NotFound)]
public async Task<IActionResult> GetAuthorById(string id)
{
_logger.LogDebug("Get Author by ID");
var authorDTO = (await DtoManager.GetAuthorById(id));
if(authorDTO == null)
{
_logger.LogInformation($"{id} not found");
return NotFound();
}
_logger.LogInformation($"{authorDTO.Name} found");
return Ok(authorDTO);
}
}
}

@ -0,0 +1,40 @@
<Project Sdk="Microsoft.NET.Sdk.Web">
<PropertyGroup>
<TargetFramework>net7.0</TargetFramework>
<Nullable>enable</Nullable>
<ImplicitUsings>enable</ImplicitUsings>
</PropertyGroup>
<PropertyGroup Condition=" '$(RunConfiguration)' == 'https' " />
<PropertyGroup Condition=" '$(RunConfiguration)' == 'http' " />
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' ">
<WarningLevel>4</WarningLevel>
</PropertyGroup>
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Release|AnyCPU' ">
<WarningLevel>4</WarningLevel>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="Microsoft.AspNetCore.OpenApi" Version="7.0.10" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\LibraryDTO\LibraryDTO.csproj">
<GlobalPropertiesToRemove></GlobalPropertiesToRemove>
</ProjectReference>
<ProjectReference Include="..\StubbedDTO\StubbedDTO.csproj">
<GlobalPropertiesToRemove></GlobalPropertiesToRemove>
</ProjectReference>
<ProjectReference Include="..\DtoAbstractLayer\DtoAbstractLayer.csproj">
<GlobalPropertiesToRemove></GlobalPropertiesToRemove>
</ProjectReference>
<ProjectReference Include="..\OpenLibraryClient\OpenLibraryClient.csproj">
<GlobalPropertiesToRemove></GlobalPropertiesToRemove>
</ProjectReference>
</ItemGroup>
<ItemGroup>
<Compile Condition=" '$(EnableDefaultCompileItems)' == 'true' " Update="Program.cs">
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
</Compile>
</ItemGroup>
</Project>

@ -0,0 +1,26 @@
using System.Reflection;
using DtoAbstractLayer;
using LibraryDTO;
using Microsoft.OpenApi.Models;
using OpenLibraryClient;
using StubbedDTO;
var builder = WebApplication.CreateBuilder(args);
// Add services to the container.
builder.Services.AddSingleton<IDtoManager,OpenLibClientAPI>();
builder.Services.AddControllers();
// Learn more about configuring Swagger/OpenAPI at https://aka.ms/aspnetcore/swashbuckle
builder.Services.AddEndpointsApiExplorer();
var app = builder.Build();
app.UseHttpsRedirection();
app.UseAuthorization();
app.MapControllers();
app.Run();

@ -0,0 +1,41 @@
{
"$schema": "https://json.schemastore.org/launchsettings.json",
"iisSettings": {
"windowsAuthentication": false,
"anonymousAuthentication": true,
"iisExpress": {
"applicationUrl": "http://localhost:34424",
"sslPort": 44344
}
},
"profiles": {
"http": {
"commandName": "Project",
"launchBrowser": true,
"launchUrl": "swagger",
"applicationUrl": "http://localhost:5117",
"environmentVariables": {
"ASPNETCORE_ENVIRONMENT": "Development"
},
"dotnetRunMessages": true
},
"https": {
"commandName": "Project",
"launchBrowser": true,
"launchUrl": "swagger",
"applicationUrl": "https://localhost:7263;http://localhost:5117",
"environmentVariables": {
"ASPNETCORE_ENVIRONMENT": "Development"
},
"dotnetRunMessages": true
},
"IIS Express": {
"commandName": "IISExpress",
"launchBrowser": true,
"launchUrl": "swagger",
"environmentVariables": {
"ASPNETCORE_ENVIRONMENT": "Development"
}
}
}
}

@ -0,0 +1,9 @@
{
"Logging": {
"LogLevel": {
"Default": "Information",
"Microsoft.AspNetCore": "Warning"
}
}
}

@ -0,0 +1,10 @@
{
"Logging": {
"LogLevel": {
"Default": "Information",
"Microsoft.AspNetCore": "Warning"
}
},
"AllowedHosts": "*"
}

@ -0,0 +1,80 @@
using DtoAbstractLayer;
using LibraryDTO;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Http.HttpResults;
using Microsoft.AspNetCore.Mvc;
using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Logging.Abstractions;
using OpenLibraryClient;
using OpenLibraryWrapper.Controllers;
using StubbedDTO;
namespace OpenLibraryWrapper_UT;
public class BookController_UT
{
private readonly BookController controller;
private readonly IDtoManager dtoManager = new OpenLibClientAPI();
public BookController_UT()
{
var logger = new NullLogger<BookController>();
controller = new BookController(logger, dtoManager);
}
[Theory]
[InlineData(true, "L'\u00c9veil du L\u00e9viathan", "9782330033118")]
[InlineData(false, null, "1782330033118")]
public async void TestGetBookByIsbn(bool expectedResult, string expectedTitle, string isbn)
{
var result = await controller.GetBookByIsbn(isbn);
Assert.Equal(expectedResult, result is OkObjectResult);
if(result is not OkObjectResult)
{
return;
}
var okResult = result as OkObjectResult;
var bookDto = okResult.Value as BookDTO;
Assert.Equal(expectedTitle, bookDto.Title);
}
[Theory]
[InlineData(true, "L'\u00c9veil du L\u00e9viathan", "OL25910297M")]
[InlineData(false, null, "OL25910xxxM")]
public async void TestGetBookById(bool expectedResult, string expectedTitle, string id)
{
var result = await controller.GetBookById(id);
Assert.Equal(expectedResult, result is OkObjectResult);
if(result is not OkObjectResult)
{
return;
}
var okResult = result as OkObjectResult;
var bookDto = okResult.Value as BookDTO;
Assert.Equal(expectedTitle, bookDto.Title);
}
[Fact]
public async void TestGetBooksByTitle()
{
var result = await controller.GetBooksByTitle("ne", 0, 5);
var okResult = result as OkObjectResult;
var booksTupple = (Tuple<long, IEnumerable<BookDTO>>)okResult.Value;
long nbBooks = booksTupple.Item1;
var books = booksTupple.Item2;
Assert.True(nbBooks > 0);
}
[Fact]
public async void TestGetBooksByAuthor()
{
var result = await controller.GetBooksByAuthor("al", 0, 5);
var okResult = result as OkObjectResult;
var booksTupple = (Tuple<long, IEnumerable<BookDTO>>)okResult.Value;
long nbBooks = booksTupple.Item1;
var books = booksTupple.Item2;
Assert.Equal(5, books.Count());
}
}

@ -0,0 +1,35 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net7.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
<IsPackable>false</IsPackable>
<IsTestProject>true</IsTestProject>
</PropertyGroup>
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' ">
<WarningLevel>4</WarningLevel>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.5.0" />
<PackageReference Include="xunit" Version="2.4.2" />
<PackageReference Include="xunit.runner.visualstudio" Version="2.4.5">
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
<PrivateAssets>all</PrivateAssets>
</PackageReference>
<PackageReference Include="coverlet.collector" Version="3.2.0">
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
<PrivateAssets>all</PrivateAssets>
</PackageReference>
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\OpenLibraryWrapper\OpenLibraryWrapper.csproj" />
<ProjectReference Include="..\StubbedDTO\StubbedDTO.csproj" />
<ProjectReference Include="..\LibraryDTO\LibraryDTO.csproj" />
<ProjectReference Include="..\DtoAbstractLayer\DtoAbstractLayer.csproj" />
<ProjectReference Include="..\OpenLibraryClient\OpenLibraryClient.csproj" />
</ItemGroup>
</Project>

@ -0,0 +1 @@
global using Xunit;

@ -0,0 +1,200 @@
using System.Data.SqlTypes;
using DtoAbstractLayer;
using JsonReader;
using LibraryDTO;
using static System.Reflection.Metadata.BlobBuilder;
namespace StubbedDTO;
public class Stub : IDtoManager
{
public static List<AuthorDTO> Authors { get; set; } = new List<AuthorDTO>();
public static List<BookDTO> Books { get; set; } = new List<BookDTO>();
public static List<WorkDTO> Works { get; set; } = new List<WorkDTO>();
public static string BasePath { get; set; } = "";
static Stub()
{
foreach(var fileAuthor in new DirectoryInfo($"{BasePath}authors/").GetFiles())
{
using(StreamReader reader = File.OpenText(fileAuthor.FullName))
{
Authors.Add(AuthorJsonReader.ReadAuthor(reader.ReadToEnd()));
}
}
foreach(var fileWork in new DirectoryInfo($"{BasePath}works/").GetFiles())
{
var ratingsFile = $"{BasePath}ratings/{fileWork.Name.Insert((int)(fileWork.Name.Length - fileWork.Extension.Length), ".ratings")}";
using(StreamReader reader = File.OpenText(fileWork.FullName))
using(StreamReader readerRatings = File.OpenText(ratingsFile))
{
var work = WorkJsonReader.ReadWork(reader.ReadToEnd(), readerRatings.ReadToEnd());
foreach(var author in work.Authors.ToList())
{
var newAuthor = Authors.SingleOrDefault(a => a.Id == author.Id);
work.Authors.Remove(author);
work.Authors.Add(newAuthor);
}
Works.Add(work);
}
}
foreach(var fileBook in new DirectoryInfo($"{BasePath}books/").GetFiles())
{
using(StreamReader reader = File.OpenText(fileBook.FullName))
{
var book = BookJsonReader.ReadBook(reader.ReadToEnd());
foreach(var author in book.Authors.ToList())
{
var newAuthor = Authors.SingleOrDefault(a => a.Id == author.Id);
book.Authors.Remove(author);
book.Authors.Add(newAuthor);
}
foreach(var work in book.Works.ToList())
{
var newWork = Works.SingleOrDefault(w => w.Id == work.Id);
book.Works.Remove(work);
book.Works.Add(newWork);
}
Books.Add(book);
}
}
}
public Task<AuthorDTO> GetAuthorById(string id)
{
var author = Stub.Authors.SingleOrDefault(a => a.Id == id);
return Task.FromResult(author);
}
private Task<Tuple<long, IEnumerable<AuthorDTO>>> OrderAuthors(IEnumerable<AuthorDTO> authors, int index, int count, string sort = "")
{
switch(sort)
{
case "name":
authors = authors.OrderBy(a => a.Name);
break;
case "name_reverse":
authors = authors.OrderByDescending(a => a.Name);
break;
}
return Task.FromResult(Tuple.Create((long)authors.Count(), authors.Skip(index*count).Take(count)));
}
public async Task<Tuple<long, IEnumerable<AuthorDTO>>> GetAuthors(int index, int count, string sort = "")
{
IEnumerable<AuthorDTO> authors = Stub.Authors;
return await OrderAuthors(authors, index, count, sort);
}
public async Task<Tuple<long, IEnumerable<AuthorDTO>>> GetAuthorsByName(string name, int index, int count, string sort = "")
{
var authors = Stub.Authors.Where(a => a.Name.Contains(name, StringComparison.OrdinalIgnoreCase)
|| a.AlternateNames.Exists(alt => alt.Contains(name, StringComparison.OrdinalIgnoreCase)));
return await OrderAuthors(authors, index, count, sort);
}
public Task<BookDTO> GetBookById(string id)
{
var book = Stub.Books.SingleOrDefault(b => b.Id.Contains(id));
return Task.FromResult(book);
}
private Task<Tuple<long, IEnumerable<BookDTO>>> OrderBooks(IEnumerable<BookDTO> books, int index, int count, string sort = "")
{
switch(sort)
{
case "title":
books = books.OrderBy(b => b.Title);
break;
case "title_reverse":
books = books.OrderByDescending(b => b.Title);
break;
case "new":
books = books.OrderByDescending(b => b.PublishDate);
break;
case "old":
books = books.OrderBy(b => b.PublishDate);
break;
}
return Task.FromResult(Tuple.Create((long)books.Count(), books.Skip(index*count).Take(count)));
}
public async Task<Tuple<long, IEnumerable<BookDTO>>> GetBooks(int index, int count, string sort = "")
{
var books = Stub.Books;
return await OrderBooks(books, index, count, sort);
}
public Task<BookDTO> GetBookByISBN(string isbn)
{
var book = Stub.Books.SingleOrDefault(b => b.ISBN13.Equals(isbn, StringComparison.OrdinalIgnoreCase));
return Task.FromResult(book);
}
public async Task<Tuple<long, IEnumerable<BookDTO>>> GetBooksByTitle(string title, int index, int count, string sort = "")
{
var books = Stub.Books.Where(b => b.Title.Contains(title, StringComparison.OrdinalIgnoreCase)
|| b.Series.Exists(s => s.Contains(title, StringComparison.OrdinalIgnoreCase)));
return await OrderBooks(books, index, count, sort);
}
public async Task<Tuple<long, IEnumerable<BookDTO>>> GetBooksByAuthorId(string authorId, int index, int count, string sort = "")
{
var books = Stub.Books.Where(b => b.Authors.Exists(a => a.Id == authorId)
|| b.Works.Exists(w => w.Authors.Exists(a => a.Id == authorId)));
return await OrderBooks(books, index, count, sort);
}
public async Task<Tuple<long, IEnumerable<BookDTO>>> GetBooksByAuthor(string name, int index, int count, string sort = "")
{
var books = Stub.Books.Where(b => ContainsAuthorName(b, name));
return await OrderBooks(books, index, count, sort);
}
private bool ContainsAuthorName(BookDTO book, string name)
{
IEnumerable<AuthorDTO> authors = new List<AuthorDTO>();
if(book.Authors != null)
{
authors = authors.Union(book.Authors);
}
if(book.Works != null)
{
authors = authors.Union(book.Works.SelectMany(w => w.Authors));
}
foreach(var author in authors)
{
if(author.Name.Contains(name, StringComparison.OrdinalIgnoreCase)
|| author.AlternateNames.Exists(alt => alt.Contains(name, StringComparison.OrdinalIgnoreCase)))
{
return true;
}
}
return false;
}
public Task<Tuple<long, IEnumerable<WorkDTO>>> GetWorks(int index, int count)
{
long nbWorks = Stub.Works.Count;
var works = Stub.Works.Skip(index*count).Take(count);
return Task.FromResult(Tuple.Create(nbWorks, works));
}
public Task<long> GetNbAuthors()
=> Task.FromResult((long)Stub.Authors.Count);
public Task<long> GetNbBooks()
=> Task.FromResult((long)Stub.Books.Count);
public Task<long> GetNbWorks()
=> Task.FromResult((long)Stub.Works.Count);
}

@ -0,0 +1,32 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net7.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
</PropertyGroup>
<ItemGroup>
<ProjectReference Include="..\LibraryDTO\LibraryDTO.csproj" />
<ProjectReference Include="..\DtoAbstractLayer\DtoAbstractLayer.csproj" />
<ProjectReference Include="..\JsonReader\JsonReader.csproj" />
</ItemGroup>
<ItemGroup>
<None Remove="books\" />
<None Remove="authors\" />
<None Remove="works\" />
<None Remove="ratings\" />
</ItemGroup>
<ItemGroup>
<Folder Include="books\" />
<Folder Include="authors\" />
<Folder Include="works\" />
<Folder Include="ratings\" />
</ItemGroup>
<ItemGroup>
<None Include="*\*.json">
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
<CopyToPublishDirectory>PreserveNewest</CopyToPublishDirectory>
</None>
</ItemGroup>
</Project>

@ -0,0 +1 @@
{"name": "Michel Demuth", "personal_name": "Michel Demuth", "last_modified": {"type": "/type/datetime", "value": "2008-08-26 02:41:15.604911"}, "key": "/authors/OL1846639A", "type": {"key": "/type/author"}, "id": 6527877, "revision": 2}

@ -0,0 +1 @@
{"personal_name": "Dick, Philip K.", "source_records": ["amazon:8445007327", "bwb:9780722129562", "amazon:0792776232", "ia:pacificpark0000dick", "amazon:2277213799", "amazon:2266163019", "bwb:9798599263227", "amazon:1433276712", "ia:ejonescreoilmond0000dick", "amazon:6051719164", "amazon:6254493632", "amazon:2277117749", "amazon:1987781619", "amazon:1433248239", "amazon:1480594407"], "alternate_names": ["Philip Kindred Dick", "Philip Dick", "Philip Kendred Dick", "Philip K Dick"], "bio": "Philip Kindred Dick was an American novelist, short story writer, and essayist whose published work during his lifetime was almost entirely in the science fiction genre. Dick explored sociological, political and metaphysical themes in novels dominated by monopolistic corporations, authoritarian governments, and altered states. In his later works, Dick's thematic focus strongly reflected his personal interest in metaphysics and theology. He often drew upon his own life experiences and addressed the nature of drug abuse, paranoia and schizophrenia, and transcendental experiences in novels such as A Scanner Darkly and VALIS.\r\n\r\nSource and more information: [Wikipedia (EN)](http://en.wikipedia.org/wiki/Philip_K._Dick)", "type": {"key": "/type/author"}, "death_date": "2 March 1982", "remote_ids": {"isni": "0000000121251093", "wikidata": "Q171091", "viaf": "27063583"}, "name": "Philip K. Dick", "links": [{"title": "Wikipedia link to Philip K Dick", "url": "http://en.wikipedia.org/wiki/Philip_K._Dick", "type": {"key": "/type/link"}}], "photos": [6295259], "birth_date": "16 December 1928", "key": "/authors/OL274606A", "latest_revision": 23, "revision": 23, "created": {"type": "/type/datetime", "value": "2008-04-01T03:28:50.625462"}, "last_modified": {"type": "/type/datetime", "value": "2022-11-29T21:21:41.951561"}}

@ -0,0 +1 @@
{"name": "H\u00e9l\u00e8ne Collon", "last_modified": {"type": "/type/datetime", "value": "2008-04-30 08:14:56.482104"}, "key": "/authors/OL3113922A", "type": {"key": "/type/author"}, "id": 11970257, "revision": 1}

@ -0,0 +1 @@
{"name": "Alain Damasio", "key": "/authors/OL3980331A", "type": {"key": "/type/author"}, "remote_ids": {"wikidata": "Q2829704"}, "birth_date": "1 August 1969", "latest_revision": 2, "revision": 2, "created": {"type": "/type/datetime", "value": "2008-04-30T20:50:18.033121"}, "last_modified": {"type": "/type/datetime", "value": "2022-12-19T19:05:32.693708"}}

@ -0,0 +1 @@
{"personal_name": "James S. A. Corey", "remote_ids": {"isni": "0000000382626033", "viaf": "266413968", "wikidata": "Q6142591"}, "source_records": ["amazon:1478933771", "amazon:1528822218", "amazon:1456121650", "bwb:9780356510385", "amazon:0678452547", "bwb:9780356517773"], "alternate_names": ["Daniel Abraham", "Ty Franck", "James S.A. Corey", "James James S. A. Corey"], "type": {"key": "/type/author"}, "key": "/authors/OL6982995A", "entity_type": "org", "links": [{"title": "Source", "url": "http://www.danielabraham.com/james-s-a-corey/", "type": {"key": "/type/link"}}], "bio": {"type": "/type/text", "value": "James S.A. Corey is the pen name used by collaborators [Daniel Abraham](https://openlibrary.org/authors/OL1427729A/Daniel_Abraham) and [Ty Franck](https://openlibrary.org/authors/OL7523472A/Ty_Franck).\r\n\r\nThe first and last name are taken from Abraham's and Franck's middle names, respectively, and S.A. are the initials of Abraham's daughter."}, "photos": [11112303], "name": "James S. A. Corey", "latest_revision": 13, "revision": 13, "created": {"type": "/type/datetime", "value": "2011-10-20T08:06:05.906616"}, "last_modified": {"type": "/type/datetime", "value": "2023-05-18T18:14:26.659278"}}

@ -0,0 +1 @@
{"key": "/authors/OL7475792A", "name": "Ada Palmer", "type": {"key": "/type/author"}, "latest_revision": 4, "revision": 4, "created": {"type": "/type/datetime", "value": "2019-03-11T19:38:25.579004"}, "last_modified": {"type": "/type/datetime", "value": "2021-12-07T07:11:29.213401"}}

@ -0,0 +1 @@
{"type": {"key": "/type/author"}, "name": "Frank Herbert", "key": "/authors/OL9956442A", "source_records": ["amazon:2221252306"], "latest_revision": 1, "revision": 1, "created": {"type": "/type/datetime", "value": "2021-11-14T17:07:35.515652"}, "last_modified": {"type": "/type/datetime", "value": "2021-11-14T17:07:35.515652"}}

@ -0,0 +1 @@
{"publishers": ["Actes Sud"], "title": "L'\u00c9veil du L\u00e9viathan", "number_of_pages": 624, "isbn_13": ["9782330033118"], "covers": [7412481], "languages": [{"key": "/languages/fre"}], "publish_date": "4 juin 2014", "key": "/books/OL25910297M", "publish_places": ["France"], "works": [{"key": "/works/OL17334140W"}], "type": {"key": "/type/edition"}, "source_records": ["amazon:2330033117"], "latest_revision": 5, "revision": 5, "created": {"type": "/type/datetime", "value": "2016-04-22T11:47:01.838591"}, "last_modified": {"type": "/type/datetime", "value": "2023-02-02T01:19:11.921173"}}

@ -0,0 +1 @@
{"works": [{"key": "/works/OL19635836W"}], "title": "La Volont\u00e9 de se battre", "publishers": ["Le B\u00e9lial'"], "publish_date": "2020", "key": "/books/OL35698083M", "type": {"key": "/type/edition"}, "identifiers": {}, "covers": [12392970], "isbn_13": ["9782843449758"], "classifications": {}, "languages": [{"key": "/languages/fre"}], "contributors": [{"name": "Michelle Charrier", "role": "Translator"}], "number_of_pages": 526, "series": ["Terra Ignota #3"], "physical_format": "paperback", "latest_revision": 3, "revision": 3, "created": {"type": "/type/datetime", "value": "2021-12-07T02:23:07.593997"}, "last_modified": {"type": "/type/datetime", "value": "2021-12-07T02:24:57.135563"}}

@ -0,0 +1 @@
{"type": {"key": "/type/edition"}, "title": "La Zone du Dehors", "authors": [{"key": "/authors/OL3980331A"}], "publish_date": "Feb 04, 2021", "source_records": ["amazon:2072927528"], "number_of_pages": 656, "publishers": ["FOLIO", "GALLIMARD"], "isbn_10": ["2072927528"], "isbn_13": ["9782072927522"], "physical_format": "pocket book", "full_title": "La Zone du Dehors", "covers": [12393645], "works": [{"key": "/works/OL19960903W"}], "key": "/books/OL35699439M", "latest_revision": 1, "revision": 1, "created": {"type": "/type/datetime", "value": "2021-12-07T22:26:13.534930"}, "last_modified": {"type": "/type/datetime", "value": "2021-12-07T22:26:13.534930"}}

@ -0,0 +1 @@
{"type": {"key": "/type/edition"}, "title": "Dune - tome 1", "authors": [{"key": "/authors/OL9956442A"}, {"key": "/authors/OL1846639A"}], "publish_date": "Nov 22, 2012", "source_records": ["amazon:2266233203"], "number_of_pages": 832, "publishers": ["POCKET", "Pocket"], "isbn_10": ["2266233203"], "isbn_13": ["9782266233200"], "physical_format": "pocket book", "notes": {"type": "/type/text", "value": "Source title: Dune - tome 1 (1)"}, "works": [{"key": "/works/OL27962193W"}], "key": "/books/OL38218739M", "latest_revision": 1, "revision": 1, "created": {"type": "/type/datetime", "value": "2022-05-30T17:18:00.228322"}, "last_modified": {"type": "/type/datetime", "value": "2022-05-30T17:18:00.228322"}}

@ -0,0 +1 @@
{"type": {"key": "/type/edition"}, "title": "Total Recall et autres r\u00e9cits", "authors": [{"key": "/authors/OL274606A"}, {"key": "/authors/OL3113922A"}], "publish_date": "Jul 12, 2012", "source_records": ["amazon:2070448908"], "number_of_pages": 448, "publishers": ["FOLIO", "GALLIMARD"], "isbn_10": ["2070448908"], "isbn_13": ["9782070448906"], "physical_format": "pocket book", "works": [{"key": "/works/OL28185064W"}], "key": "/books/OL38586212M", "covers": [13858141], "latest_revision": 3, "revision": 3, "created": {"type": "/type/datetime", "value": "2022-07-10T01:29:29.296699"}, "last_modified": {"type": "/type/datetime", "value": "2023-04-07T22:44:13.567567"}}

@ -0,0 +1 @@
{"summary": {"average": null, "count": 0}, "counts": {"1": 0, "2": 0, "3": 0, "4": 0, "5": 0}}

@ -0,0 +1 @@
{"summary": {"average": 4.8, "count": 5, "sortable": 3.216059213089321}, "counts": {"1": 0, "2": 0, "3": 0, "4": 1, "5": 4}}

@ -0,0 +1 @@
{"summary": {"average": 4.0, "count": 1, "sortable": 2.3286737413641063}, "counts": {"1": 0, "2": 0, "3": 0, "4": 1, "5": 0}}

@ -0,0 +1 @@
{"summary": {"average": 3.0, "count": 1, "sortable": 2.19488243981746}, "counts": {"1": 0, "2": 0, "3": 1, "4": 0, "5": 0}}

@ -0,0 +1 @@
{"summary": {"average": null, "count": 0}, "counts": {"1": 0, "2": 0, "3": 0, "4": 0, "5": 0}}

@ -0,0 +1 @@
{"title": "L'\u00c9veil du L\u00e9viathan", "key": "/works/OL17334140W", "authors": [{"type": {"key": "/type/author_role"}, "author": {"key": "/authors/OL6982995A"}}], "type": {"key": "/type/work"}, "covers": [7412481], "latest_revision": 3, "revision": 3, "created": {"type": "/type/datetime", "value": "2016-04-22T11:47:01.838591"}, "last_modified": {"type": "/type/datetime", "value": "2023-02-02T01:19:11.921173"}}

@ -0,0 +1 @@
{"description": "\"The long years of near-utopia have come to an abrupt end. Peace and order are now figments of the past. Corruption, deception, and insurgency hum within the once steadfast leadership of the Hives, nations without fixed location. The heartbreaking truth is that for decades, even centuries, the leaders of the great Hives bought the world's stability with a trickle of secret murders, mathematically planned. So that no faction could ever dominate. So that the balance held. The Hives' fa\u00e7ade of solidity is the only hope they have for maintaining a semblance of order, for preventing the public from succumbing to the savagery and bloodlust of wars past. But as the great secret becomes more and more widely known, that fa\u00e7ade is slipping away. Just days earlier, the world was a pinnacle of human civilization. Now everyone--Hives and hiveless, Utopians and sensayers, emperors and the downtrodden, warriors and saints--scrambles to prepare for the seemingly inevitable war\"--", "covers": [8544084, 8619055, 10180814], "key": "/works/OL19635836W", "authors": [{"author": {"key": "/authors/OL7475792A"}, "type": {"key": "/type/author_role"}}], "title": "The Will to Battle", "subjects": ["Utopias", "Fiction", "Fiction, science fiction, general", "series:terra_ignota"], "type": {"key": "/type/work"}, "latest_revision": 7, "revision": 7, "created": {"type": "/type/datetime", "value": "2019-04-21T08:07:12.674468"}, "last_modified": {"type": "/type/datetime", "value": "2021-12-07T07:08:28.885088"}}

@ -0,0 +1 @@
{"title": "La zone du dehors", "key": "/works/OL19960903W", "authors": [{"type": {"key": "/type/author_role"}, "author": {"key": "/authors/OL3980331A"}}], "type": {"key": "/type/work"}, "covers": [13472433], "latest_revision": 2, "revision": 2, "created": {"type": "/type/datetime", "value": "2019-07-17T23:01:16.580404"}, "last_modified": {"type": "/type/datetime", "value": "2023-03-15T21:59:10.897047"}}

@ -0,0 +1 @@
{"type": {"key": "/type/work"}, "title": "Dune - tome 1", "authors": [{"type": {"key": "/type/author_role"}, "author": {"key": "/authors/OL9956442A"}}, {"type": {"key": "/type/author_role"}, "author": {"key": "/authors/OL1846639A"}}], "key": "/works/OL27962193W", "covers": [13823878], "latest_revision": 2, "revision": 2, "created": {"type": "/type/datetime", "value": "2022-05-30T17:18:00.228322"}, "last_modified": {"type": "/type/datetime", "value": "2023-04-04T09:05:11.531979"}}

@ -0,0 +1 @@
{"type": {"key": "/type/work"}, "title": "Total Recall et autres r\u00e9cits", "authors": [{"type": {"key": "/type/author_role"}, "author": {"key": "/authors/OL274606A"}}, {"type": {"key": "/type/author_role"}, "author": {"key": "/authors/OL3113922A"}}], "key": "/works/OL28185064W", "covers": [13858141], "latest_revision": 3, "revision": 3, "created": {"type": "/type/datetime", "value": "2022-07-10T01:29:29.296699"}, "last_modified": {"type": "/type/datetime", "value": "2023-04-07T22:44:13.567567"}}

Binary file not shown.

After

Width:  |  Height:  |  Size: 170 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 35 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 254 KiB

Loading…
Cancel
Save