mirror of
https://github.com/zoriya/Kyoo.git
synced 2025-10-23 23:09:09 -04:00
Use another watchfile library
This commit is contained in:
parent
4bffd359b9
commit
bbe8a19189
@ -10,17 +10,16 @@ Kyoo also needs 3 files to work properly. Two of them can simply be copy-pasted
|
|||||||
Those files can be put in any directory of your choice.
|
Those files can be put in any directory of your choice.
|
||||||
|
|
||||||
Those 3 files are:
|
Those 3 files are:
|
||||||
|
|
||||||
- A `docker-compose.yml` (simply copy docker-compose.prod.yml from [here](https://raw.githubusercontent.com/zoriya/Kyoo/master/docker-compose.prod.yml)).
|
- A `docker-compose.yml` (simply copy docker-compose.prod.yml from [here](https://raw.githubusercontent.com/zoriya/Kyoo/master/docker-compose.prod.yml)).
|
||||||
- A `nginx.conf.template` copied from [here](https://raw.githubusercontent.com/zoriya/Kyoo/master/nginx.conf.template).
|
- A `nginx.conf.template` copied from [here](https://raw.githubusercontent.com/zoriya/Kyoo/master/nginx.conf.template).
|
||||||
- A `.env` file that you will need to **fill**. Look at the example [.env.example](https://raw.githubusercontent.com/zoriya/Kyoo/master/.env.example)
|
- A `.env` file that you will need to **fill**. Look at the example [.env.example](https://raw.githubusercontent.com/zoriya/Kyoo/master/.env.example)
|
||||||
|
|
||||||
|
|
||||||
> If you want an explanation of what are those files, you can read the following:
|
> If you want an explanation of what are those files, you can read the following:
|
||||||
> The `docker-compose.yml` file describes the different services of Kyoo, where they should be downloaded and their start order. \
|
> The `docker-compose.yml` file describes the different services of Kyoo, where they should be downloaded and their start order. \
|
||||||
> The `nignx.conf.template` file describes which service will be called when accessing the URL of Kyoo. \
|
> The `nignx.conf.template` file describes which service will be called when accessing the URL of Kyoo. \
|
||||||
> The `.env` file contains all the configuration options that the services in `docker-compose.yml` will read.
|
> The `.env` file contains all the configuration options that the services in `docker-compose.yml` will read.
|
||||||
|
|
||||||
|
|
||||||
To retrieve metadata, Kyoo will need to communicate with an external service. For now, that is `the movie database`.
|
To retrieve metadata, Kyoo will need to communicate with an external service. For now, that is `the movie database`.
|
||||||
For this purpose, you will need to get an API Key. For that, go to [themoviedb.org](https://www.themoviedb.org/) and create an account, then
|
For this purpose, you will need to get an API Key. For that, go to [themoviedb.org](https://www.themoviedb.org/) and create an account, then
|
||||||
go [here](https://www.themoviedb.org/settings/api) and copy the `API Key (v3 auth)`, paste it after the `THEMOVIEDB_APIKEY=` on the `.env` file.
|
go [here](https://www.themoviedb.org/settings/api) and copy the `API Key (v3 auth)`, paste it after the `THEMOVIEDB_APIKEY=` on the `.env` file.
|
||||||
@ -48,6 +47,9 @@ unsure that your `.env` contains all the options specified in the updated `.env.
|
|||||||
After that, you will need to update Kyoo's services. For that, open a terminal in the configuration's directory and run
|
After that, you will need to update Kyoo's services. For that, open a terminal in the configuration's directory and run
|
||||||
the command `docker-compose pull`. You are now ready to restart Kyoo, you can run `docker-compose up -d`.
|
the command `docker-compose pull`. You are now ready to restart Kyoo, you can run `docker-compose up -d`.
|
||||||
|
|
||||||
|
You can also enable automatic updates via an external tool like [watchtower](https://containrrr.dev/watchtower/).
|
||||||
|
TLDR: `docker run -d --name watchtower -e WATCHTOWER_CLEANUP=true -e WATCHTOWER_POLL_INTERVAL=86400 -v /var/run/docker.sock:/var/run/docker.sock containrrr/watchtower`
|
||||||
|
|
||||||
# Uninstalling
|
# Uninstalling
|
||||||
|
|
||||||
To uninstall Kyoo, you need to open a terminal in the configuration's directory and run `docker-compose down`. This will
|
To uninstall Kyoo, you need to open a terminal in the configuration's directory and run `docker-compose down`. This will
|
||||||
|
@ -2,4 +2,4 @@ guessit
|
|||||||
aiohttp
|
aiohttp
|
||||||
jsons
|
jsons
|
||||||
black-with-tabs
|
black-with-tabs
|
||||||
watchdog
|
watchfiles
|
||||||
|
@ -1,74 +1,19 @@
|
|||||||
import asyncio
|
import logging
|
||||||
from functools import wraps
|
from watchfiles import awatch, Change
|
||||||
from watchdog.observers import Observer
|
from .utils import ProviderError
|
||||||
from watchdog.events import (
|
|
||||||
FileSystemEventHandler,
|
|
||||||
DirCreatedEvent,
|
|
||||||
FileCreatedEvent,
|
|
||||||
DirMovedEvent,
|
|
||||||
FileMovedEvent,
|
|
||||||
DirDeletedEvent,
|
|
||||||
FileDeletedEvent,
|
|
||||||
)
|
|
||||||
|
|
||||||
from scanner.utils import log_errors
|
|
||||||
|
|
||||||
from .scanner import Scanner
|
from .scanner import Scanner
|
||||||
|
|
||||||
task_list = []
|
|
||||||
event = asyncio.Event()
|
|
||||||
|
|
||||||
|
|
||||||
async def monitor(path: str, scanner: Scanner):
|
async def monitor(path: str, scanner: Scanner):
|
||||||
global task_list
|
async for changes in awatch(path):
|
||||||
|
for (event, file) in changes:
|
||||||
observer = Observer()
|
try:
|
||||||
handler = EventHandler(scanner)
|
if event == Change.added:
|
||||||
observer.schedule(handler, path, recursive=True)
|
await scanner.identify(file)
|
||||||
observer.start()
|
else:
|
||||||
|
print(f"Change {event} occured for file {file}")
|
||||||
while True:
|
except ProviderError as e:
|
||||||
if any(task_list):
|
logging.error(str(e))
|
||||||
tl = task_list
|
except Exception as e:
|
||||||
task_list = []
|
logging.exception("Unhandled error", exc_info=e)
|
||||||
await asyncio.gather(*tl)
|
print("end", flush=True)
|
||||||
await event.wait()
|
|
||||||
event.clear()
|
|
||||||
# Should call .join() if the while stops one day.
|
|
||||||
# observer.join()
|
|
||||||
|
|
||||||
|
|
||||||
def async_event(f):
|
|
||||||
# Log errors of f and catch them to prevent the gather to throw.
|
|
||||||
f = log_errors(f)
|
|
||||||
|
|
||||||
@wraps(f)
|
|
||||||
def internal(*args, **kwargs):
|
|
||||||
task_list.append(f(*args, **kwargs))
|
|
||||||
event.set()
|
|
||||||
|
|
||||||
return internal
|
|
||||||
|
|
||||||
|
|
||||||
class EventHandler(FileSystemEventHandler):
|
|
||||||
def __init__(self, scanner: Scanner):
|
|
||||||
self._scanner = scanner
|
|
||||||
|
|
||||||
@async_event
|
|
||||||
async def on_created(self, event: DirCreatedEvent | FileCreatedEvent):
|
|
||||||
if event.is_directory:
|
|
||||||
return
|
|
||||||
await self._scanner.identify(event.src_path)
|
|
||||||
|
|
||||||
# TODO: Implement the following two methods
|
|
||||||
def on_moved(self, event: DirMovedEvent | FileMovedEvent):
|
|
||||||
if event.is_directory:
|
|
||||||
# TODO: Check if this event is also called for files in the directory or not.
|
|
||||||
return
|
|
||||||
print(event.src_path, event.dest_path)
|
|
||||||
|
|
||||||
def on_deleted(self, event: DirDeletedEvent | FileDeletedEvent):
|
|
||||||
if event.is_directory:
|
|
||||||
# TODO: Check if this event is also called for files in the directory or not.
|
|
||||||
return
|
|
||||||
print(event.src_path)
|
|
||||||
|
@ -26,7 +26,7 @@ class Scanner:
|
|||||||
async def scan(self, path: str):
|
async def scan(self, path: str):
|
||||||
logging.info("Starting the scan. It can take some times...")
|
logging.info("Starting the scan. It can take some times...")
|
||||||
self.registered = await self.get_registered_paths()
|
self.registered = await self.get_registered_paths()
|
||||||
videos = filter(lambda p: p.is_file(), Path(path).rglob("*"))
|
videos = (str(p) for p in Path(path).rglob("*") if p.is_file())
|
||||||
# We batch videos by 20 because too mutch at once kinda DDOS everything.
|
# We batch videos by 20 because too mutch at once kinda DDOS everything.
|
||||||
for group in batch(videos, 20):
|
for group in batch(videos, 20):
|
||||||
logging.info("Batch finished. Starting a new one")
|
logging.info("Batch finished. Starting a new one")
|
||||||
@ -44,7 +44,7 @@ class Scanner:
|
|||||||
return list(x["path"] for x in ret["items"])
|
return list(x["path"] for x in ret["items"])
|
||||||
|
|
||||||
@log_errors
|
@log_errors
|
||||||
async def identify(self, path: Path):
|
async def identify(self, path: str):
|
||||||
if path in self.registered:
|
if path in self.registered:
|
||||||
return
|
return
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user