1
0
Fork 0
mirror of https://github.com/fiso64/slsk-batchdl.git synced 2025-01-08 22:42:42 +00:00
This commit is contained in:
fiso64 2024-05-25 21:47:43 +02:00
parent 7293d45cac
commit bf4b356097
2 changed files with 76 additions and 104 deletions

View file

@ -1,6 +1,6 @@
# slsk-batchdl # slsk-batchdl
A batch downloader for Soulseek built with Soulseek.NET. Accepts CSV files or Spotify and YouTube urls. A batch downloader for Soulseek built with Soulseek.NET. Accepts CSV files and Spotify or YouTube urls.
## Examples ## Examples
@ -156,7 +156,7 @@ Options:
--pref-min-bitrate <rate> Preferred minimum bitrate (default: 200) --pref-min-bitrate <rate> Preferred minimum bitrate (default: 200)
--pref-max-bitrate <rate> Preferred maximum bitrate (default: 2200) --pref-max-bitrate <rate> Preferred maximum bitrate (default: 2200)
--pref-min-samplerate <rate> Preferred minimum sample rate --pref-min-samplerate <rate> Preferred minimum sample rate
--pref-max-samplerate <rate> Preferred maximum sample rate (default: 96000) --pref-max-samplerate <rate> Preferred maximum sample rate (default: 48000)
--pref-min-bitdepth <depth> Preferred minimum bit depth --pref-min-bitdepth <depth> Preferred minimum bit depth
--pref-max-bitdepth <depth> Preferred maximum bit depth --pref-max-bitdepth <depth> Preferred maximum bit depth
--pref-banned-users <list> Comma-separated list of users to deprioritize --pref-banned-users <list> Comma-separated list of users to deprioritize

View file

@ -5,8 +5,8 @@ using System.Collections.Concurrent;
using System.Data; using System.Data;
using System.Text.RegularExpressions; using System.Text.RegularExpressions;
using HtmlAgilityPack; using HtmlAgilityPack;
using System.Diagnostics;
using AngleSharp.Text; using AngleSharp.Text;
using System.Diagnostics;
using ProgressBar = Konsole.ProgressBar; using ProgressBar = Konsole.ProgressBar;
using SearchResponse = Soulseek.SearchResponse; using SearchResponse = Soulseek.SearchResponse;
@ -15,8 +15,6 @@ using SlFile = Soulseek.File;
using File = System.IO.File; using File = System.IO.File;
using Directory = System.IO.Directory; using Directory = System.IO.Directory;
using SlDictionary = System.Collections.Concurrent.ConcurrentDictionary<string, (Soulseek.SearchResponse, Soulseek.File)>; using SlDictionary = System.Collections.Concurrent.ConcurrentDictionary<string, (Soulseek.SearchResponse, Soulseek.File)>;
using System.Linq;
using System.IO;
// todo // todo
@ -45,7 +43,7 @@ static class Program
LengthTolerance = 2, LengthTolerance = 2,
MinBitrate = 200, MinBitrate = 200,
MaxBitrate = 2500, MaxBitrate = 2500,
MaxSampleRate = 95999, MaxSampleRate = 48000,
StrictTitle = true, StrictTitle = true,
StrictAlbum = true, StrictAlbum = true,
AcceptNoLength = false, AcceptNoLength = false,
@ -225,7 +223,7 @@ static class Program
"\n --pref-min-bitrate <rate> Preferred minimum bitrate (default: 200)" + "\n --pref-min-bitrate <rate> Preferred minimum bitrate (default: 200)" +
"\n --pref-max-bitrate <rate> Preferred maximum bitrate (default: 2200)" + "\n --pref-max-bitrate <rate> Preferred maximum bitrate (default: 2200)" +
"\n --pref-min-samplerate <rate> Preferred minimum sample rate" + "\n --pref-min-samplerate <rate> Preferred minimum sample rate" +
"\n --pref-max-samplerate <rate> Preferred maximum sample rate (default: 96000)" + "\n --pref-max-samplerate <rate> Preferred maximum sample rate (default: 48000)" +
"\n --pref-min-bitdepth <depth> Preferred minimum bit depth" + "\n --pref-min-bitdepth <depth> Preferred minimum bit depth" +
"\n --pref-max-bitdepth <depth> Preferred maximum bit depth" + "\n --pref-max-bitdepth <depth> Preferred maximum bit depth" +
"\n --pref-banned-users <list> Comma-separated list of users to deprioritize" + "\n --pref-banned-users <list> Comma-separated list of users to deprioritize" +
@ -1190,7 +1188,7 @@ static class Program
{ {
searchStr = input; searchStr = input;
inputType = "string"; inputType = "string";
var music = ParseTrackArg(searchStr, true, album); var music = ParseTrackArg(searchStr, album);
bool isAlbum = false; bool isAlbum = false;
if (album) if (album)
@ -3849,123 +3847,97 @@ static class Program
} }
static Track ParseTrackArg(string input, bool parseSingleString, bool isAlbum) // more complicated than it needs to be static Track ParseTrackArg(string input, bool isAlbum)
{ {
input = input.Trim(); input = input.Trim();
Track track = new Track(); var track = new Track();
List<string> keys = new List<string> { "title", "artist", "duration", "length", "album", "artist-maybe-wrong" }; var keys = new string[] { "title", "artist", "duration", "length", "album", "artist-maybe-wrong" };
if (!keys.Any(p => input.Replace(" ", "").Contains(p + "="))) track.IsAlbum = isAlbum;
var parts = input.Split(',');
var other = "";
var lastkeyval = true;
for (int i = 0; i < parts.Length; i++)
{ {
input = input.Replace(" — ", " - "); var x = parts[i];
if (!parseSingleString || !input.Contains(" - ")) bool keyval = false;
if (x.Contains('='))
{ {
track.Title = input; var lr = x.Split('=', 2, StringSplitOptions.TrimEntries);
if (lr.Length == 2 && lr[1] != "" && keys.Contains(lr[0]))
{
keyval = true;
switch (lr[0])
{
case "title":
track.Title = lr[1];
break;
case "artist":
track.Artist = lr[1];
break;
case "duration":
case "length":
track.Length = int.Parse(lr[1]);
break;
case "album":
track.Album = lr[1];
break;
case "artist-maybe-wrong":
if (lr[1] == "true")
track.ArtistMaybeWrong = true;
break;
}
}
}
if (!keyval)
{
if (!lastkeyval)
other += ',';
other += x;
lastkeyval = false;
} }
else else
{ {
var parts = input.Split(" - ", 3, StringSplitOptions.TrimEntries | StringSplitOptions.RemoveEmptyEntries); lastkeyval = true;
}
}
string artist = "", album = "", title = ""; string artist = "", album = "", title = "";
if (parts.Length <= 1) string splitBy = other.Contains(" -- ") ? " -- " : " - ";
parts = other.Split(splitBy, StringSplitOptions.TrimEntries | StringSplitOptions.RemoveEmptyEntries);
if (parts.Length == 1 || parts.Length > 3)
{ {
title = input; if (isAlbum)
album = input; album = other.Trim();
else
title = other.Trim();
} }
else if (parts.Length == 2) else if (parts.Length == 2)
{ {
artist = parts[0]; artist = parts[0];
if (isAlbum)
album = parts[1]; album = parts[1];
else
title = parts[1]; title = parts[1];
} }
else else if (parts.Length == 3)
{ {
artist = parts[0]; artist = parts[0];
album = parts[1]; album = parts[1];
title = parts[2]; title = parts[2];
} }
if (track.Artist == "")
track.Artist = artist; track.Artist = artist;
if (isAlbum) if (track.Album == "")
{
track.Album = album; track.Album = album;
track.IsAlbum = true; if (track.Title == "")
}
else
{
track.Title = title; track.Title = title;
if (input.Length == 3)
track.Album = album;
}
}
}
else
{
(int, int) getNextKeyIndices(int start)
{
int commaIndex = start;
int equalsIndex = input.IndexOf('=', commaIndex);
if (equalsIndex == -1)
return (-1, -1);
if (start == 0)
return keys.Any(k => k == input.Substring(0, equalsIndex).Trim()) ? (0, equalsIndex) : (-1, -1);
while (start < input.Length)
{
commaIndex = input.IndexOf(',', start);
equalsIndex = commaIndex != -1 ? input.IndexOf('=', commaIndex) : -1;
if (commaIndex == -1 || equalsIndex == -1)
return (-1, -1);
if (keys.Any(k => k == input.Substring(commaIndex + 1, equalsIndex - commaIndex - 1).Trim()))
return (commaIndex + 1, equalsIndex);
start = commaIndex + 1;
}
return (-1, -1);
}
(int start, int end) = getNextKeyIndices(0);
(int prevStart, int prevEnd) = (0, 0);
while (true)
{
if (prevEnd != 0)
{
string key = input.Substring(prevStart, prevEnd - prevStart);
int valEnd = start != -1 ? start - 1 : input.Length;
string val = input.Substring(prevEnd + 1, valEnd - prevEnd - 1);
switch (key)
{
case "title":
track.Title = val;
break;
case "artist":
track.Artist = val;
break;
case "duration":
case "length":
track.Length = (int)ParseTrackLength(val, "s");
break;
case "album":
track.Album = val;
break;
case "artist-maybe-wrong":
if (val == "true")
track.ArtistMaybeWrong = true;
break;
}
}
if (end == -1)
break;
(prevStart, prevEnd) = (start, end);
(start, end) = getNextKeyIndices(end);
}
}
if (track.Title == "" && track.Album == "" && track.Artist == "") if (track.Title == "" && track.Album == "" && track.Artist == "")
throw new ArgumentException("Track string must contain title, album or artist."); throw new ArgumentException("Track string must contain title, album or artist.");
@ -4436,10 +4408,10 @@ public struct Track
{ {
if (str != "") if (str != "")
str += " - "; str += " - ";
if (Title != "") if (IsAlbum)
str += Title;
else if (IsAlbum)
str += Album; str += Album;
else if (Title != "")
str += Title;
if (!noInfo) if (!noInfo)
{ {
if (Length > 0) if (Length > 0)