mirror of
				https://github.com/Kareadita/Kavita.git
				synced 2025-10-26 08:12:28 -04:00 
			
		
		
		
	* WebP Covers + Series Detail Enhancements (#1652) * Implemented save covers as webp. Reworked screen to provide more information up front about webp and what browsers can support it. * cleaned up pages to use compact numbering and made compact numbering expand into one decimal place (20.5K) * Fixed an issue with adding new device * If a book has an invalid language set, drop the language altogether rather than reading in a corrupted entry. * Ensure genres and tags render alphabetically. Improved support for partial volumes in Comic parser. * Ensure all people, tags, collections, and genres are in alphabetical order. * Moved some code to Extensions to clean up code. * More unit tests * Cleaned up release year filter css * Tweaked some code in all series to make bulk deletes cleaner on the UI. * Trying out want to read and unread count on series detail page * Added Want to Read button for series page to make it easy to see when something is in want to read list and toggle it. Added tooltips instead of title to buttons, but they don't style correctly. Added a continue point under cover image. * Code smells * Bump versions by dotnet-bump-version. * Fixed Series Relations Schema (#1654) * Bump loader-utils from 2.0.2 to 2.0.3 in /UI/Web Bumps [loader-utils](https://github.com/webpack/loader-utils) from 2.0.2 to 2.0.3. - [Release notes](https://github.com/webpack/loader-utils/releases) - [Changelog](https://github.com/webpack/loader-utils/blob/v2.0.3/CHANGELOG.md) - [Commits](https://github.com/webpack/loader-utils/compare/v2.0.2...v2.0.3) --- updated-dependencies: - dependency-name: loader-utils dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Fixed is want to read coming back as a string and not working correctly. * Changed from to Continue to be more explicit * Added the first migration which exports data as a csv in temp/. This is the backup in case data is lost in the migration. * Note for later * Fixed the migration for the series relation so when deleting any series on any edge of the relationship, the SeriesRelation row deletes. * Change buttons back to titles on series detail page * Wrote the code to import relations from the backup. * Added an additional version check to avoid file io on migration. * Code cleanup Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * Fresh Nightly Installs Work (#1659) * Bump loader-utils from 2.0.3 to 2.0.4 in /UI/Web Bumps [loader-utils](https://github.com/webpack/loader-utils) from 2.0.3 to 2.0.4. - [Release notes](https://github.com/webpack/loader-utils/releases) - [Changelog](https://github.com/webpack/loader-utils/blob/v2.0.4/CHANGELOG.md) - [Commits](https://github.com/webpack/loader-utils/compare/v2.0.3...v2.0.4) --- updated-dependencies: - dependency-name: loader-utils dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Removed a very old cgecj from Nov 2021 when data/appsettings was moved to config/ * Added some notes about migration * Removed a file that shouldn't have been there. Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * Library Settings Modal + New Library Settings (#1660) * Bump loader-utils from 2.0.3 to 2.0.4 in /UI/Web Bumps [loader-utils](https://github.com/webpack/loader-utils) from 2.0.3 to 2.0.4. - [Release notes](https://github.com/webpack/loader-utils/releases) - [Changelog](https://github.com/webpack/loader-utils/blob/v2.0.4/CHANGELOG.md) - [Commits](https://github.com/webpack/loader-utils/compare/v2.0.3...v2.0.4) --- updated-dependencies: - dependency-name: loader-utils dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Fixed want to read button on series detail not performing the correct action * Started the library settings. Added ability to update a cover image for a library. Updated backup db to also copy reading list (and now library) cover images. * Integrated Edit Library into new settings (not tested) and hooked up a wizard-like flow for new library. * Fixed a missing update event in backend when updating a library. * Disable Save when form invalid. Do inline validation on Library name when user types to ensure the name is valid. * Trim library names before you check anything * General code cleanup * Implemented advanced settings for library (include in dashboard, search, recommended) and ability to turn off folder watching for individual libraries. Refactored some code to streamline perf in some flows. * Removed old components replaced with new modal * Code smells Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * UX Alignment and bugfixes (#1663) * Refactored the design of reading list page to follow more in line with list view. Added release date on the reading list items, if it's set in underlying chapter. Fixed a bug where reordering the list items could sometimes not update correctly with drag and drop. * Removed a bug marker that I just fixed * When generating library covers, make them much smaller as they are only ever icons. * Fixed library settings not showing the correct image. * Fixed a bug where duplicate collection tags could be created. Fixed a bug where collection tag normalized title was being set to uppercase. Redesigned the edit collection tag modal to align with new library settings and provide inline name checks. * Updated edit reading list modal to align with new library settings modal pattern. Refactored the backend to ensure it flows correctly without allowing duplicate names. Don't show Continue point on series detail if the whole series is read. * Added some more unit tests around continue point * Fixed a bug on series detail when bulk selecting between volume and chapters, the code which determines which chapters are selected didn't take into account mixed layout for Storyline tab. * Refactored to generate an OpenAPI spec at root of Kavita. This will be loaded by a new API site for easy hosting. Deprecated EnableSwaggerUi preference as after validation new system works, this will be removed and instances can use our hosting to hit their server (or run a debug build). * Test GA * Reverted GA and instead do it in the build step. This will just force developers to commit it in. * GA please work * Removed redundant steps from test since build already does it. * Try another GA * Moved all test actions into initial build step, which should drastically cut down on time. Only run sonar if the secret is present (so not for forks). Updated build requirements for develop and stable docker pushes. * Fixed env variable * Okay not possible to do secrets in if statement * Fixed the build step to output the openapi.json where it's expected. * Fixed GA (#1664) * Bump versions by dotnet-bump-version. * Applied new _components layout structure to Kavita. All except manga as there is an open PR that drastically changes that module. (#1666) * [Experimental] Split Renderers - Double & Double (Manga) fixes (#1667) * Updated swiper and some packages for reported security issues * Fixed reading lists promotion not working * Refactor RenameFileForCopy to use iterative recursion, rather than functional. * Ensured that bookmarks are fetched and ordered by Created date. * Fixed a bug where bookmarks were coming back in the correct order, but due to filenames, would not sort correctly. * Default installs to Debug log level given errors users have and Debug not being too noisy * Added jumpbar to bookmarks page * Now added jumpbar to bookmarks * Refactored some code into pipes and added some debug messaging for prefetcher * Try loading next and prev chapter's first/last page to cache so it renders faster * Updated GetImage to do a bound check on max page. Fixed a critical bug in how manga reader updates image elements src to prefetch/load pages. I was not creating a new reference which broke Angular's ability to update DOM on changes. * Refactored the image setting code to use a single method which tries to use a cached image always. * Refactored code to use getPage which favors cache and simplifies image creation code * Started the work to split the canvas renderer into it's own component * Refactored a lot of common methods into a service for the reader to support the upcoming renderer split * Moved components to nested folder. Refactored more code to streamline image sending to child renderer. Added notes across the code to help streamline flow of data and who owns what. * Swapped out SQLite for Memory, but the one from hangfire. Added DisableConcurrentExecution on ProcessChange to avoid duplication when multiple threads execute at once. * Basic split right to left is working with canvas renderer * Left to right and right to left now work * Fixed a bug where pagesplitoption wasn't being updated when modifying menu * Canvas rendering still has a bug with switching between right to left -> left to right on the re-render, it will choose a bad state. All else works fine with it. * Updated canvas renderer to implement the ImageRenderer interface * Canvas renderer is done * Setup single renderer. Need to figure out how to share CSS between renderers and also share some global stuff, like image height. * Refactored code so that image-container is within the renderers themselves. Still broken in scaling, but working towards a solution. * Added double click to shortcut menu * Moved image containers within the renderers * Pushing up for Robbie * nothing new * Move common css to a single scss file * More css consolidation * Fixed a npe in isWideImage * Refactored page updates to renderers to include max pages. Rewrote most of renderer into observables. * Moved bookmark for second page to double renderer * Started hooking in double renderer renderPage() * Fixed height scaling, but now canvas renderer is broken again * Fixed a bug with canvas renderer not moving to next page. Streamlined the code for getting page amounts from the dfferent renderers * Added double click to bookmark for canvas * Stashing the code and taking a break * Nothing much, buffer is still broken * Got double renderer to render at least one page * Double renderer now has access to 5 images at any time, so it can make appropriate decisions on when to render double pages. * Fixed up double rendererer moving backward page calc * Forward logic seems to be working * Cleaned up dead code after testing * Moved a few loggers in folder watching to trace * Everything seems to work fine, time to do double manga renderer * Moved some css around and added the reverse double component * Only execute renderer's pipes when in the correct mode * Still working on double renderer * Fixed scaling issues on double * Updating double logic - Fixed: Fixed an issue where a second page would render when current page was wide. * Hooked up double renderer * Made changes but not sure if im making progress * double manga fixes * Claned some of robbies code * Fixing last page bug * Library Settings Modal + New Library Settings (#1660) * Bump loader-utils from 2.0.3 to 2.0.4 in /UI/Web Bumps [loader-utils](https://github.com/webpack/loader-utils) from 2.0.3 to 2.0.4. - [Release notes](https://github.com/webpack/loader-utils/releases) - [Changelog](https://github.com/webpack/loader-utils/blob/v2.0.4/CHANGELOG.md) - [Commits](https://github.com/webpack/loader-utils/compare/v2.0.3...v2.0.4) --- updated-dependencies: - dependency-name: loader-utils dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Fixed want to read button on series detail not performing the correct action * Started the library settings. Added ability to update a cover image for a library. Updated backup db to also copy reading list (and now library) cover images. * Integrated Edit Library into new settings (not tested) and hooked up a wizard-like flow for new library. * Fixed a missing update event in backend when updating a library. * Disable Save when form invalid. Do inline validation on Library name when user types to ensure the name is valid. * Trim library names before you check anything * General code cleanup * Implemented advanced settings for library (include in dashboard, search, recommended) and ability to turn off folder watching for individual libraries. Refactored some code to streamline perf in some flows. * Removed old components replaced with new modal * Code smells Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * UX Alignment and bugfixes (#1663) * Refactored the design of reading list page to follow more in line with list view. Added release date on the reading list items, if it's set in underlying chapter. Fixed a bug where reordering the list items could sometimes not update correctly with drag and drop. * Removed a bug marker that I just fixed * When generating library covers, make them much smaller as they are only ever icons. * Fixed library settings not showing the correct image. * Fixed a bug where duplicate collection tags could be created. Fixed a bug where collection tag normalized title was being set to uppercase. Redesigned the edit collection tag modal to align with new library settings and provide inline name checks. * Updated edit reading list modal to align with new library settings modal pattern. Refactored the backend to ensure it flows correctly without allowing duplicate names. Don't show Continue point on series detail if the whole series is read. * Added some more unit tests around continue point * Fixed a bug on series detail when bulk selecting between volume and chapters, the code which determines which chapters are selected didn't take into account mixed layout for Storyline tab. * Refactored to generate an OpenAPI spec at root of Kavita. This will be loaded by a new API site for easy hosting. Deprecated EnableSwaggerUi preference as after validation new system works, this will be removed and instances can use our hosting to hit their server (or run a debug build). * Test GA * Reverted GA and instead do it in the build step. This will just force developers to commit it in. * GA please work * Removed redundant steps from test since build already does it. * Try another GA * Moved all test actions into initial build step, which should drastically cut down on time. Only run sonar if the secret is present (so not for forks). Updated build requirements for develop and stable docker pushes. * Fixed env variable * Okay not possible to do secrets in if statement * Fixed the build step to output the openapi.json where it's expected. * Fixed GA (#1664) * Bump versions by dotnet-bump-version. * Applied new _components layout structure to Kavita. All except manga as there is an open PR that drastically changes that module. (#1666) * Fixed typeahead and updated manga reader to new layout structure * Fixed book reader fonts lookups * Fixed up some build issues * Fixed a bad import of css image * Some cleanup and rewrote how we log out data. * Renderer can be null on first load when performing some work. * Library Settings Modal + New Library Settings (#1660) * Bump loader-utils from 2.0.3 to 2.0.4 in /UI/Web Bumps [loader-utils](https://github.com/webpack/loader-utils) from 2.0.3 to 2.0.4. - [Release notes](https://github.com/webpack/loader-utils/releases) - [Changelog](https://github.com/webpack/loader-utils/blob/v2.0.4/CHANGELOG.md) - [Commits](https://github.com/webpack/loader-utils/compare/v2.0.3...v2.0.4) --- updated-dependencies: - dependency-name: loader-utils dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Fixed want to read button on series detail not performing the correct action * Started the library settings. Added ability to update a cover image for a library. Updated backup db to also copy reading list (and now library) cover images. * Integrated Edit Library into new settings (not tested) and hooked up a wizard-like flow for new library. * Fixed a missing update event in backend when updating a library. * Disable Save when form invalid. Do inline validation on Library name when user types to ensure the name is valid. * Trim library names before you check anything * General code cleanup * Implemented advanced settings for library (include in dashboard, search, recommended) and ability to turn off folder watching for individual libraries. Refactored some code to streamline perf in some flows. * Removed old components replaced with new modal * Code smells Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * UX Alignment and bugfixes (#1663) * Refactored the design of reading list page to follow more in line with list view. Added release date on the reading list items, if it's set in underlying chapter. Fixed a bug where reordering the list items could sometimes not update correctly with drag and drop. * Removed a bug marker that I just fixed * When generating library covers, make them much smaller as they are only ever icons. * Fixed library settings not showing the correct image. * Fixed a bug where duplicate collection tags could be created. Fixed a bug where collection tag normalized title was being set to uppercase. Redesigned the edit collection tag modal to align with new library settings and provide inline name checks. * Updated edit reading list modal to align with new library settings modal pattern. Refactored the backend to ensure it flows correctly without allowing duplicate names. Don't show Continue point on series detail if the whole series is read. * Added some more unit tests around continue point * Fixed a bug on series detail when bulk selecting between volume and chapters, the code which determines which chapters are selected didn't take into account mixed layout for Storyline tab. * Refactored to generate an OpenAPI spec at root of Kavita. This will be loaded by a new API site for easy hosting. Deprecated EnableSwaggerUi preference as after validation new system works, this will be removed and instances can use our hosting to hit their server (or run a debug build). * Test GA * Reverted GA and instead do it in the build step. This will just force developers to commit it in. * GA please work * Removed redundant steps from test since build already does it. * Try another GA * Moved all test actions into initial build step, which should drastically cut down on time. Only run sonar if the secret is present (so not for forks). Updated build requirements for develop and stable docker pushes. * Fixed env variable * Okay not possible to do secrets in if statement * Fixed the build step to output the openapi.json where it's expected. * Applied new _components layout structure to Kavita. All except manga as there is an open PR that drastically changes that module. (#1666) * Post merge cleanup * Again moving the file Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * Basic Stats (#1673) * Refactored ResponseCache profiles into consts * Refactored code to use an extension method for getting user library ids. * Started server statistics, added a charting library, and added a table sort column (not finished) * Refactored code and have a fully working example of sortable headers. Still doesn't work with default sorting state, will work on that later. * Implemented file size, but it's too expensive, so commented out. * Added a migration to provide extension and length/size information in the DB to allow for faster stat apis. * Added the ability to force a library scan from library settings. * Refactored some apis to provide more of a file breakdown rather than just file size. * Working on visualization of file breakdown * Fixed the file breakdown visual * Fixed up 2 visualizations * Added back an api for member names, started work on top reads * Hooked up the other library types and username/days. * Preparing to remove top reads and refactor into Top users * Added LibraryId to AppUserProgress to help with complex lookups. * Added the new libraryId hook into some stats methods * Updated api methods to use libraryId for progress * More places where LibraryId is needed * Added some high level server stats * Got a ton done on server stats * Updated default theme (dark) to be the default root variables. This will allow user themes to override just what they want, rather than maintain their own css variables. * Implemented a monster query for top users by reading time. It's very slow and can be cleaned up likely. * Hooked up top reads. Code needs a big refactor. Handing off for Robbie treatment and I'll switch to User stats. * Implemented last 5 recently read series (broken) and added some basic css * Fixed recently read query * Cleanup the css a bit, Robbie we need you * More css love * Cleaned up DTOs that aren't needed anymore * Fixed top readers query * When calculating top readers, don't include read events where nothing is read (0 pages) * Hooked up the date into GetTopUsers * Hooked top readers up with days and refactored and cleaned up componets not used * Fixed up query * Started on a day by day breakdown, but going to take a break from stats. * Added a temp task to run some migration manually for stats to work * Ensure OPDS-PS uses new libraryId for progress reporting * Fixed a code smell * Adding some styling * adding more styles * Removed some debug stuff from user stats * Bump qs from 6.5.2 to 6.5.3 in /UI/Web Bumps [qs](https://github.com/ljharb/qs) from 6.5.2 to 6.5.3. - [Release notes](https://github.com/ljharb/qs/releases) - [Changelog](https://github.com/ljharb/qs/blob/main/CHANGELOG.md) - [Commits](https://github.com/ljharb/qs/compare/v6.5.2...v6.5.3) --- updated-dependencies: - dependency-name: qs dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Tweaked some code for bad data cases * Refactored a chapter lookup to remove un-needed Volume join in 5 places across the code. * API push Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * Hooked up the API layer to be able to extract images from PDF again for Tachiyomi explicitly (#1686) * Bump versions by dotnet-bump-version. * OPDS Enhancements (#1687) * Bump express from 4.17.2 to 4.18.2 in /UI/Web Bumps [express](https://github.com/expressjs/express) from 4.17.2 to 4.18.2. - [Release notes](https://github.com/expressjs/express/releases) - [Changelog](https://github.com/expressjs/express/blob/master/History.md) - [Commits](https://github.com/expressjs/express/compare/4.17.2...4.18.2) --- updated-dependencies: - dependency-name: express dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Bump decode-uri-component from 0.2.0 to 0.2.2 in /UI/Web Bumps [decode-uri-component](https://github.com/SamVerschueren/decode-uri-component) from 0.2.0 to 0.2.2. - [Release notes](https://github.com/SamVerschueren/decode-uri-component/releases) - [Commits](https://github.com/SamVerschueren/decode-uri-component/compare/v0.2.0...v0.2.2) --- updated-dependencies: - dependency-name: decode-uri-component dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Bump qs and express in /UI/Web Bumps [qs](https://github.com/ljharb/qs) and [express](https://github.com/expressjs/express). These dependencies needed to be updated together. Updates `qs` from 6.5.3 to 6.11.0 - [Release notes](https://github.com/ljharb/qs/releases) - [Changelog](https://github.com/ljharb/qs/blob/main/CHANGELOG.md) - [Commits](https://github.com/ljharb/qs/compare/v6.5.3...v6.11.0) Updates `express` from 4.17.2 to 4.18.2 - [Release notes](https://github.com/expressjs/express/releases) - [Changelog](https://github.com/expressjs/express/blob/master/History.md) - [Commits](https://github.com/expressjs/express/compare/4.17.2...4.18.2) --- updated-dependencies: - dependency-name: qs dependency-type: indirect - dependency-name: express dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Added genre and authors to Series level, added summary to volume and chapter level. Force order on reading list title as Chunky enforces their own sort order and doesn't respect the spec. * Moved all the reading list formatting logic to the backend. This allows us to re-use the UI logic for OPDS streams. * Fixed a broken unit test * Code smells Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * Epub Table of Generation fixes for Sigil (#1689) * Fixed generating table of contents where key lookup could fail with how Sigil packs the epubs. * Tweaked Kavita's fallback ToC generation (when one doesn't exist in the epub) to also use CoalesceKey. * Code smells * Bump versions by dotnet-bump-version. * File Dimension API (#1690) * Implemented an api for getting file dimensions for a given chapter. This is for CDisplayEx integration. This might be usable in Double Renderer. * Added the cached filename for new API * Bump versions by dotnet-bump-version. * Send Non books to your Devices (#1691) * Only restrict non-epub/pdf for Kindle devices on Send To. * Removed restriction to email non-epub/pdfs to devices. * Bump versions by dotnet-bump-version. * Misc UI Tweaks (#1692) * Added a timeAgo pipe which shows live updates for a few areas. * Fixed some wording on stats page. Changed Total People count to just work on distinct names and not count multiple for different roles. * Tweaked the compact number so it only shows one decimal * Fixed a bug * Bump versions by dotnet-bump-version. * Reader Refactor Part 2 (#1694) * Updated swiper and some packages for reported security issues * Fixed reading lists promotion not working * Refactor RenameFileForCopy to use iterative recursion, rather than functional. * Ensured that bookmarks are fetched and ordered by Created date. * Fixed a bug where bookmarks were coming back in the correct order, but due to filenames, would not sort correctly. * Default installs to Debug log level given errors users have and Debug not being too noisy * Added jumpbar to bookmarks page * Now added jumpbar to bookmarks * Refactored some code into pipes and added some debug messaging for prefetcher * Try loading next and prev chapter's first/last page to cache so it renders faster * Updated GetImage to do a bound check on max page. Fixed a critical bug in how manga reader updates image elements src to prefetch/load pages. I was not creating a new reference which broke Angular's ability to update DOM on changes. * Refactored the image setting code to use a single method which tries to use a cached image always. * Refactored code to use getPage which favors cache and simplifies image creation code * Started the work to split the canvas renderer into it's own component * Refactored a lot of common methods into a service for the reader to support the upcoming renderer split * Moved components to nested folder. Refactored more code to streamline image sending to child renderer. Added notes across the code to help streamline flow of data and who owns what. * Swapped out SQLite for Memory, but the one from hangfire. Added DisableConcurrentExecution on ProcessChange to avoid duplication when multiple threads execute at once. * Basic split right to left is working with canvas renderer * Left to right and right to left now work * Fixed a bug where pagesplitoption wasn't being updated when modifying menu * Canvas rendering still has a bug with switching between right to left -> left to right on the re-render, it will choose a bad state. All else works fine with it. * Updated canvas renderer to implement the ImageRenderer interface * Canvas renderer is done * Setup single renderer. Need to figure out how to share CSS between renderers and also share some global stuff, like image height. * Refactored code so that image-container is within the renderers themselves. Still broken in scaling, but working towards a solution. * Added double click to shortcut menu * Moved image containers within the renderers * Pushing up for Robbie * nothing new * Move common css to a single scss file * More css consolidation * Fixed a npe in isWideImage * Refactored page updates to renderers to include max pages. Rewrote most of renderer into observables. * Moved bookmark for second page to double renderer * Started hooking in double renderer renderPage() * Fixed height scaling, but now canvas renderer is broken again * Fixed a bug with canvas renderer not moving to next page. Streamlined the code for getting page amounts from the dfferent renderers * Added double click to bookmark for canvas * Stashing the code and taking a break * Nothing much, buffer is still broken * Got double renderer to render at least one page * Double renderer now has access to 5 images at any time, so it can make appropriate decisions on when to render double pages. * Fixed up double rendererer moving backward page calc * Forward logic seems to be working * Cleaned up dead code after testing * Moved a few loggers in folder watching to trace * Everything seems to work fine, time to do double manga renderer * Moved some css around and added the reverse double component * Only execute renderer's pipes when in the correct mode * Still working on double renderer * Fixed scaling issues on double * Updating double logic - Fixed: Fixed an issue where a second page would render when current page was wide. * Hooked up double renderer * Made changes but not sure if im making progress * double manga fixes * Claned some of robbies code * Fixing last page bug * Library Settings Modal + New Library Settings (#1660) * Bump loader-utils from 2.0.3 to 2.0.4 in /UI/Web Bumps [loader-utils](https://github.com/webpack/loader-utils) from 2.0.3 to 2.0.4. - [Release notes](https://github.com/webpack/loader-utils/releases) - [Changelog](https://github.com/webpack/loader-utils/blob/v2.0.4/CHANGELOG.md) - [Commits](https://github.com/webpack/loader-utils/compare/v2.0.3...v2.0.4) --- updated-dependencies: - dependency-name: loader-utils dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Fixed want to read button on series detail not performing the correct action * Started the library settings. Added ability to update a cover image for a library. Updated backup db to also copy reading list (and now library) cover images. * Integrated Edit Library into new settings (not tested) and hooked up a wizard-like flow for new library. * Fixed a missing update event in backend when updating a library. * Disable Save when form invalid. Do inline validation on Library name when user types to ensure the name is valid. * Trim library names before you check anything * General code cleanup * Implemented advanced settings for library (include in dashboard, search, recommended) and ability to turn off folder watching for individual libraries. Refactored some code to streamline perf in some flows. * Removed old components replaced with new modal * Code smells Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * UX Alignment and bugfixes (#1663) * Refactored the design of reading list page to follow more in line with list view. Added release date on the reading list items, if it's set in underlying chapter. Fixed a bug where reordering the list items could sometimes not update correctly with drag and drop. * Removed a bug marker that I just fixed * When generating library covers, make them much smaller as they are only ever icons. * Fixed library settings not showing the correct image. * Fixed a bug where duplicate collection tags could be created. Fixed a bug where collection tag normalized title was being set to uppercase. Redesigned the edit collection tag modal to align with new library settings and provide inline name checks. * Updated edit reading list modal to align with new library settings modal pattern. Refactored the backend to ensure it flows correctly without allowing duplicate names. Don't show Continue point on series detail if the whole series is read. * Added some more unit tests around continue point * Fixed a bug on series detail when bulk selecting between volume and chapters, the code which determines which chapters are selected didn't take into account mixed layout for Storyline tab. * Refactored to generate an OpenAPI spec at root of Kavita. This will be loaded by a new API site for easy hosting. Deprecated EnableSwaggerUi preference as after validation new system works, this will be removed and instances can use our hosting to hit their server (or run a debug build). * Test GA * Reverted GA and instead do it in the build step. This will just force developers to commit it in. * GA please work * Removed redundant steps from test since build already does it. * Try another GA * Moved all test actions into initial build step, which should drastically cut down on time. Only run sonar if the secret is present (so not for forks). Updated build requirements for develop and stable docker pushes. * Fixed env variable * Okay not possible to do secrets in if statement * Fixed the build step to output the openapi.json where it's expected. * Fixed GA (#1664) * Bump versions by dotnet-bump-version. * Applied new _components layout structure to Kavita. All except manga as there is an open PR that drastically changes that module. (#1666) * Fixed typeahead and updated manga reader to new layout structure * Fixed book reader fonts lookups * Fixed up some build issues * Fixed a bad import of css image * Some cleanup and rewrote how we log out data. * Renderer can be null on first load when performing some work. * Library Settings Modal + New Library Settings (#1660) * Bump loader-utils from 2.0.3 to 2.0.4 in /UI/Web Bumps [loader-utils](https://github.com/webpack/loader-utils) from 2.0.3 to 2.0.4. - [Release notes](https://github.com/webpack/loader-utils/releases) - [Changelog](https://github.com/webpack/loader-utils/blob/v2.0.4/CHANGELOG.md) - [Commits](https://github.com/webpack/loader-utils/compare/v2.0.3...v2.0.4) --- updated-dependencies: - dependency-name: loader-utils dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Fixed want to read button on series detail not performing the correct action * Started the library settings. Added ability to update a cover image for a library. Updated backup db to also copy reading list (and now library) cover images. * Integrated Edit Library into new settings (not tested) and hooked up a wizard-like flow for new library. * Fixed a missing update event in backend when updating a library. * Disable Save when form invalid. Do inline validation on Library name when user types to ensure the name is valid. * Trim library names before you check anything * General code cleanup * Implemented advanced settings for library (include in dashboard, search, recommended) and ability to turn off folder watching for individual libraries. Refactored some code to streamline perf in some flows. * Removed old components replaced with new modal * Code smells Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * UX Alignment and bugfixes (#1663) * Refactored the design of reading list page to follow more in line with list view. Added release date on the reading list items, if it's set in underlying chapter. Fixed a bug where reordering the list items could sometimes not update correctly with drag and drop. * Removed a bug marker that I just fixed * When generating library covers, make them much smaller as they are only ever icons. * Fixed library settings not showing the correct image. * Fixed a bug where duplicate collection tags could be created. Fixed a bug where collection tag normalized title was being set to uppercase. Redesigned the edit collection tag modal to align with new library settings and provide inline name checks. * Updated edit reading list modal to align with new library settings modal pattern. Refactored the backend to ensure it flows correctly without allowing duplicate names. Don't show Continue point on series detail if the whole series is read. * Added some more unit tests around continue point * Fixed a bug on series detail when bulk selecting between volume and chapters, the code which determines which chapters are selected didn't take into account mixed layout for Storyline tab. * Refactored to generate an OpenAPI spec at root of Kavita. This will be loaded by a new API site for easy hosting. Deprecated EnableSwaggerUi preference as after validation new system works, this will be removed and instances can use our hosting to hit their server (or run a debug build). * Test GA * Reverted GA and instead do it in the build step. This will just force developers to commit it in. * GA please work * Removed redundant steps from test since build already does it. * Try another GA * Moved all test actions into initial build step, which should drastically cut down on time. Only run sonar if the secret is present (so not for forks). Updated build requirements for develop and stable docker pushes. * Fixed env variable * Okay not possible to do secrets in if statement * Fixed the build step to output the openapi.json where it's expected. * Applied new _components layout structure to Kavita. All except manga as there is an open PR that drastically changes that module. (#1666) * Post merge cleanup * Again moving the file * Fixed an issue with switching to double renderer and the image not loading for cover image. * Fixed double manga last page repeating twice * Added ability to quickly save a few settings to user preferences from manga reader * Fixed up some success messaging * Single image and canvas could stack, last page on double wouldn't render. * Stashing code, want to work on something else * Suppress a concurrency issue when opening a fresh chapter to read. * Refactored a function into a pipe * Took care of one TODO * Tightened up the logic around single renderer handling fit to screen images. * Added some code to see how long api takes on average. * First pass integration of page dimensions into single renderer and base code * Canvas renderer pass for new page dimensions * On time left, don't use the word left again * Moved the page dimension code into manga service to make it seemless * Hooked in a replacement for image based isWide * Canvas renderer is working again * Double renderer now follows how Komga does it to keep it simple. * Double renderer is working really well so far. * don't use nbsp * Added response caching to file-dimensions and chapter info api * Allow chapter info to send back file dimensions optionally * Fixed an issue with dimensions api locking files on Windows * Refactored all code to use isWidePage * More fixes and cleanup * More double reverse logic * Recently Read stats page will allow you to click the items. Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * More Reader Fixes (#1696) * Fixed resizing or layout changes causing page change on double reader * Implemented the debug log pattern on double renderers. Fixed a case when navigation backwards and showing only one page. Updated so go to page or slider update will handle selecting the right page number for pair display. * All Spread cases for double working * Cleanup dead code * Ensure we can jump to last page * Bump versions by dotnet-bump-version. * Reading History (#1699) * Added new stat graph for pages read over time for all users. * Switched to reading events rather than pages read to get a better scale * Changed query to use Created date as LastModified wont work since I just did a migration on all rows. * Small cleanup on graph * Read by day completed and ready for user stats page. * Changed the initial stat report to be in 1 day, to avoid people trying and ditching the software from muddying up the stats. * Cleaned up stats page such that stats around series show their image and tweaked some layout and wordings * Fixed recently read order * Put read history on user profile * Final cleanup, Robbie needs to do a CSS pass before release. * Bump versions by dotnet-bump-version. * Performance Improvements and Some Polish (#1702) * Auto scale reading timeline * Added benchmarks for SharpImage and NetVips. When an epub has a malformed page, catch the error and present it better to the user. * Added a hint for an upcoming feature * Slightly sped up word count for epubs * Added one more test to reflect actual code. * Some light cleanup * Use compact number for stat lists * Fixed brightness being broken on manga reader * Replaced CoverToWebP SharpImage version with NetVips which is MUCH lighter on memory and CPU. * Added last modified on the progress dto for CdDisplayEx. * Code cleanup * Forgot one cleanup * Bump versions by dotnet-bump-version. * Holiday Changes (#1706) * Fixed a bug on bookmark mode not finding correct image for prefetcher. * Fixed up the edit series relationship modal on tablet viewports. * On double page mode, only bookmark 1 page if only 1 pages is renderered on screen. * Added percentage read of a given library and average hours read per week to user stats. * Fixed a bug in the reader with paging in bookmark mode * Added a "This Week" option to top readers history * Added date ranges for reading time. Added dates that don't have anything, but might remove. * On phone, when applying a metadata filter, when clicking apply, collapse the filter automatically. * Disable jump bar and the resuming from last spot when a custom sort is applied. * Ensure all Regex.Replace or Matches have timeouts set * Bump versions by dotnet-bump-version. * First PR of the new year (#1717) * Fixed a bug on bookmark mode not finding correct image for prefetcher. * Fixed up the edit series relationship modal on tablet viewports. * On double page mode, only bookmark 1 page if only 1 pages is renderered on screen. * Added percentage read of a given library and average hours read per week to user stats. * Fixed a bug in the reader with paging in bookmark mode * Added a "This Week" option to top readers history * Added date ranges for reading time. Added dates that don't have anything, but might remove. * On phone, when applying a metadata filter, when clicking apply, collapse the filter automatically. * Disable jump bar and the resuming from last spot when a custom sort is applied. * Ensure all Regex.Replace or Matches have timeouts set * Fixed a long standing bug where fit to height on tablets wouldn't center the image * Streamlined url parsing to be more reliable * Reduced an additional db query in chapter info. * Added a missing task to convert covers to webP and added messaging to help the user understand to run it after modifying the setting. * Changed OPDS to be enabled by default for new installs. This should reduce issues with users being confused about it before it's enabled. * When there are multiple files for a chapter, show a count card on the series detail to help user understand duplicates exist. Made the unread badge smaller to avoid collision. * Added Word Count to user stats and wired up average reading per week. * Fixed word count failing on some epubs * Removed some debug code * Don't give more information than is necessary about file paths for page dimensions. * Fixed a bug where pagination area would be too small when the book's content was less that height on default mode. * Updated Default layout mode to Scroll for books. * Added bytes in the UI and at an API layer for CDisplayEx * Don't log health checks to logs at all. * Changed Word Count to Length to match the way pages work * Made reading time more clear when min hours is 0 * Apply more aggressive coalescing when remapping bad metadata keys for epubs. * Changed the amount of padding between icon and text for side nav item. * Fixed a NPE on book reader (harmless) * Fixed an ordering issue where Volume 1 was a single file but also tagged as Chapter 1 and Volume 2 was Chapter 0. Thus Volume 2 was being selected for continue point when Volume 1 should have been. * When clicking on an activity stream header from dashboard, show the title on the resulting page. * Removed a property that can't be animated * Fixed a typeahead typescript issue * Added Size into Series Info and Added some tooltip and spacing changes to better explain some fields. * Added size for volume drawers and cleaned up some date edge case handling * Fixed an annoying bug where when on mobile opening a view with a metadata filter, Kavita would open the filter automatically. * Bump versions by dotnet-bump-version. * Quick fix for Double Renderer (#1719) * Disable emulate comic book when on single page reader * Fixed a regression where double page renderer wouldn't layout the images correctly * Bump versions by dotnet-bump-version. * Feature/stats finishoff (#1720) * Added ability to click on genres, tags, and people to view all items in a modal. * Made it so we can click and open a filtered search from generic list * Fixed broken epub pagination area due to a typo in a query selector * Added day breakdown, wrapping up stats * Bump versions by dotnet-bump-version. * Docker nonroot (#1650) * Added PUID, PGID and KAVITAUSER variable support in entrypoint.sh * Update the setting of ownership to avoid changing library files * Default to run as root, using user kavita if alternate UID/GID are provided * Only chown config folder and only if needed * Revert chmod on Kavita Co-authored-by: Muggz <mug@passw0rd.org> * Bump versions by dotnet-bump-version. * Manga Reader Work (#1729) * Instead of augmenting prefetcher to move across chapter bounds, let's try to instead just load 5 images (which the browser will cache) from next/prev so when it loads, it's much faster. * Trialing loading next/prev chapters 5 pages to have better next page loading experience. * Tweaked GetChapterInfo API to actually apply conditional includeDimensions parameter. * added a basic language file for upcoming work * Moved the bottom menu up a bit for iOS devices with handlebars. * Fixed fit to width on phones still having a horizontal scrollbar * Fixed a bug where there is extra space under the image when fit to width and on a phone due to pagination going to far. * Changed which variable we use for right pagination calculation * Fixing fit to height - Fixing height calc to account for horizontal scroll bar height. * Added a comment for the height scrollbar fix * Adding screenfull package # Added: - Added screenfull package to handle cross-platform browser fullscreen code # Removed: - Removed custom fullscreen code * Fixed a bug where switching from webtoon reader to other layout modes wouldn't render anything. Webtoon continuous scroll down is now broken. * Fixed it back to how it was and all is good. Need to call detectChanges explicitly. * Removed an additional undeeded save progress call on loadPage * Laid out the test case to move the page snapping to the backend with full unit tests. Current code is broken just like UI layer. * Refactored the snap points into the backend and ensure that it works correctly. * Fixed a broken unit test * Filter out spammy hubs/messages calls in the logs * Swallow all noisy messages that are from RequestLoggingMiddleware when the log level is on Information or above. * Added a common loading component to the app. Have yet to refactor all screens to use this. * Bump json5 from 2.2.0 to 2.2.3 in /UI/Web Bumps [json5](https://github.com/json5/json5) from 2.2.0 to 2.2.3. - [Release notes](https://github.com/json5/json5/releases) - [Changelog](https://github.com/json5/json5/blob/main/CHANGELOG.md) - [Commits](https://github.com/json5/json5/compare/v2.2.0...v2.2.3) --- updated-dependencies: - dependency-name: json5 dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Alrigned all the loading messages and styles throughout the app * Webtoon reader will use max width of all images to ensure images align well. * On Original scaling mode, users can use the keyboard to scroll around the images without pagination kicking off. * Removed console logs * Fixed a public vs private issue * Fixed an issue around some cached files getting locked due to NetVips holding them during file size calculations. Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * [Manga Reader] Swipe Support (#1735) * Fixed a loading indicator that is always on * Started to add swipe directive * Implemented the ability to swipe to navigate pages in manga reader. * Swipe to paginate seems to be working reliably * Removed a bunch of junk from csproj and added a debug menu for testing on phone to smooth out experience. * Fixed a bug where reading list detail wouldn't render the set image of the reading list. * Added some instructions and code to allow connecting to dev instance easier. * Fixed up paging with keyboard where to ensure that when we hit the end of the scroll, we don't go to the next page instantly, but rather make the user press the key once more. * Fixed reading list image not properly renderering on reading list detail page. * Solved the swiping bug, need to play with threshold again. * Swipe is now working. Need to decide if I'm going to support reversing the direction with reading direction. * Hooked up swipe with reading direction code * Cleaned up some direction code to align to a new enum * Feature complete * Bump versions by dotnet-bump-version. * Fix pagination in Manga reader from last PR (#1736) * Added a new Double (No Cover) rendering mode which always has first 2 pages together unless wide. * Removed layout mode for build * Bump versions by dotnet-bump-version. * Better Themes, Stats, and bugfixes (#1740) * Fixed a bug where when clicking on a series rating for first time, the rating wasn't populating in the modal. * Fixed a bug on Scroll mode with immersive mode, the bottom bar could clip with the book body. * Cleanup some uses of var * Refactored text as json into a type so I don't have to copy/paste everywhere * Theme styles now override the defaults and theme owners no longer need to maintain all the variables themselves. Themes can now override the color of the header on mobile devices via --theme-color and Kavita will now update both theme color as well as color scheme. * Fixed a bug where last active on user stats wasn't for the particular user. * Added a more accurate word count calculation and the ability to see the word counts year over year. * Added a new table for long term statistics, like number of files over the years. No views are present for this data, I will add them later. * Bump versions by dotnet-bump-version. * Swipe Issues (#1745) * Updated theme support to be able to customize the tile color dynamically from a theme via --tile-color. In addition, --theme-color will update apple-mobile-web-app-status-bar-style as well as the non-apple variants * Removed --manga-reader-bg-color as it wasn't used anywhere. Fixed double pagination on swipe. * Cleaned up some dead threshold code for swipe. * Started refactoring tests to use an abstract test class. Stopping because I should do on the .net 7 branch to avoid large merge conflicts. Tests need to be re-designed so they can run in parallel. * Fixed a bug in reading lists where when deleting an item, order could be miscalculated. * Started adding new information for stat service. Refactored time spent reading to be more accurate by taking average time against how much of the chapter the user has read. * Hooked up total time reading at server stat level. Don't show fancy graphs on mobile. * Added new stats for v0.7 * Added a test for Clearing want to read * Fixed a few tests that weren't resetting state between runs * Fixed some broken unit tests * Ensure all Series queries sort by a case invariant string. * Added more aggressive caching of images. This will result in a min delay on pages after a cover is changed. * Fixed a bug where if during new word count calculation, new word count is zero, restoring the old count wasn't working. * Cleaned up some of the code for getting time estimates * Fixed a bug where triggering swipe right wasn't working when there was no scroll * Delete the temp folder for creating a download after a full zip is created. * Bump versions by dotnet-bump-version. * Stat hotfix (#1748) * Fixed a bug where a divide by 0 could occur * Email change now requires a password * Bump versions by dotnet-bump-version. * Holiday Bugfixes (#1762) * Don't show "not much going on" when we are actively downloading * Swipe to paginate is now behind a flag in the user preferences. * Added a new server setting for host name, if the server sits behind a reverse proxy. If this is set, email link generation will use it and will not perform any checks on accessibility (thus email will always send) * Refactored the code that checks if the server is accessible to check if host name is set, and thus return rue if so. * Added back the system drawing library for markdown parsing. * Fixed a validation error * Fixed a bug where folder watching could get re-triggered when it was disabled at a server level. * Made the manga reader loader absolute positioned for better visibility * Indentation * Bump versions by dotnet-bump-version. * Angular 15 (#1764) * Updated ngx-virtual-scroller * Removed the karma test config as it's breaking migration * Reverted to pre angular 15 * Upgraded packages and reverted target to ES6 for older devices * It's broken. Need to also find the safari version for old Ipads * Fixes some code in default pipe and many updates to packages. Removed support for old iOS versions as it restricted Kavita from using newer features. Build still broken. * More progress in getting build working on Angular 15. Removed polyfills.ts for new angular config * Remove all.css for icons and use scss instead * Removed stuff that isn't needed * Migrated extended linting to eslint, ran on project and updated issues. Removed a duplicate component that did nothing. Fixed a few places where lifecycle hooks werent being called as interface wasn't implemented. * App builds correctly. Source maps are still needed. * Fixed source maps and removed more testing stuff. I will re-add later in another release when I figure out how to properly tackle dependencies on backend. * Reverted back to old source map definition * Bump versions by dotnet-bump-version. * Angular 15 (#1765) * Refactored some code in BookService to make the code easier to understand * More lint fixes * Use npm ci for installs in pipeline * Fixed build system again by deleting nodejs. New build system uses package-lcok going forward. * Bump versions by dotnet-bump-version. * [skip ci] Misc stuff (#1766) * Refactored some code in BookService to make the code easier to understand * More lint fixes * Use npm ci for installs in pipeline * Fixed build system again by deleting nodejs. New build system uses package-lcok going forward. * Added a test case for Reading Time Estimation calculations * Some cleanup * Added even more testing to try and get scare's issue captured. * Automatic Collection Creation (#1768) * Made the unread badges slightly smaller and rounded on top right. * A bit more tweaks on the not read badges. Looking really nice now. * In order to start the work on managing collections from ScanLoop, I needed to refactor collection apis into the service layer and add unit tests. Removed ToUpper Normalization for new tags. * Hooked up ability to auto generate collections from SeriesGroup metadata tag. * Bump versions by dotnet-bump-version. * Auto Collection Bugfixes (#1769) * SeriesGroup tag can now have comma separated value to allow a series to be a part of multiple collections. * Added a missing unit test * Refactored how collection tags are created to work in the scan loop reliably. * Added a unit test for RemoveTagsWithoutSeries * Fixed a bug in reading list title generation to avoid Volume 0 if the underlying file had a title set. Fixed a misconfigured unit test. * Bump versions by dotnet-bump-version. * Scanner Performance Improvements (#1774) * Refactored the Genre code to be faster and used a dictonary to avoid some lookups. May fix the rare foreign constraint issue. * Refactored tag to the same implementation as Genre. Ensure when grabbing tags from ComicInfo, we normalize and throw out duplicates. * Removed an internal "external" field that was planned for Genres and Tags, but now with new plugin architecture, not needed. * Bump versions by dotnet-bump-version. * Spelling, grammar, and related consistency improvements (#1756) * Spelling, grammar, and word structure improvements * Email service text reworded to account for the Host Name feature * Bump versions by dotnet-bump-version. * Stat Polish (#1775) * SeriesGroup tag can now have comma separated value to allow a series to be a part of multiple collections. * Added a missing unit test * Refactored how collection tags are created to work in the scan loop reliably. * Added a unit test for RemoveTagsWithoutSeries * Fixed a bug in reading list title generation to avoid Volume 0 if the underlying file had a title set. Fixed a misconfigured unit test. * On User stats page, don't show the user selector on reading history, despite if youre an admin. Cleaned up how we show days with 0 reading events to be more clear. * Refactored the name of a component to reflect what it does * Removed plugin not using * Fix an issue where coalescing a key in epub might have multiple html files ending with the key. In this case, let's take the first. * Added PikaPods to the Readme * Tried to fix layout shift for charts, but need Robbie's help * Chart styling # Added: - Added: Added styling to force charts into their respective containers. # Removed: - Removed: Removed code blocking charts from being visible on mobile. * Merge conflict --------- Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> * Bump versions by dotnet-bump-version. * UTC Dates + CDisplayEx API Enhancements (#1781) * Introduced a new claim on the Token to get UserId as well as Username, thus allowing for many places of reduced DB calls. All users will need to reauthenticate. Introduced UTC Dates throughout the application, they are not exposed in all DTOs, that will come later when we fully switch over. For now, Utc dates will be updated along side timezone specific dates. Refactored get-progress/progress api to be 50% faster by reducing how much data is loaded from the query. * Speed up the following apis: collection/search, download/bookmarks, reader/bookmark-info, recommended/quick-reads, recommended/quick-catchup-reads, recommended/highly-rated, recommended/more-in, recommended/rediscover, want-to-read/ * Added a migration to sync all dates with their new UTC counterpart. * Added LastReadingProgressUtc onto ChapterDto for some browsing apis, but not all. Added LastReadingProgressUtc to reading list items. Refactored the migration to run raw SQL which is much faster. * Added LastReadingProgressUtc onto ChapterDto for some browsing apis, but not all. Added LastReadingProgressUtc to reading list items. Refactored the migration to run raw SQL which is much faster. * Fixed the unit tests * Fixed an issue with auto mapper which was causing progress page number to not get sent to UI * series/volume has chapter last reading progress * Added filesize and library name on reading list item dto for CDisplayEx. * Some minor code cleanup * Forgot to fill a field * Bump versions by dotnet-bump-version. * Reading List Fixes (#1784) * Add ability to save readinglist comicinfo fields in Chapter. * Added the appropriate fields and migration for Reading List generation. * Started the reading list code * Started building out the CBL import code with some initial unit tests. * Fixed first unit test * Started refactoring control code into services and writing unit tests for ReadingLists. Found a logic issue around reading list title between create/update. Will be corrected in this branch with unit tests. * Can't figure out how to mock UserManager, so had to uncomment a few tests. * Tooltip for total pages read shows the full number * Tweaked the math a bit for average reading per week. * Fixed up the reading list unit tests. Fixed an issue where when inserting chapters into a blank reading list, the initial reading list item would have an order of 1 instead of 0. * Cleaned up the code to allow the reading list code to be localized easily and fixed up a bug in last PR. * Fixed a sorting issue on reading activity * Tweaked the code around reading list actionables not showing due to some weird filter. * Fixed edit library settings not opening on library detail page * Fixed a bug where reading activity dates would be out of order due to a bug in how charts works. A temp hack has been added. * Disable promotion in edit reading list modal since non-admins can (and should have) been able to use it. * Fixed a bug where non-admins couldn't update their OWN reading lists. Made uploading a cover image for readinglists now check against the user's reading list access to allow non-admin's to set images. * Fixed an issue introduced earlier in PR where adding chapters to reading list could cause order to get skewed. * Fixed another regression from earlier commit * Hooked in Import CBL flow. No functionality yet. * Code is a mess. Shifting how the whole import process is going to be done. Commiting so I can pivot drastically. * Very rough code for first step is done. * Ui has started, I've run out of steam for this feature. * Cleaned up the UI code a bit to make the step tracker nature easier without a dedicated component. * Much flow implementation and tweaking to how validation checks and what is sent back. * Removed import via cbl code as it's not done. Pushing to next release. * Bump versions by dotnet-bump-version. * Allow changing listening ip addresses (#1713) * Allow changing listening ip address * Use Json serialize for appsettings.config saving * BOM * IP Address validation * ip address reset * ValidIpAddress regex * Bump versions by dotnet-bump-version. * Release Testing Time (#1785) * Fixed a bug with getting continue point where there was a single volume unread and a later volume with chapters inside it, the chapters were being picked. * Fixed a bug where resuming from jump key wasn't working (develop) * Cleaned up the spacing * Bump versions by dotnet-bump-version. * Release Testing Bugs (#1790) * Stop showing loading indicator when no next/prev chapter * Fixed a bug where manage collections wasn't named correctly in UI. * Slight tweaks on email flow * Bump versions by dotnet-bump-version. * Release Testing Part 2 (#1794) * Stop showing loading indicator when no next/prev chapter * Fixed a bug where manage collections wasn't named correctly in UI. * Slight tweaks on email flow * Fixed a bug where we were grabbing wrong property for book layout mode * Fixed an issue where pagination area wasn't properly spanning window on different scaling modes. * Fixed a bug where right pagination area wasn't sticking to the right hand side on original scaling * Added a note from reading an epub3 * Reworked some of the readme * Bump versions by dotnet-bump-version. * Small Build Fix (#1795) * Stop showing loading indicator when no next/prev chapter * Fixed a bug where manage collections wasn't named correctly in UI. * Slight tweaks on email flow * Fixed a bug where we were grabbing wrong property for book layout mode * Fixed an issue where pagination area wasn't properly spanning window on different scaling modes. * Fixed a bug where right pagination area wasn't sticking to the right hand side on original scaling * Added a note from reading an epub3 * Reworked some of the readme * Changed the build to ci * Bump versions by dotnet-bump-version. * Final Release Testing (#1796) * Fix some wording * Fixed up stats to have total info on hover * Fixed up a stat card not having clickable hint * Bump versions by dotnet-bump-version. * v0.7 Release (#1797) * Bump versions by dotnet-bump-version. * Aligned build with main (#1801) * Bump versions by dotnet-bump-version. * Revert "Aligned build with main (#1801)" This reverts commit 3f6eb5fe17934cc5aa3b227b165e184bf47ade14. * v0.7 Issues for Hotfix (#1812) * Fix signalr admin messages sending (#1809) * Changed messsage hub to use userIds * SignalR events are fixed * Fixed broken advanced tab on library settings * Fixed regex timeout security issues. * Added a migration for GMT+1 users where their UTC dates were getting broken somehow. * Removed a console.log * Fixed a migration name --------- Co-authored-by: Snd-R <76580768+Snd-R@users.noreply.github.com> * Bump versions by dotnet-bump-version. * Fixed manage collections not being toggleable (#1813) * Bump versions by dotnet-bump-version. * Version bump (#1814) * Bump versions by dotnet-bump-version. * Random Changes and Enhancements (#1819) * When skipping over folders in a scan, inform the ui * Try out new backout condition for library watcher. * Tweaked the code for folder watching to be more intense on killing if stuck in inotify loop. * Streamlined my implementation of enhanced LibraryWatcher * Added new extension method to make complex where statements cleaner. * Added an implementation to flatten series and not show them if they have relationships defined. Only the parent would show. Currently disabled until i figure out how to apply it. * Added the ability to collapse series that are not the primary entry point to reading. Configurable in library settings, only applies when all libraries in a filter have the property to true. * Exclude from parsing .@_thumb directories, a QNAP system folder. Show number of items a JumpKey has * Refactored some time reading to display in days, months, years or minutes. * Bump versions by dotnet-bump-version. * Epub 3.2 Grouping issue (#1823) * Added font swapping and removed some unneeded css * Fixed a bug where epub 3 tags weren't being applied for grouping * Bump versions by dotnet-bump-version. * Revert ability to run docker as non-root due to issues with Synology. (#1832) * Bump versions by dotnet-bump-version. * CBL Import (#1834) * Wrote my own step tracker and added a prev button. Works up to first conflict flow. * Everything but final import is hooked up in the UI. Polish still needed, but getting there. * Making more progress in the CBL import flow. * Ready for the last step * Cleaned up some logic to prepare for the last step and reset * Users like order to be starting at 1 * Fixed a few bugs around cbl import * CBL import is ready for some basic testing * Added a reading list hook on side nav * Fixed up unit tests * Added icons and color to the import flow * Tweaked some phrasing * Hooked up a loading variable but disabled the component as it didn't look good. * Styling it up * changed an icon to better fit --------- Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> * Bump versions by dotnet-bump-version. * .NET 7 + Spring Cleaning (#1677) * Updated to net7.0 * Updated GA to .net 7 * Updated System.IO.Abstractions to use New factory. * Converted Regex into SourceGenerator in Parser. * Updated more regex to source generators. * Enabled Nullability and more regex changes throughout codebase. * Parser is 100% GeneratedRegexified * Lots of nullability code * Enabled nullability for all repositories. * Fixed another unit test * Refactored some code around and took care of some todos. * Updating code for nullability and cleaning up methods that aren't used anymore. Refctored all uses of Parser.Normalize() to use new extension * More nullability exercises. 500 warnings to go. * Fixed a bug where custom file uploads for entities wouldn't save in webP. * Nullability is done for all DTOs * Fixed all unit tests and nullability for the project. Only OPDS is left which will be done with an upcoming OPDS enhancement. * Use localization in book service after validating * Code smells * Switched to preview build of swashbuckle for .net7 support * Fixed up merge issues * Disable emulate comic book when on single page reader * Fixed a regression where double page renderer wouldn't layout the images correctly * Updated to swashbuckle which support .net 7 * Fixed a bad GA action * Some code cleanup * More code smells * Took care of most of nullable issues * Fixed a broken test due to having more than one test run in parallel * I'm really not sure why the unit tests are failing or are so extremely slow on .net 7 * Updated all dependencies * Fixed up build and removed hardcoded framework from build scripts. (this merge removes Regex Source generators). Unit tests are completely busted. * Unit tests and code cleanup. Needs shakeout now. * Adjusted Series model since a few fields are not-nullable. Removed dead imports on the project. * Refactored to use Builder pattern for all unit tests. * Switched nullability down to warnings. It wasn't possible to switch due to constraint issues in DB Migration. * Bump versions by dotnet-bump-version. * Added noindex to prevent SEO from indexing user's sites. Reverted http3 support as docker users were having port issue (#1850) * Bump versions by dotnet-bump-version. * Fixed docker instances not being able to start due to misconfigured default for IpAddress binding (#1851) * Bump versions by dotnet-bump-version. * Fixing docker bind issue (#1852) # Fixed: - Fixed an issue for docker users conflicting with a previous feature inclusion for multiple IP addresses * Docker bind fix (#1853) * Fixing docker bind issue # Fixed: - Fixed an issue for docker users conflicting with a previous feature inclusion for multiple IP addresses * build error fix * Docker bind fix (#1854) * Fixing docker bind issue # Fixed: - Fixed an issue for docker users conflicting with a previous feature inclusion for multiple IP addresses * build error fix * re-raise * fix (#1855) * Bump versions by dotnet-bump-version. * Added vertical reading mode to the book reader (#1787) * Add vertical reading mode support and update API for reading mode preference * Removed dead code, added a fix for scroll end margins in chrome when in vertical mode(book reader). Added back some comments * Added Description()] annotation for the ReadingMode enum, like other enums, and added summary documentation * Added the ability to scroll in vertical writing style without holding down shift. Also renamed the book reader's readingMode to writing style. * Renamed the BookReadingMode to BookWritingStyle. And changed the migrations accordingly. * Fixed some minor bugs, regarding scrolling and vertical writing style when the book settings is open. * Fixed a minor bug where the graphics regarding the current page would require the mouse to be moved before it got updated when switching between writing styles. * Fixed some bugs regarding furigana getting a bit cropped same for images * Add vertical reading mode support and update API for reading mode preference * Removed dead code, added a fix for scroll end margins in chrome when in vertical mode(book reader). Added back some comments * Added Description()] annotation for the ReadingMode enum, like other enums, and added summary documentation * Added the ability to scroll in vertical writing style without holding down shift. Also renamed the book reader's readingMode to writing style. * Renamed the BookReadingMode to BookWritingStyle. And changed the migrations accordingly. * Fixed some minor bugs, regarding scrolling and vertical writing style when the book settings is open. * Fixed a minor bug where the graphics regarding the current page would require the mouse to be moved before it got updated when switching between writing styles. * Fixed some bugs regarding furigana getting a bit cropped same for images * Added reset support for writing style, after rebase. * Changes pagination for vertical scrolling such as the user will need to scroll to end before being able to paginate. Previously it felt unnatural and the user could accidentally paginate while scrolling on mobile. * Pagination would not stick to the left if the content was smaller than the reader in vertical writing style. * Fixed summary text * Added missing line, fixes build error * Addresses the comments given in code-review. * Moved columnGap outside the class, and changed it to a const * Bump versions by dotnet-bump-version. * Bumped css budget (#1856) * Bump versions by dotnet-bump-version. * Fix for docker bind (#1857) #Added: - Added: null IpAddresses to appsettings * Bump versions by dotnet-bump-version. * Default to empty string for IpAddress to allow to fallback into existing experience (#1859) Co-authored-by: Joseph Milazzo <joseph.v.milazzo@gmail.com> * Bump versions by dotnet-bump-version. * CBL Import Rework (#1862) * Fixed a typo in a log * Invalid XML files now "validate" correctly by sending back a failure. * Cleaned up messaging on backend and frontend to provide some linking on series name when collision, handle corrupt xml files, etc. * When reading list conflict occurs, show the reading list name that's conflicting. Started refactoring the code to allow multiple files to be imported at once. * Started adding new CBL elements for some enhancements I have planned with maintainers. * Default to empty string for IpAddress to allow to fallback into existing experience * Tweaked the layout of reading list page (not complete), moved some not used much controls to page extras and reordered the buttons for reading list * Edit Reading Lists now allows selection of cover image from existing items * Fixed a bug where cover chooser base64 to image would fail to write webp files. * Refactored the validate step to now handle multiple files in one go. * Clean up code * Don't show CBL name if there were xml errors that prevented showing it * Don't allow user to go prev step after they perform the import. * Cleaned up the heading code for accordions * Fixed a bug with import keeping failed items * Sort the failures to the bottom of result windows * CBL import is pretty solid. Need one pass from Robbie on Reading List Page * Bump versions by dotnet-bump-version. * Fix for themes not applying (#1873) # Fixed: - Fixed: Fixed an issue where themes would not apply properly. * Bump versions by dotnet-bump-version. * OPDS-PS v1.2 Support + a few bugfixes (#1869) * Fixed up a localization lookup test case * Refactored some webp to a unified method * Cleaned up some code * Expanded webp conversion for covers to all entities * Code cleanup * Prompt the user when they are about to delete multiple series via bulk actions * Aligned Kavita to OPDS-PS 1.2. * Fixed a bug where clearing metadata filter of series name didn't clear the actual field. * Added some documentation * Refactored how covert covers to webp works. Now we will handle all custom covers for all entities. Volumes and Series will not be touched but instead be updated via a RefreshCovers call. This will fix up the references much faster. * Fixed up the OPDS-PS 1.2 attributes to only show on PS links * Bump versions by dotnet-bump-version. * Fixes bus in vertical writing mode. (#1871) * Bump versions by dotnet-bump-version. * Misc Fixes + Enhancements (#1875) * Moved Collapse Series with relationships into a user preference rather than library setting. * Fixed bookmarks not converting to webp after initial save * Fixed a bug where when merging we'd print out a duplicate series error when we shouldn't have * Fixed a bug where clicking on a genre or tag from server stats wouldn't load all-series page in a filtered state. * Implemented the ability to have Login role and thus disable accounts. * Ensure first time flow gets the Login role * Refactored user management screen so that pending users can be edited or deleted before the end user accepts the invite. A side effect is old legacy users that were here before email was required can now be deleted. * Show a progress bar under the main series image on larger viewports to show whole series progress. * Removed code no longer needed * Cleanup tags, people, collections without connections after editing series metadata. * Moved the Entity Builders to the main project * Bump versions by dotnet-bump-version. * Reworks how the image sizes in the book reader gets handled (#1872) * Fixes bus in vertical writing mode. * Reworks how the images gets resized. They should be more adaptive and pages with single images will get centered. * Adds back an unintended change * Revert "Fixes bus in vertical writing mode." This reverts commit a1cf9675def45fd626a149d90f573104b5d280cd. * Added debounce and timeout to the updateImageSize to prevent it from getting run before the page loads * Fixes semicolons missing semicolons * Improves the detection in checkSingleImagePage. * Bump versions by dotnet-bump-version. * Base Url implementation (#1824) * Base Url implementation * PR requested changes * Bump versions by dotnet-bump-version. * Reading List Polish (#1879) * Use Reading Order to count epub pages rather than raw HTML files. * Send email on background thread for initial invite flow. * Reorder default writing style for new users so Horizontal is default * Changed reading activity to use average hours read rather than events to bring more meaningful data. * added ability to start reading incognito from the top of series detail, needs a bit of styling help though. * Refactored extensions out into their own package, added new fields for reading list to cover total run, cbl import now takes those dates and overrides on import. Replaced many instances of numbers to be comma separated. * Added ability to edit reading list run start and end year/month. Refactored some code for valid month/year into a helper method. * Added a way to see the reading list's release years. * Added some merged image code, but had to remove due to cover dimensions not fixed. * tweaked style for accessibility mode on reading list items * Tweaked css for non virtualized and virtualized containers * Fixed release updates failing * Commented out the merge code. * Typo on words read per year * Fixed unit tests * Fixed virtualized scroll * Cleanup CSS * Bump versions by dotnet-bump-version. * Fixed an issue where docker installs were trying to update wwwroot, when docker doesn't do reverse proxy stuff. (#1880) Fixed a bug in reading list date calculations. * Bump versions by dotnet-bump-version. * Security Event Logging & Bugfixes (#1882) * Fixed bookmarking failing to convert to webp * Brought the ag-swipe/ng-swipe code into Kavita due to being abandoned by developer and angular requirements. * Fixed average reading time per week finally * Cleaned up some extra decimals on time duration pipe * Don't try to update index.html for base url on local. Fixed ag-swipe on prod mode. * Updated a link on theme manager to point to the new github * Range knobs should be primary color on firefox too * Implemented the ability to get thumbnails of pages inside an archive or pdf. * Updated packages and fixed opds-ps 1.2 issue * Fixed lock file * Allow Kavita's Swagger to hit instances with CORS * Added IP/Request logging for Security Audits * Linked up Summary tag from CBL into Kavita. * Redid the migration so SecurityEvent now has UTC date as well. * Split security logging to a separate file * Update to new versions of checkout and setup * Added a PR check on PR body to ensure that it doesn't contain any characters that break our discord hook. * Updating action * optimize regex in action * Fixed an issue where fit to width would cause the actual height of the image to be shown for pagination bars, instead of rendered. * Added some new code in GetPageFromFiles to ensure pages that exceed array map down to last file. * Added comment about robots * Fixed up unit tests for new ReaderService signature * Kavita now cleans up empty reading lists at night * Don't allow nightly cleanup to run if we are running media conversion tasks * Fixed some bugs in typeahead, it should behave much more reliably. * Fix an issue where emulate comic book wasn't extending to the bottom properly * Added support for Series Chapter 001 Volume 001 * Refactor XFrameOptions="SameOrigins" out to allow users to override in appsettings.json. * Added a rate limiter for some endpoints, but it doesn't seem to be triggering --------- Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> * Security Event Logging & Bugfixes (rerelease) (#1883) * Fixed bookmarking failing to convert to webp * Brought the ag-swipe/ng-swipe code into Kavita due to being abandoned by developer and angular requirements. * Fixed average reading time per week finally * Cleaned up some extra decimals on time duration pipe * Don't try to update index.html for base url on local. Fixed ag-swipe on prod mode. * Updated a link on theme manager to point to the new github * Range knobs should be primary color on firefox too * Implemented the ability to get thumbnails of pages inside an archive or pdf. * Updated packages and fixed opds-ps 1.2 issue * Fixed lock file * Allow Kavita's Swagger to hit instances with CORS * Added IP/Request logging for Security Audits * Linked up Summary tag from CBL into Kavita. * Redid the migration so SecurityEvent now has UTC date as well. * Split security logging to a separate file * Update to new versions of checkout and setup * Added a PR check on PR body to ensure that it doesn't contain any characters that break our discord hook. * Updating action * optimize regex in action * Fixed an issue where fit to width would cause the actual height of the image to be shown for pagination bars, instead of rendered. * Added some new code in GetPageFromFiles to ensure pages that exceed array map down to last file. * Added comment about robots * Fixed up unit tests for new ReaderService signature * Kavita now cleans up empty reading lists at night * Don't allow nightly cleanup to run if we are running media conversion tasks * Fixed some bugs in typeahead, it should behave much more reliably. * Fix an issue where emulate comic book wasn't extending to the bottom properly * Added support for Series Chapter 001 Volume 001 * Refactor XFrameOptions="SameOrigins" out to allow users to override in appsettings.json. * Added a rate limiter for some endpoints, but it doesn't seem to be triggering * Fixed bad yaml --------- Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> * Bump versions by dotnet-bump-version. * Some fixes from last release (#1884) * Removed SecurityEvent middleware solution. It was out of scope originally. * Fixed manage users still calling pending when the api is no more * Added back the online indicator on manage users * Bump versions by dotnet-bump-version. * Added push to ghcr.io (#1885) * Added container push to ghcr.io * Updated login method * Bump versions by dotnet-bump-version. * Donate styling (#1886) * Donate styling # Changed - Changed the styling and positioning of the donate button to still be visible but less intrusive and more cohesive to the application. * Updated pr action * Revert "Updated pr action" This reverts commit d3f7c7b2edfdcdbb79acb9e7816247ef5ebc30ea. * Bump versions by dotnet-bump-version. * Fix workflow # Fixed: - Fixed ghcr.io build - Updated PR action * Bump versions by dotnet-bump-version. * Tweaks (#1890) * Updated number inputs with a more mobile friendly control * Started writing lots of unit tests on PersonHelper to try and hammer out foreign constraint * Fixes side-nav actionable alignment * Added some unit tests * Buffed out the unit tests * Applied input modes throughout the app * Fixed a small bug in refresh token validation to make it work correctly * Try out a new way to block multithreading from interacting with people during series metadata update. * Fixed the lock code to properly lock, which should help with any constraint issues. * Locking notes * Tweaked locking on people to prevent a constraint issue. This slows down the scanner a bit, but not much. Will tweak after validating on a user's server. * Replaced all DBFactory.Series with SeriesBuilder. * Replaced all DBFactory.Volume() with VolumeBuilder * Replaced SeriesMetadata with Builder * Replaced DBFactory.CollectionTag * Lots of refactoring to streamline entity creation * Fixed one of the unit tests * Refactored all of new Library() * Removed tag and genre * Removed new SeriesMetadata * Refactored new Volume() * MangaFile() * ReadingList() * Refactored all of Chapter and ReadingList * Add title to all event widget flows * Updated Base Url to inform user it doesn't work for docker users with non-root user. * Added unit test coverage to FormatChapterTitle and FormatChapterName. * Started on Unit test for scanner, but need to finish it later. --------- Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> * Bump versions by dotnet-bump-version. * OPDS Flattening (#1904) * Flattening OPDS Structure # Changed - Flattened OPDS structure to reduce user taps. * Fixing format * Fixing book series titles * Optimized file size to use pre-calculated data to avoid an I/O touch. * Fixes #1898 by aligning all content headers to the correct MIME types * Remove dead code * Fixed a bug with continue point where it fails on chapters or volumes tagged with a range --------- Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> * Bump versions by dotnet-bump-version. * OPDS formatting fix (#1905) # Fixed - Fixed a formatting issue where there was a unnecessary - at the beginning of titles. (develop) # Changed - Updated github action version. (develop) * Bump versions by dotnet-bump-version. * Password Validation and Google phishing error pass (#1913) * Updating password validation # Changed: - Changed: Updated password validation to use pattern so user can not submit or create passwords <6 or >32 characters long. * Removed the svg for image in case it's causing google security issues. --------- Co-authored-by: Joseph Milazzo <joseph.v.milazzo@gmail.com> * Bump versions by dotnet-bump-version. * More Google Security Changes (#1923) * Removed packages not used anymore * Reverted removing lazysizes * Removed the svg for image in case it's causing google security issues. * Removed unneeded service * Bump versions by dotnet-bump-version. * Manga Reader Fixes (#1925) * Fixed up an issue where image might be cut off in fit to height * Removed some calls to backend for translating age rating to a string * Fixed an issue with sizing on page splitting right to left. * Ensure all image access requires apikey * Removed a TODO * Bump versions by dotnet-bump-version. * Bugfix/base url in docker (#1901) * fixed: BaseUrl option is not supported in Docker - discussion in #1824, the checks should not have been added in the first place * changed: contributors require .NET 7.0+ in order to build * Bump versions by dotnet-bump-version. * Random Cleanup + OPDS Base Url Support (#1926) * Updated a ton of dependencies. PDFs reader got a big update from PDF.js 2.6 -> 3.x * Rolled back fontawesome update * Updated to latest angular patch. Fixed search being too long instead of just to the end of the browser screen. * Fixed alignment on download icon for download indicator in cards * Include progress information on Want To Read API and when marking something as Read, perform cleanup service on want to read. * Removed mark-read updating want to read. As there are series restrictions and it could be misleading. * Tweaked login page spacing when form is dirty * Replaced an object instantiation * Commented out a few tests that always break when updating NetVips (but always work) * Updated ngx-toastr * Added styles for alerts to Kavita. They were somehow missing. Fixed an issue where when OPDS was disabled, user preferences wouldn't tell them. * Wired up a reset base url button to match Ip Addresses * Disable ipAddress and port for docker users * Removed cache dir since it's kinda pointless currently * Started the update for OPDS BaseUrl support * Fixed OPDS url not reflecting base url on localhost * Added extra plumbing to allow sending a real email when testing a custom service. * Implemented OPDS support under Base Url. Added pagination to all APIs where applicable. * Added a swallowing of permission denied on Updating baseurl in index.html for inapplicable users. * Fixed a bad test * Bump versions by dotnet-bump-version. * Bugfixes from last PR (#1928) * Fixed a regression where confirm-email was validating for a real email when it's not required. * Fixed fit to screen breaking as canvas was still showing on the screen when it should have been hidden. * Fixed a missing age rating in the pipe * Fixed fit to height not properly aligning the image * Tweaked Startup check for base url * Bump versions by dotnet-bump-version. * ComicInfo Derived Reading Lists (#1929) * Implemented the ability to generate reading lists from StoryArc and StoryArcNumber ComicInfo fields. * Refactored to add AlternativeSeries support. * Fixed up the handling when we need to update reading list where order is already present. * Refactored how skipping empty reading list pairs works * Library Settings for Reading List Creation (#1930) * Added the code for Library Settings to allow turning off Reading List creation. * Fixed a typo * Bump versions by dotnet-bump-version. * Pre-Release Shakeout (#1932) * Allow users to setup an account with a fake email and the same username without hitting the validate username code. * Fixed a bug where opds url could have //. Fixed a bug where baseurl wasn't being used for invite link generation * Fixed enum mismatch causing age rating to display incorrectly on UI * Bump versions by dotnet-bump-version. * Release Testing Day 1 (#1933) * Enhance plugin/authenticate to allow RefreshToken to be returned as well. * When typing a series name, min, or max filter, press enter to apply metadata filter. * Cleaned up the documentation around MaxCount and TotalCount * Fixed a bug where PublicationStatus wasn't being correctly set due to some strange logic I coded. * Fixed bookmark mode not having access to critical page dimensions. Fetching bookmark info api now returns dimensions by default. * Fixed pagination scaling code for different fitting options * Fixed missing code to persist page split in manga reader * Removed unneeded prefetch of blank images in bookmark mode * Bump versions by dotnet-bump-version. * Release Testing Day 2 (#1937) * Removed unneded ngModel on password field * Fixed some bad validation messages on Edit Reading List modal and disabled save button * Added a lot of trace code to help debug a foreign constraint issue. * Fixed a bug where after a series is scanned, generate covers for series didn't respect webp cover generation. * Fixed library last scan being stored in Utc, but expected to be server time. * Fixed up some of that trace logging being way too verbose. Fixed a case where when a missing storyarc number, the whole pair was dropped. Now, it will default that item to the end of the reading list. Fixed a bug where Start and End dates weren't being calculated after generating a reading list. * Fixed a bug where the default admin user for reading list creation from files was selecting the wrong user. Changed so that when there is a bad pair (aka number missing) and only 1 pair, then we wont constantly reorder the item. * Fixed unit test * Bump versions by dotnet-bump-version. * Logging Change (#1941) * Tweaked trace logging for foreign constraint issue. Removed some locks where we werent touching important code that needed locking on. * Reverted back some code * Bump versions by dotnet-bump-version. * Reading List Bug (#1943) * Refactored how we calculate start and end dates for a reading list generated from Scan. * Cleaned up the logging messages a bit * By default, allow DB statements to include params, since there's no PPI involved. * Bump versions by dotnet-bump-version. * Tidying up (#1944) * Fixing nav logo * Adding emulate book for single renderer * enabling control * forcing hardware acceleration on infinite scroller * Fixing review comments - Refactoring bookshadow to use shared css - Adding issue comment for hardware acceleration hack * Fixing bookshadow happening on first and last page * Bump versions by dotnet-bump-version. * Release Testing Day 3 (#1946) * Removed extra trace messages as the people issue might have been resolved. * When registering, disable button until form is valid. Allow non-email formatted emails, but not blank. * Fixed opds not having http(s):// * Added a new API to allow scanning all libraries from end point * Moved Bookmarks directory to Media tab * Fixed an edge case for finding next chapter when we had volume 1,2 etc but they had the same chapter number. * Code cleanup * Bump versions by dotnet-bump-version. * Release Testing Day 3 (#1951) * Code cleanup. Fixed OPDS images missing api key. Fixed theme color on site manifest not being black. * Removed a console.log from timeago pipe * Reading list page is now alphabetical and the modal for adding to a reading list is ordered by most recent. * Fixed a bug where remove read from reading list failed due to Calculating Start and End date assuming chapter would always be there. * Fixed a bug where reading list cover would get reset when editing the reading list. * Fixed a bug where reading list item didn't have not read badge. It's on old style. * Fixed a bug where user-preferences was hitting an admin only api when there was a better alternative * Slight memory improvement on a common db call * Fixed a bug where resetting to default theme when a theme was deleted was throwing an exception and failing. * All Login dtos now have the active KavitaVersion to make external apps able to handle what version of the API they are connecting with. * Fixed up a case where getVolume repo method always assumed there was a volume by that Id. * Bump versions by dotnet-bump-version. * Release Testing Final Stretch (#1952) * Fixed a bug breaking ability to save server settings * Fixed a case where if a user updated appsetting.json, the DB wouldn't update to stay in alignment. * Bump versions by dotnet-bump-version. * v0.7.2 - Reading Lists, CBL Support, OPDS Flattening and More! (#1953) * Version bump * Fixed scan all libraries not doing it on background thread from request. * Bump versions by dotnet-bump-version. * Version correction * Fixed the merge --------- Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> Co-authored-by: Mike <github@emailisgood.com> Co-authored-by: Muggz <mug@passw0rd.org> Co-authored-by: Domenic Fiore <DomenicF@users.noreply.github.com> Co-authored-by: Kupferhirn <kupferhirn@brokensoft.net> Co-authored-by: Snd-R <76580768+Snd-R@users.noreply.github.com> Co-authored-by: CKolle <115696142+CKolle@users.noreply.github.com> Co-authored-by: Gazy Mahomar <gmahomarf@users.noreply.github.com> Co-authored-by: Chris Plaatjes <kizaing@gmail.com> Co-authored-by: Andy <mrclock9090@yahoo.com>
		
			
				
	
	
		
			1240 lines
		
	
	
		
			45 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			1240 lines
		
	
	
		
			45 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.Linq;
 | |
| using System.Threading.Tasks;
 | |
| using API.Data;
 | |
| using API.Data.Repositories;
 | |
| using API.DTOs;
 | |
| using API.DTOs.CollectionTags;
 | |
| using API.DTOs.Metadata;
 | |
| using API.DTOs.SeriesDetail;
 | |
| using API.Entities;
 | |
| using API.Entities.Enums;
 | |
| using API.Entities.Metadata;
 | |
| using API.Extensions;
 | |
| using API.Helpers.Builders;
 | |
| using API.Services;
 | |
| using API.SignalR;
 | |
| using API.Tests.Helpers;
 | |
| using Microsoft.Extensions.Logging;
 | |
| using NSubstitute;
 | |
| using Xunit;
 | |
| 
 | |
| namespace API.Tests.Services;
 | |
| 
 | |
| public class SeriesServiceTests : AbstractDbTest
 | |
| {
 | |
|     private readonly ISeriesService _seriesService;
 | |
| 
 | |
|     public SeriesServiceTests() : base()
 | |
|     {
 | |
|         _seriesService = new SeriesService(_unitOfWork, Substitute.For<IEventHub>(),
 | |
|             Substitute.For<ITaskScheduler>(), Substitute.For<ILogger<SeriesService>>());
 | |
|     }
 | |
|     #region Setup
 | |
| 
 | |
|     protected override async Task ResetDb()
 | |
|     {
 | |
|         _context.Series.RemoveRange(_context.Series.ToList());
 | |
|         _context.AppUserRating.RemoveRange(_context.AppUserRating.ToList());
 | |
|         _context.Genre.RemoveRange(_context.Genre.ToList());
 | |
|         _context.CollectionTag.RemoveRange(_context.CollectionTag.ToList());
 | |
|         _context.Person.RemoveRange(_context.Person.ToList());
 | |
|         _context.Library.RemoveRange(_context.Library.ToList());
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
|     }
 | |
| 
 | |
|     private static UpdateRelatedSeriesDto CreateRelationsDto(Series series)
 | |
|     {
 | |
|         return new UpdateRelatedSeriesDto()
 | |
|         {
 | |
|             SeriesId = series.Id,
 | |
|             Prequels = new List<int>(),
 | |
|             Adaptations = new List<int>(),
 | |
|             Characters = new List<int>(),
 | |
|             Contains = new List<int>(),
 | |
|             Doujinshis = new List<int>(),
 | |
|             Others = new List<int>(),
 | |
|             Sequels = new List<int>(),
 | |
|             AlternativeSettings = new List<int>(),
 | |
|             AlternativeVersions = new List<int>(),
 | |
|             SideStories = new List<int>(),
 | |
|             SpinOffs = new List<int>(),
 | |
|             Editions = new List<int>()
 | |
|         };
 | |
|     }
 | |
| 
 | |
|     #endregion
 | |
| 
 | |
|     #region SeriesDetail
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task SeriesDetail_ShouldReturnSpecials()
 | |
|     {
 | |
|         await ResetDb();
 | |
| 
 | |
|         _context.Library.Add(new LibraryBuilder("Test LIb")
 | |
|             .WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
 | |
|             .WithSeries(new SeriesBuilder("Test")
 | |
| 
 | |
|                 .WithVolume(new VolumeBuilder("0")
 | |
|                     .WithChapter(new ChapterBuilder("Omake").WithIsSpecial(true).WithTitle("Omake").WithPages(1).Build())
 | |
|                     .WithChapter(new ChapterBuilder("Something SP02").WithIsSpecial(true).WithTitle("Something").WithPages(1).Build())
 | |
|                     .Build())
 | |
|                 .WithVolume(new VolumeBuilder("2")
 | |
|                     .WithChapter(new ChapterBuilder("21").WithPages(1).Build())
 | |
|                     .WithChapter(new ChapterBuilder("21").WithPages(1).Build())
 | |
|                     .Build())
 | |
| 
 | |
|                 .WithVolume(new VolumeBuilder("3")
 | |
|                     .WithChapter(new ChapterBuilder("31").WithPages(1).Build())
 | |
|                     .WithChapter(new ChapterBuilder("32").WithPages(1).Build())
 | |
|                     .Build())
 | |
|                 .Build())
 | |
|             .Build());
 | |
| 
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var expectedRanges = new[] {"Omake", "Something SP02"};
 | |
| 
 | |
|         var detail = await _seriesService.GetSeriesDetail(1, 1);
 | |
|         Assert.NotEmpty(detail.Specials);
 | |
|         Assert.True(2 == detail.Specials.Count());
 | |
|         Assert.All(detail.Specials, dto => Assert.Contains(dto.Range, expectedRanges));
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task SeriesDetail_ShouldReturnVolumesAndChapters()
 | |
|     {
 | |
|         await ResetDb();
 | |
| 
 | |
|         _context.Library.Add(new LibraryBuilder("Test LIb")
 | |
|             .WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
 | |
|             .WithSeries(new SeriesBuilder("Test")
 | |
| 
 | |
|                 .WithVolume(new VolumeBuilder("0")
 | |
|                     .WithChapter(new ChapterBuilder("1").WithPages(1).Build())
 | |
|                     .WithChapter(new ChapterBuilder("2").WithPages(1).Build())
 | |
|                     .Build())
 | |
|                 .WithVolume(new VolumeBuilder("2")
 | |
|                     .WithChapter(new ChapterBuilder("21").WithPages(1).Build())
 | |
|                     .WithChapter(new ChapterBuilder("21").WithPages(1).Build())
 | |
|                     .Build())
 | |
| 
 | |
|                 .WithVolume(new VolumeBuilder("3")
 | |
|                     .WithChapter(new ChapterBuilder("31").WithPages(1).Build())
 | |
|                     .WithChapter(new ChapterBuilder("32").WithPages(1).Build())
 | |
|                     .Build())
 | |
|                 .Build())
 | |
|             .Build()
 | |
|         );
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var detail = await _seriesService.GetSeriesDetail(1, 1);
 | |
|         Assert.NotEmpty(detail.Chapters);
 | |
|         Assert.Equal(6, detail.Chapters.Count());
 | |
| 
 | |
|         Assert.NotEmpty(detail.Volumes);
 | |
|         Assert.Equal(2, detail.Volumes.Count()); // Volume 0 shouldn't be sent in Volumes
 | |
|         Assert.All(detail.Volumes, dto => Assert.Contains(dto.Name, new[] {"Volume 2", "Volume 3"})); // Volumes get names mapped
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task SeriesDetail_ShouldReturnVolumesAndChapters_ButRemove0Chapter()
 | |
|     {
 | |
|         await ResetDb();
 | |
| 
 | |
|         _context.Library.Add(new LibraryBuilder("Test LIb")
 | |
|             .WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
 | |
|             .WithSeries(new SeriesBuilder("Test")
 | |
| 
 | |
|                 .WithVolume(new VolumeBuilder("0")
 | |
|                     .WithChapter(new ChapterBuilder("1").WithPages(1).Build())
 | |
|                     .WithChapter(new ChapterBuilder("2").WithPages(1).Build())
 | |
|                     .Build())
 | |
|                 .WithVolume(new VolumeBuilder("2")
 | |
|                     .WithChapter(new ChapterBuilder("0").WithPages(1).Build())
 | |
|                     .Build())
 | |
| 
 | |
|                 .WithVolume(new VolumeBuilder("3")
 | |
|                     .WithChapter(new ChapterBuilder("31").WithPages(1).Build())
 | |
|                     .Build())
 | |
|                 .Build())
 | |
|             .Build());
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var detail = await _seriesService.GetSeriesDetail(1, 1);
 | |
|         Assert.NotEmpty(detail.Chapters);
 | |
|         // volume 2 has a 0 chapter aka a single chapter that is represented as a volume. We don't show in Chapters area
 | |
|         Assert.Equal(3, detail.Chapters.Count());
 | |
| 
 | |
|         Assert.NotEmpty(detail.Volumes);
 | |
|         Assert.Equal(2, detail.Volumes.Count());
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task SeriesDetail_ShouldReturnCorrectNaming_VolumeTitle()
 | |
|     {
 | |
|         await ResetDb();
 | |
| 
 | |
|         _context.Library.Add(new LibraryBuilder("Test LIb")
 | |
|             .WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
 | |
|             .WithSeries(new SeriesBuilder("Test")
 | |
|                 .WithVolume(new VolumeBuilder("0")
 | |
|                     .WithChapter(new ChapterBuilder("1").WithPages(1).Build())
 | |
|                     .WithChapter(new ChapterBuilder("2").WithPages(1).Build())
 | |
|                     .Build())
 | |
|                 .WithVolume(new VolumeBuilder("2")
 | |
|                     .WithChapter(new ChapterBuilder("0").WithPages(1).Build())
 | |
|                     .Build())
 | |
| 
 | |
|                 .WithVolume(new VolumeBuilder("3")
 | |
|                     .WithChapter(new ChapterBuilder("31").WithPages(1).Build())
 | |
|                     .Build())
 | |
|                 .Build())
 | |
|             .Build());
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var detail = await _seriesService.GetSeriesDetail(1, 1);
 | |
|         Assert.NotEmpty(detail.Chapters);
 | |
|         // volume 2 has a 0 chapter aka a single chapter that is represented as a volume. We don't show in Chapters area
 | |
|         Assert.Equal(3, detail.Chapters.Count());
 | |
| 
 | |
|         Assert.NotEmpty(detail.Volumes);
 | |
|         Assert.Equal(2, detail.Volumes.Count());
 | |
| 
 | |
|         Assert.Equal(string.Empty, detail.Chapters.First().VolumeTitle); // loose leaf chapter
 | |
|         Assert.Equal("Volume 3", detail.Chapters.Last().VolumeTitle); // volume based chapter
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task SeriesDetail_ShouldReturnChaptersOnly_WhenBookLibrary()
 | |
|     {
 | |
|         await ResetDb();
 | |
| 
 | |
|         _context.Library.Add(new LibraryBuilder("Test LIb", LibraryType.Book)
 | |
|             .WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
 | |
|             .WithSeries(new SeriesBuilder("Test")
 | |
| 
 | |
|                 .WithVolume(new VolumeBuilder("2")
 | |
|                     .WithChapter(new ChapterBuilder("0").WithPages(1).Build())
 | |
|                     .Build())
 | |
| 
 | |
|                 .WithVolume(new VolumeBuilder("3")
 | |
|                     .WithChapter(new ChapterBuilder("0").WithPages(1).Build())
 | |
|                     .Build())
 | |
|                 .Build())
 | |
|             .Build());
 | |
| 
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var detail = await _seriesService.GetSeriesDetail(1, 1);
 | |
|         Assert.NotEmpty(detail.Volumes);
 | |
| 
 | |
|         Assert.Empty(detail.Chapters); // A book library where all books are Volumes, will show no "chapters" on the UI because it doesn't make sense
 | |
|         Assert.Equal(2, detail.Volumes.Count());
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task SeriesDetail_WhenBookLibrary_ShouldReturnVolumesAndSpecial()
 | |
|     {
 | |
|         await ResetDb();
 | |
| 
 | |
|         _context.Library.Add(new LibraryBuilder("Test LIb", LibraryType.Book)
 | |
|             .WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
 | |
|             .WithSeries(new SeriesBuilder("Test")
 | |
| 
 | |
|                 .WithVolume(new VolumeBuilder("0")
 | |
|                     .WithChapter(new ChapterBuilder("Ano Orokamono ni mo Kyakkou wo! - Volume 1.epub", "Ano Orokamono ni mo Kyakkou wo! - Volume 1.epub").WithIsSpecial(true).WithPages(1).Build())
 | |
|                     .Build())
 | |
|                 .WithVolume(new VolumeBuilder("2")
 | |
|                     .WithChapter(new ChapterBuilder("Ano Orokamono ni mo Kyakkou wo! - Volume 2.epub", "Ano Orokamono ni mo Kyakkou wo! - Volume 2.epub").WithPages(1).Build())
 | |
|                     .Build())
 | |
|                 .Build())
 | |
|             .Build());
 | |
| 
 | |
| 
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var detail = await _seriesService.GetSeriesDetail(1, 1);
 | |
|         Assert.NotEmpty(detail.Volumes);
 | |
|         Assert.Equal("2 - Ano Orokamono ni mo Kyakkou wo! - Volume 2", detail.Volumes.ElementAt(0).Name);
 | |
| 
 | |
|         Assert.NotEmpty(detail.Specials);
 | |
|         Assert.Equal("Ano Orokamono ni mo Kyakkou wo! - Volume 1.epub", detail.Specials.ElementAt(0).Range);
 | |
| 
 | |
|         // A book library where all books are Volumes, will show no "chapters" on the UI because it doesn't make sense
 | |
|         Assert.Empty(detail.Chapters);
 | |
| 
 | |
|         Assert.Equal(1, detail.Volumes.Count());
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task SeriesDetail_ShouldSortVolumesByName()
 | |
|     {
 | |
|         await ResetDb();
 | |
| 
 | |
|         _context.Library.Add(new LibraryBuilder("Test LIb", LibraryType.Manga)
 | |
|             .WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
 | |
|             .WithSeries(new SeriesBuilder("Test")
 | |
| 
 | |
|                 .WithVolume(new VolumeBuilder("2")
 | |
|                     .WithChapter(new ChapterBuilder("0").WithPages(1).Build())
 | |
|                     .Build())
 | |
| 
 | |
|                 .WithVolume(new VolumeBuilder("1.2")
 | |
|                     .WithChapter(new ChapterBuilder("0").WithPages(1).Build())
 | |
|                     .Build())
 | |
|                 .WithVolume(new VolumeBuilder("1")
 | |
|                     .WithChapter(new ChapterBuilder("0").WithPages(1).Build())
 | |
|                     .Build())
 | |
|                 .Build())
 | |
|             .Build());
 | |
| 
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var detail = await _seriesService.GetSeriesDetail(1, 1);
 | |
|         Assert.Equal("Volume 1", detail.Volumes.ElementAt(0).Name);
 | |
|         Assert.Equal("Volume 1.2", detail.Volumes.ElementAt(1).Name);
 | |
|         Assert.Equal("Volume 2", detail.Volumes.ElementAt(2).Name);
 | |
|     }
 | |
| 
 | |
| 
 | |
|     #endregion
 | |
| 
 | |
| 
 | |
|     #region UpdateRating
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task UpdateRating_ShouldSetRating()
 | |
|     {
 | |
|         await ResetDb();
 | |
| 
 | |
|         _context.Library.Add(new LibraryBuilder("Test LIb", LibraryType.Manga)
 | |
|             .WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
 | |
|             .WithSeries(new SeriesBuilder("Test")
 | |
| 
 | |
|                 .WithVolume(new VolumeBuilder("1")
 | |
|                     .WithChapter(new ChapterBuilder("1").WithPages(1).Build())
 | |
|                     .Build())
 | |
|                 .Build())
 | |
|             .Build());
 | |
| 
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
| 
 | |
|         var user = await _unitOfWork.UserRepository.GetUserByUsernameAsync("majora2007", AppUserIncludes.Ratings);
 | |
| 
 | |
|         var result = await _seriesService.UpdateRating(user, new UpdateSeriesRatingDto()
 | |
|         {
 | |
|             SeriesId = 1,
 | |
|             UserRating = 3,
 | |
|             UserReview = "Average"
 | |
|         });
 | |
| 
 | |
|         Assert.True(result);
 | |
| 
 | |
|         var ratings = (await _unitOfWork.UserRepository.GetUserByUsernameAsync("majora2007", AppUserIncludes.Ratings))
 | |
|             .Ratings;
 | |
|         Assert.NotEmpty(ratings);
 | |
|         Assert.Equal(3, ratings.First().Rating);
 | |
|         Assert.Equal("Average", ratings.First().Review);
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task UpdateRating_ShouldUpdateExistingRating()
 | |
|     {
 | |
|         await ResetDb();
 | |
| 
 | |
|         _context.Library.Add(new LibraryBuilder("Test LIb")
 | |
|             .WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
 | |
|             .WithSeries(new SeriesBuilder("Test")
 | |
| 
 | |
|                 .WithVolume(new VolumeBuilder("1")
 | |
|                     .WithChapter(new ChapterBuilder("1").WithPages(1).Build())
 | |
|                     .Build())
 | |
|                 .Build())
 | |
|             .Build());
 | |
| 
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var user = await _unitOfWork.UserRepository.GetUserByUsernameAsync("majora2007", AppUserIncludes.Ratings);
 | |
| 
 | |
|         var result = await _seriesService.UpdateRating(user, new UpdateSeriesRatingDto()
 | |
|         {
 | |
|             SeriesId = 1,
 | |
|             UserRating = 3,
 | |
|             UserReview = "Average"
 | |
|         });
 | |
| 
 | |
|         Assert.True(result);
 | |
| 
 | |
|         var ratings = (await _unitOfWork.UserRepository.GetUserByUsernameAsync("majora2007", AppUserIncludes.Ratings))
 | |
|             .Ratings;
 | |
|         Assert.NotEmpty(ratings);
 | |
|         Assert.Equal(3, ratings.First().Rating);
 | |
|         Assert.Equal("Average", ratings.First().Review);
 | |
| 
 | |
|         // Update the DB again
 | |
| 
 | |
|         var result2 = await _seriesService.UpdateRating(user, new UpdateSeriesRatingDto()
 | |
|         {
 | |
|             SeriesId = 1,
 | |
|             UserRating = 5,
 | |
|             UserReview = "Average"
 | |
|         });
 | |
| 
 | |
|         Assert.True(result2);
 | |
| 
 | |
|         var ratings2 = (await _unitOfWork.UserRepository.GetUserByUsernameAsync("majora2007", AppUserIncludes.Ratings))
 | |
|             .Ratings;
 | |
|         Assert.NotEmpty(ratings2);
 | |
|         Assert.True(ratings2.Count == 1);
 | |
|         Assert.Equal(5, ratings2.First().Rating);
 | |
|         Assert.Equal("Average", ratings2.First().Review);
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task UpdateRating_ShouldClampRatingAt5()
 | |
|     {
 | |
|         await ResetDb();
 | |
| 
 | |
|         _context.Library.Add(new LibraryBuilder("Test LIb")
 | |
|             .WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
 | |
|             .WithSeries(new SeriesBuilder("Test")
 | |
| 
 | |
|                 .WithVolume(new VolumeBuilder("1")
 | |
|                     .WithChapter(new ChapterBuilder("1").WithPages(1).Build())
 | |
|                     .Build())
 | |
|                 .Build())
 | |
|             .Build());
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var user = await _unitOfWork.UserRepository.GetUserByUsernameAsync("majora2007", AppUserIncludes.Ratings);
 | |
| 
 | |
|         var result = await _seriesService.UpdateRating(user, new UpdateSeriesRatingDto()
 | |
|         {
 | |
|             SeriesId = 1,
 | |
|             UserRating = 10,
 | |
|             UserReview = "Average"
 | |
|         });
 | |
| 
 | |
|         Assert.True(result);
 | |
| 
 | |
|         var ratings = (await _unitOfWork.UserRepository.GetUserByUsernameAsync("majora2007", AppUserIncludes.Ratings))
 | |
|             .Ratings;
 | |
|         Assert.NotEmpty(ratings);
 | |
|         Assert.Equal(5, ratings.First().Rating);
 | |
|         Assert.Equal("Average", ratings.First().Review);
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task UpdateRating_ShouldReturnFalseWhenSeriesDoesntExist()
 | |
|     {
 | |
|         await ResetDb();
 | |
| 
 | |
|         _context.Library.Add(new LibraryBuilder("Test LIb", LibraryType.Book)
 | |
|             .WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
 | |
|             .WithSeries(new SeriesBuilder("Test")
 | |
| 
 | |
|                 .WithVolume(new VolumeBuilder("1")
 | |
|                     .WithChapter(new ChapterBuilder("1").WithPages(1).Build())
 | |
|                     .Build())
 | |
|                 .Build())
 | |
|             .Build());
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var user = await _unitOfWork.UserRepository.GetUserByUsernameAsync("majora2007", AppUserIncludes.Ratings);
 | |
| 
 | |
|         var result = await _seriesService.UpdateRating(user, new UpdateSeriesRatingDto()
 | |
|         {
 | |
|             SeriesId = 2,
 | |
|             UserRating = 5,
 | |
|             UserReview = "Average"
 | |
|         });
 | |
| 
 | |
|         Assert.False(result);
 | |
| 
 | |
|         var ratings = user.Ratings;
 | |
|         Assert.Empty(ratings);
 | |
|     }
 | |
| 
 | |
|     #endregion
 | |
| 
 | |
|     #region UpdateSeriesMetadata
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task UpdateSeriesMetadata_ShouldCreateEmptyMetadata_IfDoesntExist()
 | |
|     {
 | |
|         await ResetDb();
 | |
|         var s = new SeriesBuilder("Test")
 | |
|             .Build();
 | |
|         s.Library = new LibraryBuilder("Test LIb", LibraryType.Book).Build();
 | |
| 
 | |
|         _context.Series.Add(s);
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var success = await _seriesService.UpdateSeriesMetadata(new UpdateSeriesMetadataDto()
 | |
|         {
 | |
|             SeriesMetadata = new SeriesMetadataDto()
 | |
|             {
 | |
|                 SeriesId = 1,
 | |
|                 Genres = new List<GenreTagDto>() {new GenreTagDto() {Id = 0, Title = "New Genre"}}
 | |
|             },
 | |
|             CollectionTags = new List<CollectionTagDto>()
 | |
|         });
 | |
| 
 | |
|         Assert.True(success);
 | |
| 
 | |
|         var series = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1);
 | |
|         Assert.NotNull(series.Metadata);
 | |
|         Assert.True(series.Metadata.Genres.Select(g => g.Title).Contains("New Genre".SentenceCase()));
 | |
| 
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task UpdateSeriesMetadata_ShouldCreateNewTags_IfNoneExist()
 | |
|     {
 | |
|         await ResetDb();
 | |
|         var s = new SeriesBuilder("Test")
 | |
|             .Build();
 | |
|         s.Library = new LibraryBuilder("Test LIb", LibraryType.Book).Build();
 | |
| 
 | |
|         _context.Series.Add(s);
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var success = await _seriesService.UpdateSeriesMetadata(new UpdateSeriesMetadataDto()
 | |
|         {
 | |
|             SeriesMetadata = new SeriesMetadataDto()
 | |
|             {
 | |
|                 SeriesId = 1,
 | |
|                 Genres = new List<GenreTagDto>() {new GenreTagDto() {Id = 0, Title = "New Genre"}},
 | |
|                 Tags = new List<TagDto>() {new TagDto() {Id = 0, Title = "New Tag"}},
 | |
|                 Characters = new List<PersonDto>() {new PersonDto() {Id = 0, Name = "Joe Shmo", Role = PersonRole.Character}},
 | |
|                 Colorists = new List<PersonDto>() {new PersonDto() {Id = 0, Name = "Joe Shmo", Role = PersonRole.Colorist}},
 | |
|                 Pencillers = new List<PersonDto>() {new PersonDto() {Id = 0, Name = "Joe Shmo 2", Role = PersonRole.Penciller}},
 | |
|             },
 | |
|             CollectionTags = new List<CollectionTagDto>()
 | |
|             {
 | |
|                 new CollectionTagDto() {Id = 0, Promoted = false, Summary = string.Empty, CoverImageLocked = false, Title = "New Collection"}
 | |
|             }
 | |
|         });
 | |
| 
 | |
|         Assert.True(success);
 | |
| 
 | |
|         var series = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1);
 | |
|         Assert.NotNull(series.Metadata);
 | |
|         Assert.True(series.Metadata.Genres.Select(g => g.Title).Contains("New Genre".SentenceCase()));
 | |
|         Assert.True(series.Metadata.People.All(g => g.Name is "Joe Shmo" or "Joe Shmo 2"));
 | |
|         Assert.True(series.Metadata.Tags.Select(g => g.Title).Contains("New Tag".SentenceCase()));
 | |
|         Assert.True(series.Metadata.CollectionTags.Select(g => g.Title).Contains("New Collection"));
 | |
| 
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task UpdateSeriesMetadata_ShouldRemoveExistingTags()
 | |
|     {
 | |
|         await ResetDb();
 | |
|         var s = new SeriesBuilder("Test")
 | |
|             .WithMetadata(new SeriesMetadataBuilder().Build())
 | |
|             .Build();
 | |
|         s.Library = new LibraryBuilder("Test LIb", LibraryType.Book).Build();
 | |
| 
 | |
|         var g = new GenreBuilder("Existing Genre").Build();
 | |
|         s.Metadata.Genres = new List<Genre>() {g};
 | |
|         _context.Series.Add(s);
 | |
| 
 | |
|         _context.Genre.Add(g);
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var success = await _seriesService.UpdateSeriesMetadata(new UpdateSeriesMetadataDto()
 | |
|         {
 | |
|             SeriesMetadata = new SeriesMetadataDto()
 | |
|             {
 | |
|                 SeriesId = 1,
 | |
|                 Genres = new List<GenreTagDto>() {new () {Id = 0, Title = "New Genre"}},
 | |
|             },
 | |
|             CollectionTags = new List<CollectionTagDto>()
 | |
|         });
 | |
| 
 | |
|         Assert.True(success);
 | |
| 
 | |
|         var series = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1);
 | |
|         Assert.NotNull(series.Metadata);
 | |
|         Assert.True(series.Metadata.Genres.Select(g1 => g1.Title).All(g2 => g2 == "New Genre".SentenceCase()));
 | |
|         Assert.False(series.Metadata.GenresLocked); // GenreLocked is false unless the UI Explicitly says it should be locked
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task UpdateSeriesMetadata_ShouldAddNewPerson_NoExistingPeople()
 | |
|     {
 | |
|         await ResetDb();
 | |
|         var s = new SeriesBuilder("Test")
 | |
|             .WithMetadata(new SeriesMetadataBuilder().Build())
 | |
|             .Build();
 | |
|         s.Library = new LibraryBuilder("Test LIb", LibraryType.Book).Build();
 | |
| 
 | |
|         var g = new PersonBuilder("Existing Person", PersonRole.Publisher).Build();
 | |
|         _context.Series.Add(s);
 | |
| 
 | |
|         _context.Person.Add(g);
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var success = await _seriesService.UpdateSeriesMetadata(new UpdateSeriesMetadataDto()
 | |
|         {
 | |
|             SeriesMetadata = new SeriesMetadataDto()
 | |
|             {
 | |
|                 SeriesId = 1,
 | |
|                 Publishers = new List<PersonDto>() {new () {Id = 0, Name = "Existing Person", Role = PersonRole.Publisher}},
 | |
|             },
 | |
|             CollectionTags = new List<CollectionTagDto>()
 | |
|         });
 | |
| 
 | |
|         Assert.True(success);
 | |
| 
 | |
|         var series = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1);
 | |
|         Assert.NotNull(series.Metadata);
 | |
|         Assert.True(series.Metadata.People.Select(g => g.Name).All(g => g == "Existing Person"));
 | |
|         Assert.False(series.Metadata.PublisherLocked); // PublisherLocked is false unless the UI Explicitly says it should be locked
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task UpdateSeriesMetadata_ShouldAddNewPerson_ExistingPeople()
 | |
|     {
 | |
|         await ResetDb();
 | |
|         var s = new SeriesBuilder("Test")
 | |
|             .WithMetadata(new SeriesMetadataBuilder().Build())
 | |
|             .Build();
 | |
|         s.Library = new LibraryBuilder("Test LIb", LibraryType.Book).Build();
 | |
|         var g = new PersonBuilder("Existing Person", PersonRole.Publisher).Build();
 | |
|         s.Metadata.People = new List<Person>() {new PersonBuilder("Existing Writer", PersonRole.Writer).Build(),
 | |
|             new PersonBuilder("Existing Translator", PersonRole.Translator).Build(), new PersonBuilder("Existing Publisher 2", PersonRole.Publisher).Build()};
 | |
|         _context.Series.Add(s);
 | |
| 
 | |
|         _context.Person.Add(g);
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var success = await _seriesService.UpdateSeriesMetadata(new UpdateSeriesMetadataDto()
 | |
|         {
 | |
|             SeriesMetadata = new SeriesMetadataDto()
 | |
|             {
 | |
|                 SeriesId = 1,
 | |
|                 Publishers = new List<PersonDto>() {new () {Id = 0, Name = "Existing Person", Role = PersonRole.Publisher}},
 | |
|                 PublishersLocked = true
 | |
|             },
 | |
|             CollectionTags = new List<CollectionTagDto>()
 | |
|         });
 | |
| 
 | |
|         Assert.True(success);
 | |
| 
 | |
|         var series = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1);
 | |
|         Assert.NotNull(series.Metadata);
 | |
|         Assert.True(series.Metadata.People.Select(g => g.Name).All(g => g == "Existing Person"));
 | |
|         Assert.True(series.Metadata.PublisherLocked);
 | |
|     }
 | |
| 
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task UpdateSeriesMetadata_ShouldRemoveExistingPerson()
 | |
|     {
 | |
|         await ResetDb();
 | |
|         var s = new SeriesBuilder("Test")
 | |
|             .WithMetadata(new SeriesMetadataBuilder().Build())
 | |
|             .Build();
 | |
|         s.Library = new LibraryBuilder("Test LIb", LibraryType.Book).Build();
 | |
|         var g = new PersonBuilder("Existing Person", PersonRole.Publisher).Build();
 | |
|         _context.Series.Add(s);
 | |
| 
 | |
|         _context.Person.Add(g);
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var success = await _seriesService.UpdateSeriesMetadata(new UpdateSeriesMetadataDto()
 | |
|         {
 | |
|             SeriesMetadata = new SeriesMetadataDto()
 | |
|             {
 | |
|                 SeriesId = 1,
 | |
|                 Publishers = new List<PersonDto>() {},
 | |
|             },
 | |
|             CollectionTags = new List<CollectionTagDto>()
 | |
|         });
 | |
| 
 | |
|         Assert.True(success);
 | |
| 
 | |
|         var series = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1);
 | |
|         Assert.NotNull(series.Metadata);
 | |
|         Assert.False(series.Metadata.People.Any());
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task UpdateSeriesMetadata_ShouldLockIfTold()
 | |
|     {
 | |
|         await ResetDb();
 | |
|         var s = new SeriesBuilder("Test")
 | |
|             .WithMetadata(new SeriesMetadataBuilder().Build())
 | |
|             .Build();
 | |
|         s.Library = new LibraryBuilder("Test LIb", LibraryType.Book).Build();
 | |
|         var g = new GenreBuilder("Existing Genre").Build();
 | |
|         s.Metadata.Genres = new List<Genre>() {g};
 | |
|         s.Metadata.GenresLocked = true;
 | |
|         _context.Series.Add(s);
 | |
| 
 | |
|         _context.Genre.Add(g);
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var success = await _seriesService.UpdateSeriesMetadata(new UpdateSeriesMetadataDto()
 | |
|         {
 | |
|             SeriesMetadata = new SeriesMetadataDto()
 | |
|             {
 | |
|                 SeriesId = 1,
 | |
|                 Genres = new List<GenreTagDto>() {new () {Id = 1, Title = "Existing Genre"}},
 | |
|                 GenresLocked = true
 | |
|             },
 | |
|             CollectionTags = new List<CollectionTagDto>()
 | |
|         });
 | |
| 
 | |
|         Assert.True(success);
 | |
| 
 | |
|         var series = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1);
 | |
|         Assert.NotNull(series.Metadata);
 | |
|         Assert.True(series.Metadata.Genres.Select(g => g.Title).All(g => g == "Existing Genre".SentenceCase()));
 | |
|         Assert.True(series.Metadata.GenresLocked);
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task UpdateSeriesMetadata_ShouldNotUpdateReleaseYear_IfLessThan1000()
 | |
|     {
 | |
|         await ResetDb();
 | |
|         var s = new SeriesBuilder("Test")
 | |
|             .WithMetadata(new SeriesMetadataBuilder().Build())
 | |
|             .Build();
 | |
|         s.Library = new LibraryBuilder("Test LIb", LibraryType.Book).Build();
 | |
|         _context.Series.Add(s);
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var success = await _seriesService.UpdateSeriesMetadata(new UpdateSeriesMetadataDto()
 | |
|         {
 | |
|             SeriesMetadata = new SeriesMetadataDto()
 | |
|             {
 | |
|                 SeriesId = 1,
 | |
|                 ReleaseYear = 100,
 | |
|             },
 | |
|             CollectionTags = new List<CollectionTagDto>()
 | |
|         });
 | |
| 
 | |
|         Assert.True(success);
 | |
| 
 | |
|         var series = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1);
 | |
|         Assert.NotNull(series.Metadata);
 | |
|         Assert.Equal(0, series.Metadata.ReleaseYear);
 | |
|         Assert.False(series.Metadata.ReleaseYearLocked);
 | |
|     }
 | |
| 
 | |
|     #endregion
 | |
| 
 | |
|     #region GetFirstChapterForMetadata
 | |
| 
 | |
|     private static Series CreateSeriesMock()
 | |
|     {
 | |
|         var file = new MangaFileBuilder("Test.cbz", MangaFormat.Archive, 1).Build();
 | |
| 
 | |
|         var series = new SeriesBuilder("Test")
 | |
|             .WithVolume(new VolumeBuilder("0")
 | |
|                 .WithChapter(new ChapterBuilder("95").WithPages(1).WithFile(file).Build())
 | |
|                 .WithChapter(new ChapterBuilder("96").WithPages(1).WithFile(file).Build())
 | |
|                 .WithChapter(new ChapterBuilder("A Special Case").WithIsSpecial(true).WithFile(file).WithPages(1).Build())
 | |
|                 .Build())
 | |
|             .WithVolume(new VolumeBuilder("1")
 | |
|                 .WithChapter(new ChapterBuilder("1").WithPages(1).WithFile(file).Build())
 | |
|                 .WithChapter(new ChapterBuilder("2").WithPages(1).WithFile(file).Build())
 | |
|                 .Build())
 | |
| 
 | |
|             .WithVolume(new VolumeBuilder("2")
 | |
|                 .WithChapter(new ChapterBuilder("21").WithPages(1).WithFile(file).Build())
 | |
|                 .WithChapter(new ChapterBuilder("22").WithPages(1).WithFile(file).Build())
 | |
|                 .Build())
 | |
| 
 | |
|             .WithVolume(new VolumeBuilder("3")
 | |
|                 .WithChapter(new ChapterBuilder("31").WithPages(1).WithFile(file).Build())
 | |
|                 .WithChapter(new ChapterBuilder("32").WithPages(1).WithFile(file).Build())
 | |
|                 .Build())
 | |
|             .Build();
 | |
|         series.Library = new LibraryBuilder("Test LIb", LibraryType.Book).Build();
 | |
| 
 | |
|         return series;
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public void GetFirstChapterForMetadata_Book_Test()
 | |
|     {
 | |
|         var series = CreateSeriesMock();
 | |
| 
 | |
|         var firstChapter = SeriesService.GetFirstChapterForMetadata(series, true);
 | |
|         Assert.Same("1", firstChapter.Range);
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public void GetFirstChapterForMetadata_NonBook_ShouldReturnVolume1()
 | |
|     {
 | |
|         var series = CreateSeriesMock();
 | |
| 
 | |
|         var firstChapter = SeriesService.GetFirstChapterForMetadata(series, false);
 | |
|         Assert.Same("1", firstChapter.Range);
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public void GetFirstChapterForMetadata_NonBook_ShouldReturnVolume1_WhenFirstChapterIsFloat()
 | |
|     {
 | |
|         var series = CreateSeriesMock();
 | |
|         var files = new List<MangaFile>()
 | |
|         {
 | |
|             new MangaFileBuilder("Test.cbz", MangaFormat.Archive, 1).Build()
 | |
|         };
 | |
|         series.Volumes[1].Chapters = new List<Chapter>()
 | |
|         {
 | |
|             new ChapterBuilder("2").WithFiles(files).WithPages(1).Build(),
 | |
|             new ChapterBuilder("1.1").WithFiles(files).WithPages(1).Build(),
 | |
|             new ChapterBuilder("1.2").WithFiles(files).WithPages(1).Build(),
 | |
|         };
 | |
| 
 | |
|         var firstChapter = SeriesService.GetFirstChapterForMetadata(series, false);
 | |
|         Assert.Same("1.1", firstChapter.Range);
 | |
|     }
 | |
| 
 | |
|     #endregion
 | |
| 
 | |
|     #region SeriesRelation
 | |
|     [Fact]
 | |
|     public async Task UpdateRelatedSeries_ShouldAddAllRelations()
 | |
|     {
 | |
|         await ResetDb();
 | |
|         _context.Library.Add(new Library()
 | |
|         {
 | |
|             AppUsers = new List<AppUser>()
 | |
|             {
 | |
|                 new AppUser()
 | |
|                 {
 | |
|                     UserName = "majora2007"
 | |
|                 }
 | |
|             },
 | |
|             Name = "Test LIb",
 | |
|             Type = LibraryType.Book,
 | |
|             Series = new List<Series>()
 | |
|             {
 | |
|                 new SeriesBuilder("Test Series").Build(),
 | |
|                 new SeriesBuilder("Test Series Prequels").Build(),
 | |
|                 new SeriesBuilder("Test Series Sequels").Build(),
 | |
|             }
 | |
|         });
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var series1 = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1, SeriesIncludes.Related);
 | |
|         // Add relations
 | |
|         var addRelationDto = CreateRelationsDto(series1);
 | |
|         addRelationDto.Adaptations.Add(2);
 | |
|         addRelationDto.Sequels.Add(3);
 | |
|         await _seriesService.UpdateRelatedSeries(addRelationDto);
 | |
|         Assert.Equal(2, series1.Relations.Single(s => s.TargetSeriesId == 2).TargetSeriesId);
 | |
|         Assert.Equal(3, series1.Relations.Single(s => s.TargetSeriesId == 3).TargetSeriesId);
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task UpdateRelatedSeries_DeleteAllRelations()
 | |
|     {
 | |
|         await ResetDb();
 | |
|         _context.Library.Add(new Library()
 | |
|         {
 | |
|             AppUsers = new List<AppUser>()
 | |
|             {
 | |
|                 new AppUser()
 | |
|                 {
 | |
|                     UserName = "majora2007"
 | |
|                 }
 | |
|             },
 | |
|             Name = "Test LIb",
 | |
|             Type = LibraryType.Book,
 | |
|             Series = new List<Series>()
 | |
|             {
 | |
|                 new SeriesBuilder("Test Series").Build(),
 | |
|                 new SeriesBuilder("Test Series Prequels").Build(),
 | |
|                 new SeriesBuilder("Test Series Sequels").Build(),
 | |
|             }
 | |
|         });
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var series1 = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1, SeriesIncludes.Related);
 | |
|         // Add relations
 | |
|         var addRelationDto = CreateRelationsDto(series1);
 | |
|         addRelationDto.Adaptations.Add(2);
 | |
|         addRelationDto.Sequels.Add(3);
 | |
|         await _seriesService.UpdateRelatedSeries(addRelationDto);
 | |
|         Assert.Equal(2, series1.Relations.Single(s => s.TargetSeriesId == 2).TargetSeriesId);
 | |
|         Assert.Equal(3, series1.Relations.Single(s => s.TargetSeriesId == 3).TargetSeriesId);
 | |
| 
 | |
|         // Remove relations
 | |
|         var removeRelationDto = CreateRelationsDto(series1);
 | |
|         await _seriesService.UpdateRelatedSeries(removeRelationDto);
 | |
|         Assert.Empty(series1.Relations.Where(s => s.TargetSeriesId == 1));
 | |
|         Assert.Empty(series1.Relations.Where(s => s.TargetSeriesId == 2));
 | |
|     }
 | |
| 
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task UpdateRelatedSeries_DeleteTargetSeries_ShouldSucceed()
 | |
|     {
 | |
|         await ResetDb();
 | |
|         _context.Library.Add(new Library()
 | |
|         {
 | |
|             AppUsers = new List<AppUser>()
 | |
|             {
 | |
|                 new AppUser()
 | |
|                 {
 | |
|                     UserName = "majora2007"
 | |
|                 }
 | |
|             },
 | |
|             Name = "Test LIb",
 | |
|             Type = LibraryType.Book,
 | |
|             Series = new List<Series>()
 | |
|             {
 | |
|                 new SeriesBuilder("Series A").Build(),
 | |
|                 new SeriesBuilder("Series B").Build(),
 | |
|             }
 | |
|         });
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var series1 = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1, SeriesIncludes.Related);
 | |
|         // Add relations
 | |
|         var addRelationDto = CreateRelationsDto(series1);
 | |
|         addRelationDto.Adaptations.Add(2);
 | |
|         await _seriesService.UpdateRelatedSeries(addRelationDto);
 | |
|         Assert.Equal(2, series1.Relations.Single(s => s.TargetSeriesId == 2).TargetSeriesId);
 | |
| 
 | |
|         _context.Series.Remove(await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(2));
 | |
|         try
 | |
|         {
 | |
|             await _context.SaveChangesAsync();
 | |
|         }
 | |
|         catch (Exception)
 | |
|         {
 | |
|             Assert.Fail("Delete of Target Series Failed");
 | |
|         }
 | |
| 
 | |
|         // Remove relations
 | |
|         Assert.Empty((await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1, SeriesIncludes.Related)).Relations);
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task UpdateRelatedSeries_DeleteSourceSeries_ShouldSucceed()
 | |
|     {
 | |
|         await ResetDb();
 | |
|         _context.Library.Add(new Library()
 | |
|         {
 | |
|             AppUsers = new List<AppUser>()
 | |
|             {
 | |
|                 new AppUser()
 | |
|                 {
 | |
|                     UserName = "majora2007"
 | |
|                 }
 | |
|             },
 | |
|             Name = "Test LIb",
 | |
|             Type = LibraryType.Book,
 | |
|             Series = new List<Series>()
 | |
|             {
 | |
|                 new SeriesBuilder("Series A").Build(),
 | |
|                 new SeriesBuilder("Series B").Build(),
 | |
|             }
 | |
|         });
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var series1 = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1, SeriesIncludes.Related);
 | |
|         // Add relations
 | |
|         var addRelationDto = CreateRelationsDto(series1);
 | |
|         addRelationDto.Adaptations.Add(2);
 | |
|         await _seriesService.UpdateRelatedSeries(addRelationDto);
 | |
|         Assert.Equal(2, series1.Relations.Single(s => s.TargetSeriesId == 2).TargetSeriesId);
 | |
| 
 | |
|         _context.Series.Remove(await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1));
 | |
|         try
 | |
|         {
 | |
|             await _context.SaveChangesAsync();
 | |
|         }
 | |
|         catch (Exception)
 | |
|         {
 | |
|             Assert.Fail("Delete of Target Series Failed");
 | |
|         }
 | |
| 
 | |
|         // Remove relations
 | |
|         Assert.Empty((await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(2, SeriesIncludes.Related)).Relations);
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task UpdateRelatedSeries_ShouldNotAllowDuplicates()
 | |
|     {
 | |
|         await ResetDb();
 | |
|         _context.Library.Add(new Library()
 | |
|         {
 | |
|             AppUsers = new List<AppUser>()
 | |
|             {
 | |
|                 new AppUser()
 | |
|                 {
 | |
|                     UserName = "majora2007"
 | |
|                 }
 | |
|             },
 | |
|             Name = "Test LIb",
 | |
|             Type = LibraryType.Book,
 | |
|             Series = new List<Series>()
 | |
|             {
 | |
|                 new SeriesBuilder("Test Series").Build(),
 | |
|                 new SeriesBuilder("Test Series Prequels").Build(),
 | |
|             }
 | |
|         });
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var series1 = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1, SeriesIncludes.Related);
 | |
|         var relation = new SeriesRelation()
 | |
|         {
 | |
|             Series = series1,
 | |
|             SeriesId = series1.Id,
 | |
|             TargetSeriesId = 2, // Target series id
 | |
|             RelationKind = RelationKind.Prequel
 | |
| 
 | |
|         };
 | |
|         // Manually create a relation
 | |
|         series1.Relations.Add(relation);
 | |
| 
 | |
|         // Create a new dto with the previous relation as well
 | |
|         var relationDto = CreateRelationsDto(series1);
 | |
|         relationDto.Adaptations.Add(2);
 | |
| 
 | |
|         await _seriesService.UpdateRelatedSeries(relationDto);
 | |
|         // Expected is only one instance of the relation (hence not duping)
 | |
|         Assert.Equal(2, series1.Relations.Single(s => s.TargetSeriesId == 2).TargetSeriesId);
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task GetRelatedSeries_EditionPrequelSequel_ShouldNotHaveParent()
 | |
|     {
 | |
|         await ResetDb();
 | |
|         _context.Library.Add(new Library()
 | |
|         {
 | |
|             AppUsers = new List<AppUser>()
 | |
|             {
 | |
|                 new AppUser()
 | |
|                 {
 | |
|                     UserName = "majora2007"
 | |
|                 }
 | |
|             },
 | |
|             Name = "Test LIb",
 | |
|             Type = LibraryType.Book,
 | |
|             Series = new List<Series>()
 | |
|             {
 | |
|                 new SeriesBuilder("Test Series").Build(),
 | |
|                 new SeriesBuilder("Test Series Editions").Build(),
 | |
|                 new SeriesBuilder("Test Series Prequels").Build(),
 | |
|                 new SeriesBuilder("Test Series Sequels").Build(),
 | |
|                 new SeriesBuilder("Test Series Adaption").Build(),
 | |
|             }
 | |
|         });
 | |
|         await _context.SaveChangesAsync();
 | |
|         var series1 = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1, SeriesIncludes.Related);
 | |
|         // Add relations
 | |
|         var addRelationDto = CreateRelationsDto(series1);
 | |
|         addRelationDto.Editions.Add(2);
 | |
|         addRelationDto.Prequels.Add(3);
 | |
|         addRelationDto.Sequels.Add(4);
 | |
|         addRelationDto.Adaptations.Add(5);
 | |
|         await _seriesService.UpdateRelatedSeries(addRelationDto);
 | |
| 
 | |
| 
 | |
|         Assert.Empty(_seriesService.GetRelatedSeries(1, 2).Result.Parent);
 | |
|         Assert.Empty(_seriesService.GetRelatedSeries(1, 3).Result.Parent);
 | |
|         Assert.Empty(_seriesService.GetRelatedSeries(1, 4).Result.Parent);
 | |
|         Assert.NotEmpty(_seriesService.GetRelatedSeries(1, 5).Result.Parent);
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task SeriesRelation_ShouldAllowDeleteOnLibrary()
 | |
|     {
 | |
|         await ResetDb();
 | |
|         var lib = new LibraryBuilder("Test LIb")
 | |
|             .WithSeries(new SeriesBuilder("Test Series").Build())
 | |
|             .WithSeries(new SeriesBuilder("Test Series Prequels").Build())
 | |
|             .WithSeries(new SeriesBuilder("Test Series Sequels").Build())
 | |
|             .WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
 | |
|             .Build();
 | |
|         _context.Library.Add(lib);
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var series1 = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1, SeriesIncludes.Related);
 | |
|         // Add relations
 | |
|         var addRelationDto = CreateRelationsDto(series1);
 | |
|         addRelationDto.Adaptations.Add(2);
 | |
|         addRelationDto.Sequels.Add(3);
 | |
|         await _seriesService.UpdateRelatedSeries(addRelationDto);
 | |
| 
 | |
|         var library = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(lib.Id);
 | |
|         _unitOfWork.LibraryRepository.Delete(library);
 | |
| 
 | |
|         try
 | |
|         {
 | |
|             await _unitOfWork.CommitAsync();
 | |
|         }
 | |
|         catch (Exception)
 | |
|         {
 | |
|             Assert.False(true);
 | |
|         }
 | |
| 
 | |
|         Assert.Null(await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(1));
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public async Task SeriesRelation_ShouldAllowDeleteOnLibrary_WhenSeriesCrossLibraries()
 | |
|     {
 | |
|         await ResetDb();
 | |
|         var lib1 = new LibraryBuilder("Test LIb")
 | |
|             .WithSeries(new SeriesBuilder("Test Series")
 | |
|                 .WithVolume(new VolumeBuilder("0")
 | |
|                     .WithChapter(new ChapterBuilder("1").WithFile(
 | |
|                         new MangaFileBuilder($"{DataDirectory}1.zip", MangaFormat.Archive)
 | |
|                             .WithPages(1)
 | |
|                             .Build()
 | |
|                         ).Build())
 | |
|                     .Build())
 | |
|                 .Build())
 | |
|             .WithSeries(new SeriesBuilder("Test Series Prequels").Build())
 | |
|             .WithSeries(new SeriesBuilder("Test Series Sequels").Build())
 | |
|             .WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
 | |
|             .Build();
 | |
|         _context.Library.Add(lib1);
 | |
| 
 | |
|         var lib2 = new LibraryBuilder("Test LIb 2", LibraryType.Book)
 | |
|             .WithSeries(new SeriesBuilder("Test Series 2").Build())
 | |
|             .WithSeries(new SeriesBuilder("Test Series Prequels 2").Build())
 | |
|             .WithSeries(new SeriesBuilder("Test Series Prequels 2").Build())// TODO: Is this a bug
 | |
|             .WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
 | |
|             .Build();
 | |
|         _context.Library.Add(lib2);
 | |
| 
 | |
|         await _context.SaveChangesAsync();
 | |
| 
 | |
|         var series1 = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1, SeriesIncludes.Related);
 | |
|         // Add relations
 | |
|         var addRelationDto = CreateRelationsDto(series1);
 | |
|         addRelationDto.Adaptations.Add(4); // cross library link
 | |
|         await _seriesService.UpdateRelatedSeries(addRelationDto);
 | |
| 
 | |
|         var library = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(lib1.Id, LibraryIncludes.Series);
 | |
|         _unitOfWork.LibraryRepository.Delete(library);
 | |
| 
 | |
|         try
 | |
|         {
 | |
|             await _unitOfWork.CommitAsync();
 | |
|         }
 | |
|         catch (Exception)
 | |
|         {
 | |
|             Assert.False(true);
 | |
|         }
 | |
| 
 | |
|         Assert.Null(await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(1));
 | |
|     }
 | |
| 
 | |
|     #endregion
 | |
| 
 | |
|     #region UpdateRelatedList
 | |
| 
 | |
|     // TODO: Implement UpdateRelatedList
 | |
| 
 | |
|     #endregion
 | |
| 
 | |
|     #region FormatChapterName
 | |
| 
 | |
|     [Theory]
 | |
|     [InlineData(LibraryType.Manga, false, "Chapter")]
 | |
|     [InlineData(LibraryType.Comic, false, "Issue")]
 | |
|     [InlineData(LibraryType.Comic, true, "Issue #")]
 | |
|     [InlineData(LibraryType.Book, false, "Book")]
 | |
|     public void FormatChapterNameTest(LibraryType libraryType, bool withHash, string expected )
 | |
|     {
 | |
|         Assert.Equal(expected, SeriesService.FormatChapterName(libraryType, withHash));
 | |
|     }
 | |
| 
 | |
|     #endregion
 | |
| 
 | |
|     #region FormatChapterTitle
 | |
| 
 | |
|     [Fact]
 | |
|     public void FormatChapterTitle_Manga_NonSpecial()
 | |
|     {
 | |
|         var chapter = new ChapterBuilder("1").WithTitle("Some title").WithIsSpecial(false).Build();
 | |
|         Assert.Equal("Chapter Some title", SeriesService.FormatChapterTitle(chapter, LibraryType.Manga, false));
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public void FormatChapterTitle_Manga_Special()
 | |
|     {
 | |
|         var chapter = new ChapterBuilder("1").WithTitle("Some title").WithIsSpecial(true).Build();
 | |
|         Assert.Equal("Some title", SeriesService.FormatChapterTitle(chapter, LibraryType.Manga, false));
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public void FormatChapterTitle_Comic_NonSpecial_WithoutHash()
 | |
|     {
 | |
|         var chapter = new ChapterBuilder("1").WithTitle("Some title").WithIsSpecial(false).Build();
 | |
|         Assert.Equal("Issue Some title", SeriesService.FormatChapterTitle(chapter, LibraryType.Comic, false));
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public void FormatChapterTitle_Comic_Special_WithoutHash()
 | |
|     {
 | |
|         var chapter = new ChapterBuilder("1").WithTitle("Some title").WithIsSpecial(true).Build();
 | |
|         Assert.Equal("Some title", SeriesService.FormatChapterTitle(chapter, LibraryType.Comic, false));
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public void FormatChapterTitle_Comic_NonSpecial_WithHash()
 | |
|     {
 | |
|         var chapter = new ChapterBuilder("1").WithTitle("Some title").WithIsSpecial(false).Build();
 | |
|         Assert.Equal("Issue #Some title", SeriesService.FormatChapterTitle(chapter, LibraryType.Comic, true));
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public void FormatChapterTitle_Comic_Special_WithHash()
 | |
|     {
 | |
|         var chapter = new ChapterBuilder("1").WithTitle("Some title").WithIsSpecial(true).Build();
 | |
|         Assert.Equal("Some title", SeriesService.FormatChapterTitle(chapter, LibraryType.Comic, true));
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public void FormatChapterTitle_Book_NonSpecial()
 | |
|     {
 | |
|         var chapter = new ChapterBuilder("1").WithTitle("Some title").WithIsSpecial(false).Build();
 | |
|         Assert.Equal("Book Some title", SeriesService.FormatChapterTitle(chapter, LibraryType.Book, false));
 | |
|     }
 | |
| 
 | |
|     [Fact]
 | |
|     public void FormatChapterTitle_Book_Special()
 | |
|     {
 | |
|         var chapter = new ChapterBuilder("1").WithTitle("Some title").WithIsSpecial(true).Build();
 | |
|         Assert.Equal("Some title", SeriesService.FormatChapterTitle(chapter, LibraryType.Book, false));
 | |
|     }
 | |
| 
 | |
|     #endregion
 | |
| }
 |