mirror of
https://github.com/fiso64/slsk-batchdl.git
synced 2024-12-31 18:52:41 +00:00
ignore " ", "_", "-", ".", "(", ")" when checking if a file exists
This commit is contained in:
parent
cabcfa946a
commit
b859d02a16
2 changed files with 64 additions and 103 deletions
10
README.md
10
README.md
|
@ -23,11 +23,6 @@ Options:
|
||||||
--create-m3u Create an m3u playlist file in the output dir
|
--create-m3u Create an m3u playlist file in the output dir
|
||||||
--m3u-only Only create an m3u playlist file with existing tracks and exit
|
--m3u-only Only create an m3u playlist file with existing tracks and exit
|
||||||
|
|
||||||
--search-timeout <timeout> Maximal search time (default: 15000)
|
|
||||||
--download-max-stale-time <time> Maximal download time with no progress (default: 60000)
|
|
||||||
--max-concurrent-processes <num> Max concurrent searches / downloads (default: 2)
|
|
||||||
--max-retries-per-file <num> Maximum number of users to try downloading from before skipping track (default: 30)
|
|
||||||
|
|
||||||
--pref-format <format> Preferred file format (default: mp3)
|
--pref-format <format> Preferred file format (default: mp3)
|
||||||
--pref-length-tolerance <tol> Preferred length tolerance (if length col provided) (default: 3)
|
--pref-length-tolerance <tol> Preferred length tolerance (if length col provided) (default: 3)
|
||||||
--pref-min-bitrate <rate> Preferred minimum bitrate (default: 200)
|
--pref-min-bitrate <rate> Preferred minimum bitrate (default: 200)
|
||||||
|
@ -38,6 +33,11 @@ Options:
|
||||||
--nec-min-bitrate <rate> Necessary minimum bitrate
|
--nec-min-bitrate <rate> Necessary minimum bitrate
|
||||||
--nec-max-bitrate <rate> Necessary maximum bitrate
|
--nec-max-bitrate <rate> Necessary maximum bitrate
|
||||||
--nec-max-sample-rate <rate> Necessary maximum sample rate
|
--nec-max-sample-rate <rate> Necessary maximum sample rate
|
||||||
|
|
||||||
|
--search-timeout <timeout> Maximal search time (default: 15000)
|
||||||
|
--download-max-stale-time <time> Maximal download time with no progress (default: 80000)
|
||||||
|
--max-concurrent-processes <num> Max concurrent searches / downloads (default: 2)
|
||||||
|
--max-retries-per-file <num> Maximum number of users to try downloading from before skipping track (default: 30)
|
||||||
```
|
```
|
||||||
- Provide either both a track-col and artist-col (ideally), or full-title-col in case separate artist and track names are unavailable. You can also specify --uploader-col (channel names) in that case to use as artist names whenever full-title-col doesn't contain them.
|
- Provide either both a track-col and artist-col (ideally), or full-title-col in case separate artist and track names are unavailable. You can also specify --uploader-col (channel names) in that case to use as artist names whenever full-title-col doesn't contain them.
|
||||||
- Always provide a length-col or get wrong results
|
- Always provide a length-col or get wrong results
|
||||||
|
|
|
@ -41,21 +41,20 @@ class Program
|
||||||
Console.WriteLine(" --csv <path> The csv file containing track information (in case it's not in the output folder)");
|
Console.WriteLine(" --csv <path> The csv file containing track information (in case it's not in the output folder)");
|
||||||
Console.WriteLine(" --username <username> Soulseek username");
|
Console.WriteLine(" --username <username> Soulseek username");
|
||||||
Console.WriteLine(" --password <password> Soulseek password");
|
Console.WriteLine(" --password <password> Soulseek password");
|
||||||
|
Console.WriteLine();
|
||||||
Console.WriteLine(" --artist-col <column> Specify if the csv file contains an artist name column");
|
Console.WriteLine(" --artist-col <column> Specify if the csv file contains an artist name column");
|
||||||
Console.WriteLine(" --track-col <column> Specify if if the csv file contains an track name column");
|
Console.WriteLine(" --track-col <column> Specify if if the csv file contains an track name column");
|
||||||
Console.WriteLine(" --full-title-col <column> Specify only if there are no separate artist and track name columns in the csv");
|
Console.WriteLine(" --full-title-col <column> Specify only if there are no separate artist and track name columns in the csv");
|
||||||
Console.WriteLine(" --uploader-col <column> Specify when using full title col if there is also an uploader column in the csv (fallback in case artist name cannot be extracted from title)");
|
Console.WriteLine(" --uploader-col <column> Specify when using full title col if there is also an uploader column in the csv (fallback in case artist name cannot be extracted from title)");
|
||||||
Console.WriteLine(" --length-col <column> Specify the name of the track duration column, if exists");
|
Console.WriteLine(" --length-col <column> Specify the name of the track duration column, if exists");
|
||||||
Console.WriteLine(" --time-unit <unit> Time unit for the track duration column, ms or s (default: s)");
|
Console.WriteLine(" --time-unit <unit> Time unit for the track duration column, ms or s (default: s)");
|
||||||
|
Console.WriteLine();
|
||||||
Console.WriteLine(" --skip-existing Skip if a track matching the conditions is found in the output folder or your music library (if provided)");
|
Console.WriteLine(" --skip-existing Skip if a track matching the conditions is found in the output folder or your music library (if provided)");
|
||||||
Console.WriteLine(" --music-dir <path> Specify to also skip downloading tracks which are in your library, use with --skip-existing");
|
Console.WriteLine(" --music-dir <path> Specify to also skip downloading tracks which are in your library, use with --skip-existing");
|
||||||
Console.WriteLine(" --skip-if-pref-failed Skip if preferred versions of a track exist but failed to download. If no pref. versions were found, download as normal.");
|
Console.WriteLine(" --skip-if-pref-failed Skip if preferred versions of a track exist but failed to download. If no pref. versions were found, download as normal.");
|
||||||
Console.WriteLine(" --create-m3u Create an m3u playlist file in the output dir");
|
Console.WriteLine(" --create-m3u Create an m3u playlist file in the output dir");
|
||||||
Console.WriteLine(" --m3u-only Only create an m3u playlist file with existing tracks and exit");
|
Console.WriteLine(" --m3u-only Only create an m3u playlist file with existing tracks and exit");
|
||||||
Console.WriteLine(" --search-timeout <timeout> Maximal search time (default: 15000)");
|
Console.WriteLine();
|
||||||
Console.WriteLine(" --download-max-stale-time <time> Maximal download time with no progress (default: 60000)");
|
|
||||||
Console.WriteLine(" --max-concurrent-processes <num> Max concurrent searches / downloads (default: 2)");
|
|
||||||
Console.WriteLine(" --max-retries-per-file <num> Maximum number of users to try downloading from before skipping track (default: 30)");
|
|
||||||
Console.WriteLine(" --pref-format <format> Preferred file format (default: mp3)");
|
Console.WriteLine(" --pref-format <format> Preferred file format (default: mp3)");
|
||||||
Console.WriteLine(" --pref-length-tolerance <tol> Preferred length tolerance (if length col provided) (default: 3)");
|
Console.WriteLine(" --pref-length-tolerance <tol> Preferred length tolerance (if length col provided) (default: 3)");
|
||||||
Console.WriteLine(" --pref-min-bitrate <rate> Preferred minimum bitrate (default: 200)");
|
Console.WriteLine(" --pref-min-bitrate <rate> Preferred minimum bitrate (default: 200)");
|
||||||
|
@ -66,10 +65,18 @@ class Program
|
||||||
Console.WriteLine(" --nec-min-bitrate <rate> Necessary minimum bitrate");
|
Console.WriteLine(" --nec-min-bitrate <rate> Necessary minimum bitrate");
|
||||||
Console.WriteLine(" --nec-max-bitrate <rate> Necessary maximum bitrate");
|
Console.WriteLine(" --nec-max-bitrate <rate> Necessary maximum bitrate");
|
||||||
Console.WriteLine(" --nec-max-sample-rate <rate> Necessary maximum sample rate");
|
Console.WriteLine(" --nec-max-sample-rate <rate> Necessary maximum sample rate");
|
||||||
|
Console.WriteLine();
|
||||||
|
Console.WriteLine(" --search-timeout <timeout> Maximal search time (default: 15000)");
|
||||||
|
Console.WriteLine(" --download-max-stale-time <time> Maximal download time with no progress (default: 80000)");
|
||||||
|
Console.WriteLine(" --max-concurrent-processes <num> Max concurrent searches / downloads (default: 2)");
|
||||||
|
Console.WriteLine(" --max-retries-per-file <num> Maximum number of users to try downloading from before skipping track (default: 30)");
|
||||||
}
|
}
|
||||||
|
|
||||||
static async Task Main(string[] args)
|
static async Task Main(string[] args)
|
||||||
{
|
{
|
||||||
|
AppDomain.CurrentDomain.UnhandledException += (sender, e) => {
|
||||||
|
Console.WriteLine($"{e.ExceptionObject}");
|
||||||
|
};
|
||||||
Console.OutputEncoding = System.Text.Encoding.UTF8;
|
Console.OutputEncoding = System.Text.Encoding.UTF8;
|
||||||
Console.WriteLine();
|
Console.WriteLine();
|
||||||
lastLine = Console.CursorTop;
|
lastLine = Console.CursorTop;
|
||||||
|
@ -79,20 +86,6 @@ class Program
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
//tracks = new List<Track>()
|
|
||||||
//{
|
|
||||||
// new Track { ArtistName = "Daft Punk", TrackTitle = "One More Time", Length = 320 },
|
|
||||||
// new Track { ArtistName = "The Chemical Brothers", TrackTitle = "Block Rockin' Beats", Length = 294 },
|
|
||||||
// new Track { ArtistName = "Fatboy Slim", TrackTitle = "Praise You", Length = 324 },
|
|
||||||
// new Track { ArtistName = "The Prodigy", TrackTitle = "Firestarter", Length = 279 },
|
|
||||||
// new Track { ArtistName = "Underworld", TrackTitle = "Born Slippy", Length = 456 },
|
|
||||||
// new Track { ArtistName = "Orbital", TrackTitle = "Chime", Length = 194 },
|
|
||||||
// new Track { ArtistName = "Aphex Twin", TrackTitle = "Windowlicker", Length = 365 },
|
|
||||||
// new Track { ArtistName = "Moby", TrackTitle = "Porcelain", Length = 238 },
|
|
||||||
// new Track { ArtistName = "Leftfield", TrackTitle = "Phat Planet", Length = 323 },
|
|
||||||
// new Track { ArtistName = "The Crystal Method", TrackTitle = "Busy Child", Length = 444 },
|
|
||||||
// new Track { ArtistName = "Real Musician", TrackTitle = "This track does not EXIST!!! LOL!!!! LMAO!!!! HAHAHAHHAA!!! !", Length = 324 }
|
|
||||||
//};
|
|
||||||
outputFolder = "";
|
outputFolder = "";
|
||||||
musicDir = "";
|
musicDir = "";
|
||||||
string tracksCsv = "";
|
string tracksCsv = "";
|
||||||
|
@ -109,7 +102,7 @@ class Program
|
||||||
bool createM3u = false;
|
bool createM3u = false;
|
||||||
bool m3uOnly = false;
|
bool m3uOnly = false;
|
||||||
int searchTimeout = 15000;
|
int searchTimeout = 15000;
|
||||||
downloadMaxStaleTime = 60000;
|
downloadMaxStaleTime = 80000;
|
||||||
int maxConcurrentProcesses = 2;
|
int maxConcurrentProcesses = 2;
|
||||||
int maxRetriesPerFile = 30;
|
int maxRetriesPerFile = 30;
|
||||||
var preferredCond = new FileConditions
|
var preferredCond = new FileConditions
|
||||||
|
@ -227,10 +220,6 @@ class Program
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
AppDomain.CurrentDomain.UnhandledException += (sender, e) => {
|
|
||||||
Console.WriteLine($"{e.ExceptionObject}");
|
|
||||||
};
|
|
||||||
|
|
||||||
if ((trackCol == "" && artistCol == "" && fullTitleCol == "") || (trackCol != "" && artistCol == "") || (fullTitleCol != "" && (artistCol != "" || trackCol != "")))
|
if ((trackCol == "" && artistCol == "" && fullTitleCol == "") || (trackCol != "" && artistCol == "") || (fullTitleCol != "" && (artistCol != "" || trackCol != "")))
|
||||||
throw new Exception("Use one of: full title column, (artist column AND track name)");
|
throw new Exception("Use one of: full title column, (artist column AND track name)");
|
||||||
if (lengthCol == "")
|
if (lengthCol == "")
|
||||||
|
@ -471,6 +460,7 @@ class Program
|
||||||
WriteLastLine($"Failed to find: {title}, skipping", ConsoleColor.Red);
|
WriteLastLine($"Failed to find: {title}, skipping", ConsoleColor.Red);
|
||||||
var failedDownloadInfo = $"{title} ({track.Length}s) [Reason: No file found with matching criteria]";
|
var failedDownloadInfo = $"{title} ({track.Length}s) [Reason: No file found with matching criteria]";
|
||||||
WriteLineOutputFile(failedDownloadInfo);
|
WriteLineOutputFile(failedDownloadInfo);
|
||||||
|
cts.Dispose();
|
||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -483,31 +473,42 @@ class Program
|
||||||
WriteLastLine($"Pref. version of the file exists, but couldn't be downloaded: {title}, skipping", ConsoleColor.Red);
|
WriteLastLine($"Pref. version of the file exists, but couldn't be downloaded: {title}, skipping", ConsoleColor.Red);
|
||||||
var failedDownloadInfo = $"{title} ({track.Length}s) [Preferred version of the file exists, but couldn't be downloaded]";
|
var failedDownloadInfo = $"{title} ({track.Length}s) [Preferred version of the file exists, but couldn't be downloaded]";
|
||||||
WriteLineOutputFile(failedDownloadInfo);
|
WriteLineOutputFile(failedDownloadInfo);
|
||||||
saveFilePath = "";
|
cts.Dispose();
|
||||||
break;
|
return "";
|
||||||
}
|
}
|
||||||
else if (pref)
|
|
||||||
attemptedDownloadPref = true;
|
|
||||||
|
|
||||||
saveFilePath = GetSavePath(x.file, track);
|
saveFilePath = GetSavePath(x.file, track);
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
downloading = true;
|
||||||
|
if (pref)
|
||||||
|
attemptedDownloadPref = true;
|
||||||
await DownloadFile(x.response, x.file, saveFilePath);
|
await DownloadFile(x.response, x.file, saveFilePath);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
catch
|
catch
|
||||||
{
|
{
|
||||||
|
downloading = false;
|
||||||
if (--downloadRetries <= 0)
|
if (--downloadRetries <= 0)
|
||||||
{
|
{
|
||||||
saveFilePath = "";
|
|
||||||
WriteLastLine($"Failed to download: {title}, skipping", ConsoleColor.Red);
|
WriteLastLine($"Failed to download: {title}, skipping", ConsoleColor.Red);
|
||||||
var failedDownloadInfo = $"{title} ({track.Length}s) [Reason: Out of download retries]";
|
var failedDownloadInfo = $"{title} ({track.Length}s) [Reason: Out of download retries]";
|
||||||
WriteLineOutputFile(failedDownloadInfo);
|
WriteLineOutputFile(failedDownloadInfo);
|
||||||
break;
|
cts.Dispose();
|
||||||
|
return "";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!downloading)
|
||||||
|
{
|
||||||
|
WriteLastLine($"Failed to download: {title}", ConsoleColor.Red);
|
||||||
|
var failedDownloadInfo = $"{title} ({track.Length}s) [Reason: All downloads failed]";
|
||||||
|
WriteLineOutputFile(failedDownloadInfo);
|
||||||
|
cts.Dispose();
|
||||||
|
return "";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
cts.Dispose();
|
cts.Dispose();
|
||||||
|
@ -612,10 +613,7 @@ class Program
|
||||||
static string GetSavePath(Soulseek.File file, Track track)
|
static string GetSavePath(Soulseek.File file, Track track)
|
||||||
{
|
{
|
||||||
string name = track.TrackTitle == "" ? $"{track.UnparsedTitle}" : $"{track.ArtistName} - {track.TrackTitle}";
|
string name = track.TrackTitle == "" ? $"{track.UnparsedTitle}" : $"{track.ArtistName} - {track.TrackTitle}";
|
||||||
char[] invalidChars = Path.GetInvalidFileNameChars();
|
return Path.Combine(outputFolder, $"{RemoveInvalidChars(name, " ")}{Path.GetExtension(file.Filename)}");
|
||||||
foreach (char c in invalidChars)
|
|
||||||
name = name.Replace(c, ' ');
|
|
||||||
return Path.Combine(outputFolder, $"{name}{Path.GetExtension(file.Filename)}");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
struct Track
|
struct Track
|
||||||
|
@ -628,61 +626,6 @@ class Program
|
||||||
public Track() { }
|
public Track() { }
|
||||||
}
|
}
|
||||||
|
|
||||||
//class ConsoleProcessDisplay
|
|
||||||
//{
|
|
||||||
// public DownloadInfo? download;
|
|
||||||
// public SearchInfo? search;
|
|
||||||
// public string displayText = "";
|
|
||||||
// public bool finished = false;
|
|
||||||
|
|
||||||
// private int rotatingBarState = 0;
|
|
||||||
// private int displayPos = 0;
|
|
||||||
|
|
||||||
// public ConsoleProcessDisplay(DownloadInfo? download, SearchInfo? search)
|
|
||||||
// {
|
|
||||||
// this.download = download;
|
|
||||||
// this.search = search;
|
|
||||||
// MoveCursorLastLine();
|
|
||||||
// displayPos = Console.CursorTop;
|
|
||||||
// }
|
|
||||||
|
|
||||||
// public void UpdateText()
|
|
||||||
// {
|
|
||||||
// Console.SetCursorPosition(0, displayPos);
|
|
||||||
// if (finished && (download == null || !download.success))
|
|
||||||
// {
|
|
||||||
// Console.WriteLine(displayText);
|
|
||||||
// return;
|
|
||||||
// }
|
|
||||||
|
|
||||||
// char[] bars = { '/', '|', '\\', '—' };
|
|
||||||
// rotatingBarState++;
|
|
||||||
// rotatingBarState %= bars.Length;
|
|
||||||
// string bar = bars[rotatingBarState] + " ";
|
|
||||||
|
|
||||||
// if (download != null)
|
|
||||||
// {
|
|
||||||
// string sampleRate = download.file.SampleRate.HasValue ? $" / {download.file.SampleRate}Hz" : "";
|
|
||||||
// string bitRate = download.file.BitRate.HasValue ? $" / {download.file.BitRate}kbps" : "";
|
|
||||||
// string fileSize = $"{download.file.Size / (float)(1024 * 1024):F1}MB";
|
|
||||||
// displayText = $"{download.response.Username}\\..\\{download.file.Filename.Split('\\').Last()} " +
|
|
||||||
// $"[{download.file.Length}s{sampleRate}{bitRate} / {fileSize}]";
|
|
||||||
// float percentage = download.bytesTransferred / (float)download.file.Size;
|
|
||||||
// //queued = transfer?.State is TransferStates.Remotely or TransferStates.Locally or TransferStates.Queued;
|
|
||||||
// Console.WriteLine($"{(download.success ? bar : "")}[{percentage:P}] {download?.transfer?.State}: {displayText}");
|
|
||||||
// }
|
|
||||||
// else if (search != null)
|
|
||||||
// {
|
|
||||||
// Console.WriteLine($"{bar}Searching for: {search.query.SearchText}");
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
|
|
||||||
// public bool IsDone()
|
|
||||||
// {
|
|
||||||
// return download == null && search == null;
|
|
||||||
// }
|
|
||||||
//}
|
|
||||||
|
|
||||||
class DownloadInfo
|
class DownloadInfo
|
||||||
{
|
{
|
||||||
public string savePath;
|
public string savePath;
|
||||||
|
@ -903,18 +846,18 @@ class Program
|
||||||
}
|
}
|
||||||
static bool IsMusicFile(string fileName)
|
static bool IsMusicFile(string fileName)
|
||||||
{
|
{
|
||||||
var musicExtensions = new string[] { ".mp3", ".wav", ".flac", ".ogg", ".aac", ".wma", ".m4a", ".alac", ".ape", ".dsd", ".dff", ".dsf", ".ogg", ".opus" };
|
var musicExtensions = new string[] { ".mp3", ".wav", ".flac", ".ogg", ".aac", ".wma", ".m4a", ".alac", ".ape", ".dsd", ".dff", ".dsf", ".opus" };
|
||||||
var extension = Path.GetExtension(fileName).ToLower();
|
var extension = Path.GetExtension(fileName).ToLower();
|
||||||
return musicExtensions.Contains(extension);
|
return musicExtensions.Contains(extension);
|
||||||
}
|
}
|
||||||
static bool FileExistsInCollection(string searchName, int length, FileConditions conditions, IEnumerable<string> collection, out string? foundPath)
|
static bool FileExistsInCollection(string searchName, int length, FileConditions conditions, IEnumerable<string> collection, out string? foundPath)
|
||||||
{
|
{
|
||||||
char[] invalidChars = Path.GetInvalidFileNameChars();
|
string[] ignore = new string[] { " ", "_", "-", ".", "(", ")" };
|
||||||
foreach (char c in invalidChars)
|
searchName = searchName.Replace(ignore, "");
|
||||||
searchName = searchName.Replace(c.ToString(), "");
|
searchName = RemoveInvalidChars(searchName, "");
|
||||||
searchName = searchName.Replace(" ", "");
|
|
||||||
|
|
||||||
var matchingFiles = collection.Where(fileName => fileName.Replace(" ", "").Contains(searchName, StringComparison.OrdinalIgnoreCase)).ToArray();
|
var matchingFiles = collection
|
||||||
|
.Where(fileName => fileName.Replace(ignore, "").Contains(searchName, StringComparison.OrdinalIgnoreCase)).ToArray();
|
||||||
|
|
||||||
foreach (var p in matchingFiles)
|
foreach (var p in matchingFiles)
|
||||||
{
|
{
|
||||||
|
@ -932,7 +875,8 @@ class Program
|
||||||
if (searchName.Count(c => c == '-') == 1)
|
if (searchName.Count(c => c == '-') == 1)
|
||||||
{
|
{
|
||||||
searchName = searchName.Split('-')[1];
|
searchName = searchName.Split('-')[1];
|
||||||
matchingFiles = collection.Where(fileName => fileName.Replace(" ", "").Contains(searchName, StringComparison.OrdinalIgnoreCase)).ToArray();
|
matchingFiles = collection
|
||||||
|
.Where(fileName => fileName.Replace(ignore, "").Contains(searchName, StringComparison.OrdinalIgnoreCase)).ToArray();
|
||||||
|
|
||||||
foreach (var p in matchingFiles)
|
foreach (var p in matchingFiles)
|
||||||
{
|
{
|
||||||
|
@ -979,4 +923,21 @@ class Program
|
||||||
args[i] = matches[i].Value.Trim('"');
|
args[i] = matches[i].Value.Trim('"');
|
||||||
return args;
|
return args;
|
||||||
}
|
}
|
||||||
|
static string RemoveInvalidChars(string str, string replaceStr)
|
||||||
|
{
|
||||||
|
char[] invalidChars = Path.GetInvalidFileNameChars();
|
||||||
|
foreach (char c in invalidChars)
|
||||||
|
str = str.Replace(c.ToString(), replaceStr);
|
||||||
|
return str;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class ExtensionMethods
|
||||||
|
{
|
||||||
|
public static string Replace(this string s, string[] separators, string newVal)
|
||||||
|
{
|
||||||
|
string[] temp;
|
||||||
|
temp = s.Split(separators, StringSplitOptions.RemoveEmptyEntries);
|
||||||
|
return String.Join(newVal, temp);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue