Unbewohnte
4 years ago
committed by
GitHub
1 changed files with 0 additions and 339 deletions
@ -1,339 +0,0 @@ |
|||||||
package main |
|
||||||
|
|
||||||
import ( |
|
||||||
"encoding/json" |
|
||||||
"errors" |
|
||||||
"fmt" |
|
||||||
"image" |
|
||||||
"image/color" |
|
||||||
"image/png" |
|
||||||
"io" |
|
||||||
"log" |
|
||||||
"os" |
|
||||||
"path/filepath" |
|
||||||
"strings" |
|
||||||
"sync" |
|
||||||
"time" |
|
||||||
) |
|
||||||
|
|
||||||
var ( |
|
||||||
// used as a flag if the program executed for "the first time"
|
|
||||||
settingsFileExisted bool = false |
|
||||||
WG sync.WaitGroup |
|
||||||
) |
|
||||||
|
|
||||||
const ( |
|
||||||
settingsFilename string = "settings.json" |
|
||||||
) |
|
||||||
|
|
||||||
// struct for json settings` file contents
|
|
||||||
type Settings struct { |
|
||||||
OsuDir string `json:"pathToOsu"` |
|
||||||
ReplacementImagePath string `json:"pathToimage"` |
|
||||||
CreateBlackBGImage bool `json:"createBlackBackgoundImage"` |
|
||||||
Workers uint `json:"Workers"` |
|
||||||
} |
|
||||||
|
|
||||||
// creates directory for logs and sets output to file
|
|
||||||
func setUpLogs() { |
|
||||||
logsDir := filepath.Join(".", "logs") |
|
||||||
err := os.MkdirAll(logsDir, os.ModePerm) |
|
||||||
if err != nil { |
|
||||||
panic(err) |
|
||||||
} |
|
||||||
file, err := os.Create(filepath.Join(logsDir, "logs.log")) |
|
||||||
log.SetOutput(file) |
|
||||||
} |
|
||||||
|
|
||||||
// creates "settings.json" and sets the flag
|
|
||||||
func createSettingsFile() { |
|
||||||
files, err := os.ReadDir(".") |
|
||||||
if err != nil { |
|
||||||
log.Fatal("ERROR : Unable to read current directory") |
|
||||||
} |
|
||||||
for _, file := range files { |
|
||||||
if file.IsDir() == false { |
|
||||||
if file.Name() == settingsFilename { |
|
||||||
log.Println("Found settings file") |
|
||||||
settingsFileExisted = true |
|
||||||
return |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
file, err := os.Create("settings.json") |
|
||||||
if err != nil { |
|
||||||
log.Fatal("ERROR: Error creating settings file... : ", err) |
|
||||||
} |
|
||||||
settings := Settings{ |
|
||||||
OsuDir: "", |
|
||||||
ReplacementImagePath: "", |
|
||||||
CreateBlackBGImage: true, |
|
||||||
Workers: 100, |
|
||||||
} |
|
||||||
jsonEncodedSettings, err := json.MarshalIndent(settings, "", " ") |
|
||||||
if err != nil { |
|
||||||
log.Println("ERROR: Error creating settings file... : ", err) |
|
||||||
} |
|
||||||
file.Write(jsonEncodedSettings) |
|
||||||
|
|
||||||
file.Close() |
|
||||||
log.Println("Successfully created new settingsFile") |
|
||||||
} |
|
||||||
|
|
||||||
// filepath.Joins the main osu directory with its songs folder
|
|
||||||
func getSongsDir(osudir string) string { |
|
||||||
songsDir := filepath.Join(osudir, "Songs") |
|
||||||
|
|
||||||
stat, err := os.Stat(songsDir) |
|
||||||
if err != nil { |
|
||||||
log.Fatal("ERROR: Error reading path : ", err) |
|
||||||
} |
|
||||||
if !stat.IsDir() { |
|
||||||
log.Fatal("ERROR: Given osu! directory is not a directory") |
|
||||||
} |
|
||||||
|
|
||||||
return songsDir |
|
||||||
} |
|
||||||
|
|
||||||
// unmarshalls settings.json into struct
|
|
||||||
func getSettings() Settings { |
|
||||||
settingsFile, err := os.ReadFile(settingsFilename) |
|
||||||
if err != nil { |
|
||||||
log.Fatal("ERROR: Could not read settings file : ", err) |
|
||||||
} |
|
||||||
var settings Settings |
|
||||||
err = json.Unmarshal(settingsFile, &settings) |
|
||||||
if err != nil { |
|
||||||
log.Fatal("ERROR: Error unmarshalling json file : ", err) |
|
||||||
} |
|
||||||
if settings.Workers <= 0 { |
|
||||||
log.Println("`maxConcurrentWorkers` is set to 0 or less. Replaced with 1") |
|
||||||
settings.Workers = 1 |
|
||||||
} |
|
||||||
return settings |
|
||||||
} |
|
||||||
|
|
||||||
// checks if given string contains ".osu"
|
|
||||||
func isBeatmap(filename string) bool { |
|
||||||
if len(filename) < 5 { |
|
||||||
return false |
|
||||||
} |
|
||||||
if filename[len(filename)-4:] == ".osu" { |
|
||||||
return true |
|
||||||
} |
|
||||||
return false |
|
||||||
} |
|
||||||
|
|
||||||
func isImage(filename string) bool { |
|
||||||
var imageExtentions []string = []string{"jpeg", "jpg", "png", "JPEG", "JPG", "PNG"} |
|
||||||
for _, extention := range imageExtentions { |
|
||||||
if strings.Contains(filename, extention) { |
|
||||||
return true |
|
||||||
} |
|
||||||
} |
|
||||||
return false |
|
||||||
} |
|
||||||
|
|
||||||
// parses .osu file and returns the filename of its background
|
|
||||||
func getBackgroundName(pathToOSUbeatmap string) (string, error) { |
|
||||||
beatmapBytes, err := os.ReadFile(pathToOSUbeatmap) |
|
||||||
if err != nil { |
|
||||||
return "", err |
|
||||||
} |
|
||||||
beatmapContents := string(beatmapBytes) |
|
||||||
|
|
||||||
// get index of "[Events]" (this is where BG filename is stored)
|
|
||||||
eventsIndex := strings.Index(beatmapContents, "[Events]") |
|
||||||
if eventsIndex == -1 { |
|
||||||
return "", errors.New("Could not retrieve index of \"[Events]\"") |
|
||||||
} |
|
||||||
// get index of [TimingPoints] (this tag is right after the previous "[Events]" tag,
|
|
||||||
// so we can grab the whole "[Events]" tag contents)
|
|
||||||
timingPointsIndex := strings.Index(beatmapContents, "[TimingPoints]") |
|
||||||
if timingPointsIndex == -1 { |
|
||||||
return "", errors.New("Could not retrieve index of \"[TimingPoints]\"") |
|
||||||
} |
|
||||||
contentBetween := strings.Split(beatmapContents[eventsIndex:timingPointsIndex], ",") |
|
||||||
|
|
||||||
for _, chunk := range contentBetween { |
|
||||||
if isImage(chunk) { |
|
||||||
return strings.Split(chunk, "\"")[1], nil |
|
||||||
} |
|
||||||
} |
|
||||||
return "", nil |
|
||||||
} |
|
||||||
|
|
||||||
// opens given files, copies one into another
|
|
||||||
func copyFile(src, dst string) error { |
|
||||||
srcFile, err := os.Open(src) |
|
||||||
if err != nil { |
|
||||||
log.Println("ERROR: ", err) |
|
||||||
return err |
|
||||||
} |
|
||||||
defer srcFile.Close() |
|
||||||
|
|
||||||
dstFile, err := os.OpenFile(dst, os.O_WRONLY, os.ModePerm) |
|
||||||
if err != nil { |
|
||||||
log.Println("ERROR: ", err) |
|
||||||
return err |
|
||||||
} |
|
||||||
defer dstFile.Close() |
|
||||||
|
|
||||||
_, err = io.Copy(dstFile, srcFile) |
|
||||||
if err != nil { |
|
||||||
log.Println("ERROR: Error copying files : ", err) |
|
||||||
return err |
|
||||||
} |
|
||||||
return nil |
|
||||||
} |
|
||||||
|
|
||||||
// reads contents of given dir; searches for .osu files; parses them for background info;
|
|
||||||
// removes original background and replaces it with copied version of given image
|
|
||||||
func replaceBackgrounds(beatmapFolder, replacementPicPath string) (successful, failed uint64) { |
|
||||||
files, err := os.ReadDir(beatmapFolder) |
|
||||||
if err != nil { |
|
||||||
log.Fatal("ERROR: Wrong path : ", err) |
|
||||||
} |
|
||||||
for _, file := range files { |
|
||||||
filename := file.Name() |
|
||||||
|
|
||||||
if isBeatmap(filename) { |
|
||||||
beatmap := filename |
|
||||||
|
|
||||||
// getting BG filename
|
|
||||||
beatmapBackgroundFilename, err := getBackgroundName(filepath.Join(beatmapFolder, beatmap)) |
|
||||||
if err != nil { |
|
||||||
log.Println(fmt.Sprintf("BEATMAP: %s: ERROR: Error getting background filename: %s", beatmap, err)) |
|
||||||
failed++ |
|
||||||
continue |
|
||||||
} |
|
||||||
if beatmapBackgroundFilename == "" { |
|
||||||
log.Println(fmt.Sprintf("BEATMAP: %s Could not find beatmap`s background filename", beatmap)) |
|
||||||
failed++ |
|
||||||
continue |
|
||||||
} |
|
||||||
|
|
||||||
backgroundPath := filepath.Join(beatmapFolder, beatmapBackgroundFilename) |
|
||||||
|
|
||||||
// remove old background
|
|
||||||
err = os.Remove(backgroundPath) |
|
||||||
if err != nil { |
|
||||||
failed++ |
|
||||||
log.Println(fmt.Sprintf("BEATMAP: %v: ERROR: Error removing old background : %s", beatmap, err)) |
|
||||||
} |
|
||||||
|
|
||||||
// create new background file
|
|
||||||
bgFile, err := os.Create(backgroundPath) |
|
||||||
if err != nil { |
|
||||||
failed++ |
|
||||||
log.Println(fmt.Sprintf("BEATMAP: %s: ERROR: Error creating new background file : %s", beatmap, err)) |
|
||||||
continue |
|
||||||
} |
|
||||||
defer bgFile.Close() |
|
||||||
|
|
||||||
// copy the contents of a given image to the newly created bg file
|
|
||||||
err = copyFile(replacementPicPath, backgroundPath) |
|
||||||
if err != nil { |
|
||||||
log.Println(fmt.Sprintf("BEATMAP: %s: ERROR: Error copying file: %s", beatmap, err)) |
|
||||||
failed++ |
|
||||||
continue |
|
||||||
} |
|
||||||
successful++ |
|
||||||
} |
|
||||||
|
|
||||||
} |
|
||||||
return successful, failed |
|
||||||
} |
|
||||||
|
|
||||||
// creates a complete black image file
|
|
||||||
func createBlackBG(width, height int) { |
|
||||||
bg, err := os.Create("blackBG.png") |
|
||||||
if err != nil { |
|
||||||
log.Println("ERROR: Error creating black background : ", err, "Continuing to run...") |
|
||||||
} |
|
||||||
image := image.NewRGBA(image.Rect(0, 0, width, height)) |
|
||||||
bounds := image.Bounds() |
|
||||||
|
|
||||||
for y := 0; y < bounds.Max.Y; y++ { |
|
||||||
for x := 0; x < bounds.Max.X; x++ { |
|
||||||
image.Set(x, y, color.Black) |
|
||||||
} |
|
||||||
} |
|
||||||
png.Encode(bg, image) |
|
||||||
bg.Close() |
|
||||||
|
|
||||||
log.Println("Successfully created black background") |
|
||||||
} |
|
||||||
|
|
||||||
// a basic implementation of a concurrent worker
|
|
||||||
func worker(paths <-chan string, replacementImage string, successful, failed *uint64, WG *sync.WaitGroup) { |
|
||||||
defer WG.Done() |
|
||||||
for songPath := range paths { |
|
||||||
s, f := replaceBackgrounds(songPath, replacementImage) |
|
||||||
*successful += s |
|
||||||
*failed += f |
|
||||||
} |
|
||||||
|
|
||||||
} |
|
||||||
|
|
||||||
func init() { |
|
||||||
setUpLogs() |
|
||||||
createSettingsFile() |
|
||||||
} |
|
||||||
|
|
||||||
func main() { |
|
||||||
// settings file didn`t exist, created now
|
|
||||||
if !settingsFileExisted { |
|
||||||
return |
|
||||||
} |
|
||||||
startingTime := time.Now().UTC() |
|
||||||
|
|
||||||
settings := getSettings() |
|
||||||
|
|
||||||
// process the given settings
|
|
||||||
if settings.CreateBlackBGImage == true { |
|
||||||
createBlackBG(1920, 1080) |
|
||||||
} |
|
||||||
|
|
||||||
osuSongsDir := getSongsDir(settings.OsuDir) |
|
||||||
|
|
||||||
replacementImage := settings.ReplacementImagePath |
|
||||||
if replacementImage == "" || replacementImage == " " { |
|
||||||
log.Fatal("Image path not specified ! Specify `pathToimage` in settings file !") |
|
||||||
} |
|
||||||
|
|
||||||
// reading contents of `Songs` folder
|
|
||||||
osuSongsDirContents, err := os.ReadDir(osuSongsDir) |
|
||||||
if err != nil { |
|
||||||
log.Fatal("ERROR: Error reading osu songs directory : ", err) |
|
||||||
} |
|
||||||
|
|
||||||
// storing all paths to each beatmap
|
|
||||||
songPaths := make(chan string, len(osuSongsDirContents)) |
|
||||||
for _, songDir := range osuSongsDirContents { |
|
||||||
if songDir.IsDir() { |
|
||||||
songPaths <- filepath.Join(osuSongsDir, songDir.Name()) |
|
||||||
} |
|
||||||
} |
|
||||||
log.Printf("Found %d song folders", len(songPaths)) |
|
||||||
|
|
||||||
// check if there is less job than workers
|
|
||||||
if int(settings.Workers) > len(songPaths) { |
|
||||||
settings.Workers = uint(len(songPaths)) |
|
||||||
} |
|
||||||
|
|
||||||
// replacing backgrounds for each beatmap concurrently
|
|
||||||
var successful, failed uint64 = 0, 0 |
|
||||||
for i := 0; i < int(settings.Workers); i++ { |
|
||||||
WG.Add(1) |
|
||||||
go worker(songPaths, replacementImage, &successful, &failed, &WG) |
|
||||||
} |
|
||||||
|
|
||||||
close(songPaths) |
|
||||||
WG.Wait() |
|
||||||
|
|
||||||
endTime := time.Now().UTC() |
|
||||||
|
|
||||||
log.Printf("\n\nDONE in %v . %d successful; %d failed", endTime.Sub(startingTime), successful, failed) |
|
||||||
|
|
||||||
} |
|
Loading…
Reference in new issue