Skip to content

Commit

Permalink
feat: Add support for 4/3
Browse files Browse the repository at this point in the history
  • Loading branch information
Flowtter committed Jan 5, 2024
1 parent 61fa06e commit 42e3473
Show file tree
Hide file tree
Showing 11 changed files with 134 additions and 41 deletions.
5 changes: 5 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ config example:
"second-after": 3,
"second-between-kills": 5
},
"stretch": false,
"game": "valorant"
}
```
Expand All @@ -60,6 +61,7 @@ The following settings are adjustable:
- second-after: Seconds of gameplay included after the highlight.
- second-between-kills: Transition time between highlights. If the time between two highlights is less than this value, the both highlights will be merged.
- game: Chosen game (either "valorant", "overwatch" or "csgo2")
- stretch: This is an option in case you're playing on a 4:3 resolution but your clips are recorded in 16:9.

### Recommended settings

Expand All @@ -79,6 +81,7 @@ Here are some settings that I found to work well for me:
"second-after": 0.5,
"second-between-kills": 3
},
"stretch": false,
"game": "valorant"
}
```
Expand All @@ -96,6 +99,7 @@ Here are some settings that I found to work well for me:
"second-after": 3,
"second-between-kills": 5
},
"stretch": false,
"game": "overwatch"
}
```
Expand All @@ -113,6 +117,7 @@ Here are some settings that I found to work well for me:
"second-after": 1,
"second-between-kills": 3
},
"stretch": false,
"game": "csgo2"
}
```
Expand Down
9 changes: 5 additions & 4 deletions crispy-api/api/__init__.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import logging
import os
import subprocess
from typing import Optional

Expand All @@ -11,7 +12,7 @@
from montydb import MontyClient, set_storage
from pydantic.json import ENCODERS_BY_TYPE

from api.config import DATABASE_PATH, DEBUG, FRAMERATE, GAME, MUSICS, VIDEOS
from api.config import ASSETS, DATABASE_PATH, DEBUG, FRAMERATE, GAME, MUSICS, VIDEOS
from api.tools.AI.network import NeuralNetwork
from api.tools.enums import SupportedGames
from api.tools.filters import apply_filters # noqa
Expand All @@ -22,11 +23,11 @@
neural_network = NeuralNetwork(GAME)

if GAME == SupportedGames.OVERWATCH:
neural_network.load("./assets/overwatch.npy")
neural_network.load(os.path.join(ASSETS, "overwatch.npy"))
elif GAME == SupportedGames.VALORANT:
neural_network.load("./assets/valorant.npy")
neural_network.load(os.path.join(ASSETS, "valorant.npy"))
elif GAME == SupportedGames.CSGO2:
neural_network.load("./assets/csgo2.npy")
neural_network.load(os.path.join(ASSETS, "csgo2.npy"))
else:
raise ValueError(f"game {GAME} not supported")

Expand Down
1 change: 1 addition & 0 deletions crispy-api/api/config.py
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,7 @@

CONFIDENCE = __neural_network.get("confidence", 0.6)

STRETCH = __settings.get("stretch", False)
GAME = __settings.get("game")
if GAME is None:
raise KeyError("game not found in settings.json")
Expand Down
83 changes: 69 additions & 14 deletions crispy-api/api/models/highlight.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,42 @@
csgo2_mask = Image.open(CSGO2_MASK_PATH)


class Box:
def __init__(
self,
offset_x: int,
y: int,
width: int,
height: int,
shift_x: int,
stretch: bool,
) -> None:
"""
:param offset_x: Offset in pixels from the center of the video to the left
:param y: Offset in pixels from the top of the video
:param width: Width of the box in pixels
:param height: Height of the box in pixels
:param shift_x: Shift the box by a certain amount of pixels to the right
example:
If you want to create a box at 50 px from the center on x, but shifted by 20px to the right
you would do:
Box(50, 0, 100, 100, 20)
"""
half = 720 if stretch else 960

self.x = half - offset_x + shift_x
self.y = y
self.width = width
self.height = height

def __iter__(self) -> Any:
yield self.x
yield self.y
yield self.width
yield self.height


class Highlight(Thingy):
segments_path: Optional[str]
local_filters: Optional[Dict[str, Any]]
Expand Down Expand Up @@ -55,7 +91,7 @@ async def extract_thumbnails(self) -> bool:
async def extract_images(
self,
post_process: Callable,
coordinates: Tuple,
coordinates: Box,
framerate: int = 4,
) -> bool:
"""
Expand Down Expand Up @@ -93,7 +129,9 @@ async def extract_images(

return True

async def extract_overwatch_images(self, framerate: int = 4) -> bool:
async def extract_overwatch_images(
self, framerate: int = 4, stretch: bool = False
) -> bool:
def post_process(image: Image) -> Image:
r, g, b = image.split()
for x in range(image.width):
Expand Down Expand Up @@ -121,10 +159,12 @@ def post_process(image: Image) -> Image:
return final

return await self.extract_images(
post_process, (910, 490, 100, 100), framerate=framerate
post_process, Box(50, 490, 100, 100, 0, stretch), framerate=framerate
)

async def extract_valorant_images(self, framerate: int = 4) -> bool:
async def extract_valorant_images(
self, framerate: int = 4, stretch: bool = False
) -> bool:
def _apply_filter_and_do_operations(
image: Image, image_filter: ImageFilter
) -> Image:
Expand Down Expand Up @@ -159,10 +199,12 @@ def post_process(image: Image) -> Image:
return final

return await self.extract_images(
post_process, (899, 801, 122, 62), framerate=framerate
post_process, Box(61, 801, 122, 62, 0, stretch), framerate=framerate
)

async def extract_csgo2_images(self, framerate: int = 4) -> bool:
async def extract_csgo2_images(
self, framerate: int = 4, stretch: bool = False
) -> bool:
def post_process(image: Image) -> Image:
image = ImageOps.grayscale(
image.filter(ImageFilter.FIND_EDGES).filter(
Expand All @@ -175,18 +217,18 @@ def post_process(image: Image) -> Image:
return final

return await self.extract_images(
post_process, (930, 925, 100, 100), framerate=framerate
post_process, Box(50, 925, 100, 100, 20, stretch), framerate=framerate
)

async def extract_images_from_game(
self, game: SupportedGames, framerate: int = 4
self, game: SupportedGames, framerate: int = 4, stretch: bool = False
) -> bool:
if game == SupportedGames.OVERWATCH:
return await self.extract_overwatch_images(framerate)
return await self.extract_overwatch_images(framerate, stretch)
elif game == SupportedGames.VALORANT:
return await self.extract_valorant_images(framerate)
return await self.extract_valorant_images(framerate, stretch)
elif game == SupportedGames.CSGO2:
return await self.extract_csgo2_images(framerate)
return await self.extract_csgo2_images(framerate, stretch)
else:
raise NotImplementedError

Expand All @@ -212,7 +254,7 @@ def recompile(self) -> bool:
return result

async def extract_segments(
self, timestamps: List[Tuple[float, float]]
self, timestamps: List[Tuple[float, float]], stretch: bool = False
) -> List[Segment]:
"""
Segment a video into multiple videos
Expand Down Expand Up @@ -269,11 +311,13 @@ async def extract_segments(

for (start, end) in new_timestamps:
segment_save_path = os.path.join(self.segments_path, f"{start}-{end}.mp4")
dar = 4 / 3 if stretch else 16 / 9
(
ffmpeg.input(
self.path,
)
.apply_filters(self.id)
.filter("setdar", dar)
.output(audio, segment_save_path, ss=f"{start}", to=f"{end}")
.overwrite_output()
.run(quiet=True)
Expand Down Expand Up @@ -318,7 +362,11 @@ async def concatenate_segments(self) -> bool:
return True

async def scale_video(
self, width: int = 1920, height: int = 1080, backup: str = BACKUP
self,
width: int = 1920,
height: int = 1080,
backup: str = BACKUP,
stretch: bool = False,
) -> None:
"""
Scale (up or down) a video.
Expand All @@ -344,6 +392,9 @@ async def scale_video(
audio = silence_if_no_audio(video.audio, backup_path)

video = video.filter("scale", w=width, h=height)
video = (
video.filter("setdar", 4 / 3) if stretch else video.filter("setdar", 16 / 9)
)

ffmpeg.output(video, audio, self.path, start_number=0).run(quiet=True)

Expand All @@ -358,7 +409,7 @@ async def remove(self) -> None:
Filter.delete_one({"highlight_id": self.id})
self.delete()

async def extract_snippet_in_lower_resolution(self) -> bool:
async def extract_snippet_in_lower_resolution(self, stretch: bool = False) -> bool:
"""Extract 5 seconds of a highlight in lower resolution"""
if self.snippet_path:
return False
Expand All @@ -367,7 +418,11 @@ async def extract_snippet_in_lower_resolution(self) -> bool:

video = ffmpeg.input(self.path, sseof="-20")
audio = silence_if_no_audio(video.audio, self.path)

video = video.filter("scale", 640, -1)
video = (
video.filter("setdar", 4 / 3) if stretch else video.filter("setdar", 16 / 9)
)
ffmpeg.output(video, audio, snippet_path, t="00:00:5").run(quiet=True)

self.update({"snippet_path": snippet_path})
Expand Down
2 changes: 1 addition & 1 deletion crispy-api/api/tools/ffmpeg.py
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ async def merge_videos(
else:
clips = []
for filename in videos_path:
clips.append(mpe.VideoFileClip(filename))
clips.append(mpe.VideoFileClip(filename).resize((1920, 1080)))

final_clip = mpe.concatenate_videoclips(clips)

Expand Down
39 changes: 19 additions & 20 deletions crispy-api/api/tools/setup.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,3 @@
import asyncio
import logging
import os
import shutil
Expand All @@ -7,7 +6,7 @@
import ffmpeg
from PIL import Image

from api.config import SESSION, SILENCE_PATH
from api.config import SESSION, SILENCE_PATH, STRETCH
from api.models.filter import Filter
from api.models.highlight import Highlight
from api.models.music import Music
Expand All @@ -29,6 +28,7 @@ async def handle_highlights(
game: SupportedGames,
framerate: int = 4,
session: str = SESSION,
stretch: bool = STRETCH,
) -> List[Highlight]:

if not os.path.exists(session):
Expand All @@ -45,7 +45,6 @@ async def handle_highlights(
logger.info(f"Removing highlight {highlight.path}")
await highlight.remove()

job_scheduler = JobScheduler(4)
new_highlights = []
for file in sorted(os.listdir(path)):
file_path = os.path.join(path, file)
Expand All @@ -67,20 +66,14 @@ async def handle_highlights(
).save()
Filter({"highlight_id": highlight.id}).save()
new_highlights.append(highlight)
await highlight.extract_thumbnails()

job_scheduler.schedule(
highlight.extract_images_from_game,
kwargs={"game": game, "framerate": framerate},
)
job_scheduler.schedule(highlight.extract_thumbnails)
job_scheduler.schedule(highlight.extract_snippet_in_lower_resolution)
index += 1

logger.info(f"Adding {len(new_highlights)} highlights, this may take a while.")
logger.warning("Wait for `Application startup complete.` to use Crispy.")

job_scheduler.run_in_thread().join()

target_size = (1440, 1080) if stretch else (1920, 1080)
for highlight in new_highlights:
if not video_has_audio(highlight.path):
tmp_path = os.path.join(highlight.directory, "tmp.mp4")
Expand All @@ -90,18 +83,24 @@ async def handle_highlights(

ffmpeg.input(highlight.path).output(
video, audio, tmp_path, vcodec="copy", acodec="aac"
).overwrite_output().run()
).overwrite_output().run(quiet=True)
shutil.move(tmp_path, highlight.path)

if Image.open(highlight.thumbnail_path_full_size).size != (1920, 1080):
await highlight.scale_video()
coroutines = [
highlight.extract_thumbnails(),
highlight.extract_snippet_in_lower_resolution(),
highlight.extract_images_from_game(game, framerate),
]
if Image.open(highlight.thumbnail_path_full_size).size != target_size:
await highlight.scale_video(*target_size, stretch=stretch)
await highlight.extract_thumbnails()

await asyncio.gather(*coroutines)
job_scheduler = JobScheduler(4)
for highlight in new_highlights:
job_scheduler.schedule(
highlight.extract_images_from_game,
kwargs={"game": game, "framerate": framerate, "stretch": stretch},
)
job_scheduler.schedule(
highlight.extract_snippet_in_lower_resolution, kwargs={"stretch": stretch}
)

job_scheduler.run_in_thread().join()

Highlight.update_many({}, {"$set": {"job_id": None}})

Expand Down
1 change: 1 addition & 0 deletions crispy-api/settings.json
Original file line number Diff line number Diff line change
Expand Up @@ -8,5 +8,6 @@
"second-after": 0.5,
"second-between-kills": 1
},
"stretch": false,
"game": "valorant"
}
2 changes: 1 addition & 1 deletion crispy-api/tests/assets
Submodule assets updated 39 files
+292 −212 compare/test_create_dataset/result.csv
+ compare/test_extract_segments_stretch/segments/3.375-3.75-00000000.jpg
+ compare/test_extract_segments_stretch/segments/3.375-3.75-00000001.jpg
+ compare/test_extract_segments_stretch/segments/3.375-3.75-00000002.jpg
+ compare/test_extract_segments_stretch/segments/3.375-3.75_downscaled-00000000.jpg
+ compare/test_extract_segments_stretch/segments/3.375-3.75_downscaled-00000001.jpg
+ compare/test_extract_segments_stretch/segments/3.375-3.75_downscaled-00000002.jpg
+ compare/test_extract_segments_stretch/segments/4.25-4.875-00000000.jpg
+ compare/test_extract_segments_stretch/segments/4.25-4.875-00000001.jpg
+ compare/test_extract_segments_stretch/segments/4.25-4.875-00000002.jpg
+ compare/test_extract_segments_stretch/segments/4.25-4.875-00000003.jpg
+ compare/test_extract_segments_stretch/segments/4.25-4.875-00000004.jpg
+ compare/test_extract_segments_stretch/segments/4.25-4.875_downscaled-00000000.jpg
+ compare/test_extract_segments_stretch/segments/4.25-4.875_downscaled-00000001.jpg
+ compare/test_extract_segments_stretch/segments/4.25-4.875_downscaled-00000002.jpg
+ compare/test_extract_segments_stretch/segments/4.25-4.875_downscaled-00000003.jpg
+ compare/test_extract_segments_stretch/segments/4.25-4.875_downscaled-00000004.jpg
+ compare/test_handle_highlights_stretch/images/00000000.bmp
+ compare/test_handle_highlights_stretch/images/00000001.bmp
+ compare/test_handle_highlights_stretch/images/00000002.bmp
+ compare/test_handle_highlights_stretch/images/00000003.bmp
+ compare/test_handle_highlights_stretch/images/00000004.bmp
+ compare/test_handle_highlights_stretch/images/00000005.bmp
+ compare/test_handle_highlights_stretch/images/00000006.bmp
+ compare/test_handle_highlights_stretch/images/00000007.bmp
+ compare/test_handle_highlights_stretch/images/00000008.bmp
+ compare/test_handle_highlights_stretch/images/00000009.bmp
+ compare/test_handle_highlights_stretch/images/00000010.bmp
+ compare/test_handle_highlights_stretch/images/00000011.bmp
+ compare/test_handle_highlights_stretch/images/00000012.bmp
+ compare/test_handle_highlights_stretch/images/00000013.bmp
+ compare/test_handle_highlights_stretch/images/00000014.bmp
+ compare/test_handle_highlights_stretch/images/00000015.bmp
+ compare/test_handle_highlights_stretch/images/00000016.bmp
+ compare/test_handle_highlights_stretch/images/00000017.bmp
+ compare/test_handle_highlights_stretch/images/00000018.bmp
+ compare/test_handle_highlights_stretch/images/00000019.bmp
+ videos/main-video-1440.mp4
+ videos/main-video-stretch.mp4
2 changes: 2 additions & 0 deletions crispy-api/tests/constants.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,8 @@
MAIN_VIDEO = os.path.join(VIDEOS_PATH, "main-video.mp4")
MAIN_VIDEO_DOWNSCALED = os.path.join(VIDEOS_PATH, "main-video_downscaled.mp4")
MAIN_VIDEO_NO_AUDIO = os.path.join(VIDEOS_PATH, "main-video-no-audio.mp4")
MAIN_VIDEO_STRETCH = os.path.join(VIDEOS_PATH, "main-video-stretch.mp4")
MAIN_VIDEO_1440 = os.path.join(VIDEOS_PATH, "main-video-1440.mp4")
MAIN_VIDEO_OVERWATCH = os.path.join(VIDEOS_PATH, "main-video-overwatch.mp4")
MAIN_VIDEO_CSGO2 = os.path.join(VIDEOS_PATH, "main-video-csgo2.mp4")
MAIN_SEGMENT = os.path.join(VIDEOS_PATH, "main-video-segment.mp4")
Expand Down
Loading

0 comments on commit 42e3473

Please sign in to comment.