mirror of
https://github.com/fiso64/slsk-batchdl.git
synced 2025-01-10 23:42:42 +00:00
commit
This commit is contained in:
parent
923cd2cbf5
commit
c0b9a859ee
20 changed files with 677 additions and 593 deletions
50
README.md
50
README.md
|
@ -11,6 +11,7 @@ See the [examples](#examples-1).
|
||||||
- [Spotify](#spotify)
|
- [Spotify](#spotify)
|
||||||
- [Bandcamp](#bandcamp)
|
- [Bandcamp](#bandcamp)
|
||||||
- [Search string](#search-string)
|
- [Search string](#search-string)
|
||||||
|
- [List](#list)
|
||||||
- [Download modes](#download-modes)
|
- [Download modes](#download-modes)
|
||||||
- [Normal](#normal)
|
- [Normal](#normal)
|
||||||
- [Album](#album)
|
- [Album](#album)
|
||||||
|
@ -51,7 +52,7 @@ Usage: sldl <input> [OPTIONS]
|
||||||
--profile <names> Configuration profile(s) to use. See --help "config".
|
--profile <names> Configuration profile(s) to use. See --help "config".
|
||||||
--concurrent-downloads <num> Max concurrent downloads (default: 2)
|
--concurrent-downloads <num> Max concurrent downloads (default: 2)
|
||||||
--m3u <option> Create an m3u8 playlist file in the output directory
|
--m3u <option> Create an m3u8 playlist file in the output directory
|
||||||
'none' (default for single inputs): Do not create
|
'none' (default for string input): Do not create
|
||||||
'index' (default): Write a line indexing all downloaded
|
'index' (default): Write a line indexing all downloaded
|
||||||
files, required for skip-not-found or skip-existing=m3u
|
files, required for skip-not-found or skip-existing=m3u
|
||||||
'all': Write the index and a list of paths and fails
|
'all': Write the index and a list of paths and fails
|
||||||
|
@ -234,8 +235,8 @@ The input type is usually determined automatically. To force a specific input ty
|
||||||
Path to a local CSV file: Use a csv file containing track info of the songs to download.
|
Path to a local CSV file: Use a csv file containing track info of the songs to download.
|
||||||
The names of the columns should be Artist, Title, Album, Length, although alternative names
|
The names of the columns should be Artist, Title, Album, Length, although alternative names
|
||||||
are usually detected as well. Only the title or album column is required, but extra info may
|
are usually detected as well. Only the title or album column is required, but extra info may
|
||||||
improve search results. Every row that does not have a title column text will be treated as an
|
improve search result ranking. Every row that does not have a title column text will be treated
|
||||||
album download.
|
as an album download.
|
||||||
|
|
||||||
### YouTube
|
### YouTube
|
||||||
A playlist url: Download songs from a youtube playlist.
|
A playlist url: Download songs from a youtube playlist.
|
||||||
|
@ -244,10 +245,6 @@ the ones which are unavailable. To get all video titles, you can use the officia
|
||||||
providing a key with --youtube-key. Get it here https://console.cloud.google.com. Create a
|
providing a key with --youtube-key. Get it here https://console.cloud.google.com. Create a
|
||||||
new project, click "Enable Api" and search for "youtube data", then follow the prompts.
|
new project, click "Enable Api" and search for "youtube data", then follow the prompts.
|
||||||
|
|
||||||
Tip: For playlists containing music videos, it may be better to remove all text in parentheses
|
|
||||||
(to remove (Lyrics), (Official), etc) and disable song duration checking:
|
|
||||||
--regex "[\[\(].*?[\]\)]" --pref-length-tol -1
|
|
||||||
|
|
||||||
### Spotify
|
### Spotify
|
||||||
A playlist/album url or 'spotify-likes': Download a spotify playlist, album, or your
|
A playlist/album url or 'spotify-likes': Download a spotify playlist, album, or your
|
||||||
liked songs. Credentials are required when downloading a private playlist or liked music.
|
liked songs. Credentials are required when downloading a private playlist or liked music.
|
||||||
|
@ -285,7 +282,7 @@ Name of the track, album, or artist to search for: Can either be any typical sea
|
||||||
(like what you would enter into the soulseek search bar), or a comma-separated list of
|
(like what you would enter into the soulseek search bar), or a comma-separated list of
|
||||||
properties like 'title=Song Name, artist=Artist Name, length=215'.
|
properties like 'title=Song Name, artist=Artist Name, length=215'.
|
||||||
|
|
||||||
The following properties are allowed:
|
The following properties are accepted:
|
||||||
```
|
```
|
||||||
title
|
title
|
||||||
artist
|
artist
|
||||||
|
@ -336,7 +333,7 @@ one user will be ignored.
|
||||||
### Album Aggregate
|
### Album Aggregate
|
||||||
Activated when both --album and --aggregate are enabled. sldl will group shares and download
|
Activated when both --album and --aggregate are enabled. sldl will group shares and download
|
||||||
one of each distinct album, starting with the one shared by the most users. It's
|
one of each distinct album, starting with the one shared by the most users. It's
|
||||||
recommended to pair this with --interactive.
|
recommended to pair this with --interactive.
|
||||||
Note that --min-shares-aggregate is 2 by default, which means that albums shared by only
|
Note that --min-shares-aggregate is 2 by default, which means that albums shared by only
|
||||||
one user will be ignored.
|
one user will be ignored.
|
||||||
|
|
||||||
|
@ -452,6 +449,7 @@ disc Disc number
|
||||||
filename Soulseek filename without extension
|
filename Soulseek filename without extension
|
||||||
foldername Soulseek folder name
|
foldername Soulseek folder name
|
||||||
extractor Name of the extractor used (CSV/Spotify/YouTube/etc)
|
extractor Name of the extractor used (CSV/Spotify/YouTube/etc)
|
||||||
|
default-folder Default sldl folder name (usually the playlist name)
|
||||||
```
|
```
|
||||||
|
|
||||||
## Skip existing
|
## Skip existing
|
||||||
|
@ -547,20 +545,20 @@ sldl spotify-likes
|
||||||
|
|
||||||
Download albums for every song in a spotify playlist:
|
Download albums for every song in a spotify playlist:
|
||||||
```
|
```
|
||||||
sldl https://spotify/playlist/url --album --skip-existing
|
sldl https://spotify/playlist/id --album --skip-existing
|
||||||
```
|
```
|
||||||
|
|
||||||
<br>
|
<br>
|
||||||
|
|
||||||
Retrieve deleted video names, then download from a youtube playlist with fallback to yt-dlp:
|
Retrieve deleted video names, then download from a youtube playlist with fallback to yt-dlp:
|
||||||
```
|
```
|
||||||
sldl "https://www.youtube.com/playlist?list=PLI_eFW8NAFzYAXZ5DrU6E6mQ_XfhaLBUX" --get-deleted --yt-dlp
|
sldl https://www.youtube.com/playlist/id --get-deleted --yt-dlp
|
||||||
```
|
```
|
||||||
<br>
|
<br>
|
||||||
|
|
||||||
Search & download a specific song, preferring lossless:
|
Search & download a specific song, preferring lossless:
|
||||||
```
|
```
|
||||||
sldl "title=MC MENTAL @ HIS BEST,length=242" --pref-format "flac,wav"
|
sldl "MC MENTAL @ HIS BEST, length=242" --pref-format "flac,wav"
|
||||||
```
|
```
|
||||||
<br>
|
<br>
|
||||||
|
|
||||||
|
@ -578,23 +576,31 @@ sldl "artist=MC MENTAL" --aggregate --skip-existing --music-dir "path/to/music"
|
||||||
|
|
||||||
Download all albums by an artist found on soulseek:
|
Download all albums by an artist found on soulseek:
|
||||||
```
|
```
|
||||||
sldl "artist=MC MENTAL" --aggregate --album
|
sldl "artist=MC MENTAL" --aggregate --album --interactive
|
||||||
```
|
```
|
||||||
<hr style="height:0px; visibility:hidden;" />
|
|
||||||
|
|
||||||
#### Advanced example: Automatic wishlist downloader
|
#### Advanced example: Automatic wishlist downloader
|
||||||
Create a file named `wishlist.txt`, and add some wishlist items:
|
Create a file named `wishlist.txt`, and add some items as detailed in [Input types: List](#list):
|
||||||
```bash
|
```bash
|
||||||
echo title=My Favorite Song, artist=Artist >> wishlist.txt
|
echo "title=My Favorite Song, artist=Artist" >> wishlist.txt
|
||||||
echo https://spotify/album/url >> wishlist.txt
|
echo "album=Album" "format=mp3" >> wishlist.txt
|
||||||
```
|
```
|
||||||
Set up a cron job (or scheduled task on windows) to periodically run sldl on every line of the wishlist file with the following options:
|
Add a profile to your `sldl.conf`:
|
||||||
```
|
```
|
||||||
--skip-existing --skip-mode m3u --m3u index --m3u-path wishlist-archive.sldl
|
[wishlist]
|
||||||
|
input = wishlist.txt
|
||||||
|
input-type = list
|
||||||
|
skip-existing = true
|
||||||
|
skip-mode = m3u
|
||||||
|
m3u = index
|
||||||
|
m3u-path = wishlist-archive.sldl
|
||||||
```
|
```
|
||||||
This will create a global archive file `wishlist-archive.sldl` which will be scanned every time sldl is run to skip wishlist items that have already been downloaded.
|
This will create a global archive file `wishlist-archive.sldl` which will be scanned every time sldl is run to skip wishlist items that have already been downloaded. You can also use `--skip-mode m3u-cond` together with `--skip-existing-pref-cond` and specify some preferred conditions to (e.g) only stop searching for an item once a lossless version is downloaded.
|
||||||
You can also use `--skip-mode m3u-cond` together with `--skip-existing-pref-cond` and specify some preferred conditions to (e.g) only stop searching for an item once a lossless version is downloaded.
|
Finally, set up a cron job (or a scheduled task on windows) to periodically run sldl with the following option:
|
||||||
If you expect to have a lot of individual songs in your wishlist, it may be better to use a csv file as that will allow sldl to use concurrency when downloading.
|
```
|
||||||
|
sldl --profile wishlist
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
## Notes
|
## Notes
|
||||||
- For macOS builds you can use publish.sh to build the app. Download dotnet from https://dotnet.microsoft.com/en-us/download/dotnet/6.0, then run `chmod +x publish.sh && sh publish.sh`. For intel macs, uncomment the x64 and comment the arm64 section in publish.sh.
|
- For macOS builds you can use publish.sh to build the app. Download dotnet from https://dotnet.microsoft.com/en-us/download/dotnet/6.0, then run `chmod +x publish.sh && sh publish.sh`. For intel macs, uncomment the x64 and comment the arm64 section in publish.sh.
|
||||||
|
|
|
@ -5,11 +5,11 @@ using System.Text;
|
||||||
using System.Text.RegularExpressions;
|
using System.Text.RegularExpressions;
|
||||||
|
|
||||||
|
|
||||||
static class Config
|
public class Config
|
||||||
{
|
{
|
||||||
public static FileConditions necessaryCond = new();
|
public FileConditions necessaryCond = new();
|
||||||
|
|
||||||
public static FileConditions preferredCond = new()
|
public FileConditions preferredCond = new()
|
||||||
{
|
{
|
||||||
Formats = new string[] { "mp3" },
|
Formats = new string[] { "mp3" },
|
||||||
LengthTolerance = 3,
|
LengthTolerance = 3,
|
||||||
|
@ -21,180 +21,174 @@ static class Config
|
||||||
AcceptNoLength = false,
|
AcceptNoLength = false,
|
||||||
};
|
};
|
||||||
|
|
||||||
public static string parentDir = Directory.GetCurrentDirectory();
|
public string parentDir = Directory.GetCurrentDirectory();
|
||||||
public static string input = "";
|
public string input = "";
|
||||||
public static string m3uFilePath = "";
|
public string m3uFilePath = "";
|
||||||
public static string musicDir = "";
|
public string musicDir = "";
|
||||||
public static string spotifyId = "";
|
public string spotifyId = "";
|
||||||
public static string spotifySecret = "";
|
public string spotifySecret = "";
|
||||||
public static string spotifyToken = "";
|
public string spotifyToken = "";
|
||||||
public static string spotifyRefresh = "";
|
public string spotifyRefresh = "";
|
||||||
public static string ytKey = "";
|
public string ytKey = "";
|
||||||
public static string username = "";
|
public string username = "";
|
||||||
public static string password = "";
|
public string password = "";
|
||||||
public static string artistCol = "";
|
public string artistCol = "";
|
||||||
public static string albumCol = "";
|
public string albumCol = "";
|
||||||
public static string trackCol = "";
|
public string trackCol = "";
|
||||||
public static string ytIdCol = "";
|
public string ytIdCol = "";
|
||||||
public static string descCol = "";
|
public string descCol = "";
|
||||||
public static string trackCountCol = "";
|
public string trackCountCol = "";
|
||||||
public static string lengthCol = "";
|
public string lengthCol = "";
|
||||||
public static string timeUnit = "s";
|
public string timeUnit = "s";
|
||||||
public static string nameFormat = "";
|
public string nameFormat = "";
|
||||||
public static string invalidReplaceStr = " ";
|
public string invalidReplaceStr = " ";
|
||||||
public static string ytdlpArgument = "";
|
public string ytdlpArgument = "";
|
||||||
public static string onComplete = "";
|
public string onComplete = "";
|
||||||
public static string confPath = "";
|
public string confPath = "";
|
||||||
public static string profile = "";
|
public string profile = "";
|
||||||
public static string failedAlbumPath = "";
|
public string failedAlbumPath = "";
|
||||||
public static bool aggregate = false;
|
public bool aggregate = false;
|
||||||
public static bool album = false;
|
public bool album = false;
|
||||||
public static bool albumArtOnly = false;
|
public bool albumArtOnly = false;
|
||||||
public static bool interactiveMode = false;
|
public bool interactiveMode = false;
|
||||||
public static bool setAlbumMinTrackCount = true;
|
public bool setAlbumMinTrackCount = true;
|
||||||
public static bool setAlbumMaxTrackCount = false;
|
public bool setAlbumMaxTrackCount = false;
|
||||||
public static bool skipNotFound = false;
|
public bool skipNotFound = false;
|
||||||
public static bool desperateSearch = false;
|
public bool desperateSearch = false;
|
||||||
public static bool noRemoveSpecialChars = false;
|
public bool noRemoveSpecialChars = false;
|
||||||
public static bool artistMaybeWrong = false;
|
public bool artistMaybeWrong = false;
|
||||||
public static bool fastSearch = false;
|
public bool fastSearch = false;
|
||||||
public static bool ytParse = false;
|
public bool ytParse = false;
|
||||||
public static bool removeFt = false;
|
public bool removeFt = false;
|
||||||
public static bool removeBrackets = false;
|
public bool removeBrackets = false;
|
||||||
public static bool reverse = false;
|
public bool reverse = false;
|
||||||
public static bool useYtdlp = false;
|
public bool useYtdlp = false;
|
||||||
public static bool skipExisting = false;
|
public bool skipExisting = false;
|
||||||
public static bool removeTracksFromSource = false;
|
public bool removeTracksFromSource = false;
|
||||||
public static bool getDeleted = false;
|
public bool getDeleted = false;
|
||||||
public static bool deletedOnly = false;
|
public bool deletedOnly = false;
|
||||||
public static bool removeSingleCharacterSearchTerms = false;
|
public bool removeSingleCharacterSearchTerms = false;
|
||||||
public static bool relax = false;
|
public bool relax = false;
|
||||||
public static bool debugInfo = false;
|
public bool debugInfo = false;
|
||||||
public static bool noModifyShareCount = false;
|
public bool noModifyShareCount = false;
|
||||||
public static bool useRandomLogin = false;
|
public bool useRandomLogin = false;
|
||||||
public static bool noBrowseFolder = false;
|
public bool noBrowseFolder = false;
|
||||||
public static bool skipExistingPrefCond = false;
|
public bool skipExistingPrefCond = false;
|
||||||
public static int downrankOn = -1;
|
public int downrankOn = -1;
|
||||||
public static int ignoreOn = -2;
|
public int ignoreOn = -2;
|
||||||
public static int minAlbumTrackCount = -1;
|
public int minAlbumTrackCount = -1;
|
||||||
public static int maxAlbumTrackCount = -1;
|
public int maxAlbumTrackCount = -1;
|
||||||
public static int fastSearchDelay = 300;
|
public int fastSearchDelay = 300;
|
||||||
public static int minSharesAggregate = 2;
|
public int minSharesAggregate = 2;
|
||||||
public static int maxTracks = int.MaxValue;
|
public int maxTracks = int.MaxValue;
|
||||||
public static int offset = 0;
|
public int offset = 0;
|
||||||
public static int maxStaleTime = 50000;
|
public int maxStaleTime = 50000;
|
||||||
public static int updateDelay = 100;
|
public int updateDelay = 100;
|
||||||
public static int searchTimeout = 6000;
|
public int searchTimeout = 6000;
|
||||||
public static int concurrentProcesses = 2;
|
public int concurrentProcesses = 2;
|
||||||
public static int unknownErrorRetries = 2;
|
public int unknownErrorRetries = 2;
|
||||||
public static int maxRetriesPerTrack = 30;
|
public int maxRetriesPerTrack = 30;
|
||||||
public static int listenPort = 49998;
|
public int listenPort = 49998;
|
||||||
public static int searchesPerTime = 34;
|
public int searchesPerTime = 34;
|
||||||
public static int searchRenewTime = 220;
|
public int searchRenewTime = 220;
|
||||||
public static int aggregateLengthTol = 3;
|
public int aggregateLengthTol = 3;
|
||||||
public static double fastSearchMinUpSpeed = 1.0;
|
public double fastSearchMinUpSpeed = 1.0;
|
||||||
public static Track regexToReplace = new();
|
public Track regexToReplace = new();
|
||||||
public static Track regexReplaceBy = new();
|
public Track regexReplaceBy = new();
|
||||||
public static AlbumArtOption albumArtOption = AlbumArtOption.Default;
|
public AlbumArtOption albumArtOption = AlbumArtOption.Default;
|
||||||
public static M3uOption m3uOption = M3uOption.Index;
|
public M3uOption m3uOption = M3uOption.Index;
|
||||||
public static DisplayMode displayMode = DisplayMode.Single;
|
public DisplayMode displayMode = DisplayMode.Single;
|
||||||
public static InputType inputType = InputType.None;
|
public InputType inputType = InputType.None;
|
||||||
public static SkipMode skipMode = SkipMode.M3u;
|
public SkipMode skipMode = SkipMode.M3u;
|
||||||
public static SkipMode skipModeMusicDir = SkipMode.Name;
|
public SkipMode skipModeMusicDir = SkipMode.Name;
|
||||||
public static PrintOption printOption = PrintOption.None;
|
public PrintOption printOption = PrintOption.None;
|
||||||
|
|
||||||
public static bool HasAutoProfiles { get; private set; } = false;
|
public bool HasAutoProfiles { get; private set; } = false;
|
||||||
public static bool DoNotDownload => (printOption & (PrintOption.Results | PrintOption.Tracks)) != 0;
|
public bool DoNotDownload => (printOption & (PrintOption.Results | PrintOption.Tracks)) != 0;
|
||||||
public static bool PrintTracks => (printOption & PrintOption.Tracks) != 0;
|
public bool PrintTracks => (printOption & PrintOption.Tracks) != 0;
|
||||||
public static bool PrintResults => (printOption & PrintOption.Results) != 0;
|
public bool PrintResults => (printOption & PrintOption.Results) != 0;
|
||||||
public static bool PrintTracksFull => (printOption & PrintOption.Tracks) != 0 && (printOption & PrintOption.Full) != 0;
|
public bool PrintTracksFull => (printOption & PrintOption.Tracks) != 0 && (printOption & PrintOption.Full) != 0;
|
||||||
public static bool PrintResultsFull => (printOption & PrintOption.Results) != 0 && (printOption & PrintOption.Full) != 0;
|
public bool PrintResultsFull => (printOption & PrintOption.Results) != 0 && (printOption & PrintOption.Full) != 0;
|
||||||
public static bool DeleteAlbumOnFail => failedAlbumPath == "delete";
|
public bool DeleteAlbumOnFail => failedAlbumPath == "delete";
|
||||||
public static bool IgnoreAlbumFail => failedAlbumPath == "disable";
|
public bool IgnoreAlbumFail => failedAlbumPath == "disable";
|
||||||
|
|
||||||
static readonly Dictionary<string, (List<string> args, string? cond)> profiles = new();
|
readonly Dictionary<string, (List<string> args, string? cond)> configProfiles = new();
|
||||||
static readonly HashSet<string> appliedProfiles = new();
|
readonly HashSet<string> appliedProfiles = new();
|
||||||
static bool hasConfiguredM3uMode = false;
|
bool hasConfiguredM3uMode = false;
|
||||||
static bool confPathChanged = false;
|
bool confPathChanged = false;
|
||||||
static string[] arguments;
|
string[] arguments;
|
||||||
static FileConditions? prevConds = null;
|
FileConditionsMod? undoTempConds = null;
|
||||||
static FileConditions? prevPrefConds = null;
|
FileConditionsMod? undoTempPrefConds = null;
|
||||||
|
|
||||||
public static bool ParseArgsAndReadConfig(string[] args)
|
private static Config Instance = new();
|
||||||
|
|
||||||
|
public static Config I { get { return Instance; } }
|
||||||
|
|
||||||
|
private Config() { }
|
||||||
|
|
||||||
|
private Config(Dictionary<string, (List<string> args, string? cond)> cfg, string[] args)
|
||||||
|
{
|
||||||
|
configProfiles = cfg;
|
||||||
|
arguments = args;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public void Load(string[] args)
|
||||||
{
|
{
|
||||||
args = args.SelectMany(arg =>
|
arguments = args.SelectMany(arg =>
|
||||||
{
|
{
|
||||||
if (arg.Length > 3 && arg.StartsWith("--") && arg.Contains('='))
|
if (arg.Length > 2 && arg[0] == '-')
|
||||||
{
|
{
|
||||||
var parts = arg.Split('=', 2);
|
if (arg[1] == '-')
|
||||||
return new[] { parts[0], parts[1] };
|
{
|
||||||
}
|
if (arg.Length > 3 && arg.Contains('='))
|
||||||
|
return arg.Split('=', 2); // --arg=val becomes --arg val
|
||||||
|
}
|
||||||
|
else if (!arg.Contains(' '))
|
||||||
|
{
|
||||||
|
return arg[1..].Select(c => $"-{c}"); // -abc becomes -a -b -c
|
||||||
|
}
|
||||||
|
}
|
||||||
return new[] { arg };
|
return new[] { arg };
|
||||||
}).ToArray();
|
}).ToArray();
|
||||||
|
|
||||||
SetConfigPath(args);
|
SetConfigPath(arguments);
|
||||||
|
|
||||||
if (confPath != "none" && (confPathChanged || File.Exists(confPath)))
|
if (confPath != "none" && (confPathChanged || File.Exists(confPath)))
|
||||||
{
|
{
|
||||||
if (File.Exists(Path.Join(AppDomain.CurrentDomain.BaseDirectory, confPath)))
|
|
||||||
confPath = Path.Join(AppDomain.CurrentDomain.BaseDirectory, confPath);
|
|
||||||
ParseConfig(confPath);
|
ParseConfig(confPath);
|
||||||
|
ApplyDefaultConfig();
|
||||||
}
|
}
|
||||||
|
|
||||||
args = args.SelectMany(arg =>
|
int profileIndex = Array.FindLastIndex(arguments, x => x == "--profile");
|
||||||
|
|
||||||
|
if (profileIndex != -1)
|
||||||
{
|
{
|
||||||
if (arg.Length > 2 && arg[0] == '-' && arg[1] != '-' && !arg.Contains(' '))
|
profile = arguments[profileIndex + 1];
|
||||||
return arg[1..].Select(c => $"-{c}");
|
|
||||||
return new[] { arg };
|
|
||||||
}).ToArray();
|
|
||||||
|
|
||||||
arguments = args;
|
|
||||||
|
|
||||||
int profileIndex = Array.FindLastIndex(args, x => x == "--profile");
|
|
||||||
|
|
||||||
if (profileIndex != -1 && profileIndex < args.Length - 1)
|
|
||||||
{
|
|
||||||
profile = args[profileIndex + 1];
|
|
||||||
if (profile == "help")
|
if (profile == "help")
|
||||||
{
|
{
|
||||||
ListProfiles();
|
ListProfiles();
|
||||||
return false;
|
Environment.Exit(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (profiles.ContainsKey("default"))
|
|
||||||
{
|
|
||||||
ProcessArgs(profiles["default"].args);
|
|
||||||
appliedProfiles.Add("default");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (HasAutoProfiles)
|
|
||||||
{
|
|
||||||
ProcessArgs(args);
|
|
||||||
ApplyAutoProfiles();
|
|
||||||
}
|
|
||||||
|
|
||||||
ApplyProfiles(profile);
|
ApplyProfiles(profile);
|
||||||
|
|
||||||
ProcessArgs(args);
|
ProcessArgs(arguments);
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void SetConfigPath(string[] args)
|
void SetConfigPath(string[] args)
|
||||||
{
|
{
|
||||||
int idx = Array.LastIndexOf(args, "-c");
|
int idx = Array.FindLastIndex(args, x => x == "-c" || x == "--config");
|
||||||
int idx2 = Array.LastIndexOf(args, "--config");
|
|
||||||
idx = idx > idx2 ? idx : idx2;
|
|
||||||
if (idx != -1)
|
if (idx != -1)
|
||||||
{
|
{
|
||||||
confPath = Utils.ExpandUser(args[idx + 1]);
|
confPath = Utils.ExpandUser(args[idx + 1]);
|
||||||
}
|
|
||||||
|
|
||||||
if (confPath.Length > 0)
|
|
||||||
{
|
|
||||||
confPathChanged = true;
|
confPathChanged = true;
|
||||||
|
|
||||||
|
if(File.Exists(Path.Join(AppDomain.CurrentDomain.BaseDirectory, confPath)))
|
||||||
|
confPath = Path.Join(AppDomain.CurrentDomain.BaseDirectory, confPath);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!confPathChanged)
|
if (!confPathChanged)
|
||||||
|
@ -204,7 +198,6 @@ static class Config
|
||||||
Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.UserProfile), ".config", "sldl", "sldl.conf"),
|
Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.UserProfile), ".config", "sldl", "sldl.conf"),
|
||||||
Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData), "sldl", "sldl.conf"),
|
Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData), "sldl", "sldl.conf"),
|
||||||
Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "sldl.conf"),
|
Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "sldl.conf"),
|
||||||
Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "slsk-batchdl.conf"),
|
|
||||||
};
|
};
|
||||||
|
|
||||||
foreach (var path in configPaths)
|
foreach (var path in configPaths)
|
||||||
|
@ -219,7 +212,7 @@ static class Config
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public static void PostProcessArgs()
|
public void PostProcessArgs() // must be run after extracting tracklist
|
||||||
{
|
{
|
||||||
if (DoNotDownload || debugInfo)
|
if (DoNotDownload || debugInfo)
|
||||||
concurrentProcesses = 1;
|
concurrentProcesses = 1;
|
||||||
|
@ -230,6 +223,8 @@ static class Config
|
||||||
m3uOption = M3uOption.None;
|
m3uOption = M3uOption.None;
|
||||||
else if (!hasConfiguredM3uMode && inputType == InputType.String)
|
else if (!hasConfiguredM3uMode && inputType == InputType.String)
|
||||||
m3uOption = M3uOption.None;
|
m3uOption = M3uOption.None;
|
||||||
|
else if (!hasConfiguredM3uMode && !Program.trackLists.Flattened(true, true).Skip(1).Any())
|
||||||
|
m3uOption = M3uOption.None;
|
||||||
|
|
||||||
if (albumArtOnly && albumArtOption == AlbumArtOption.Default)
|
if (albumArtOnly && albumArtOption == AlbumArtOption.Default)
|
||||||
albumArtOption = AlbumArtOption.Largest;
|
albumArtOption = AlbumArtOption.Largest;
|
||||||
|
@ -246,7 +241,7 @@ static class Config
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void ParseConfig(string path)
|
void ParseConfig(string path)
|
||||||
{
|
{
|
||||||
var lines = File.ReadAllLines(path);
|
var lines = File.ReadAllLines(path);
|
||||||
var curProfile = "default";
|
var curProfile = "default";
|
||||||
|
@ -274,13 +269,13 @@ static class Config
|
||||||
if (val[0] == '"' && val[^1] == '"')
|
if (val[0] == '"' && val[^1] == '"')
|
||||||
val = val[1..^1];
|
val = val[1..^1];
|
||||||
|
|
||||||
if (!profiles.ContainsKey(curProfile))
|
if (!configProfiles.ContainsKey(curProfile))
|
||||||
profiles[curProfile] = (new List<string>(), null);
|
configProfiles[curProfile] = (new List<string>(), null);
|
||||||
|
|
||||||
if (key == "profile-cond" && curProfile != "default")
|
if (key == "profile-cond" && curProfile != "default")
|
||||||
{
|
{
|
||||||
var a = profiles[curProfile].args;
|
var a = configProfiles[curProfile].args;
|
||||||
profiles[curProfile] = (a, val);
|
configProfiles[curProfile] = (a, val);
|
||||||
HasAutoProfiles = true;
|
HasAutoProfiles = true;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@ -290,42 +285,82 @@ static class Config
|
||||||
else
|
else
|
||||||
key = "--" + key;
|
key = "--" + key;
|
||||||
|
|
||||||
profiles[curProfile].args.Add(key);
|
configProfiles[curProfile].args.Add(key);
|
||||||
profiles[curProfile].args.Add(val);
|
configProfiles[curProfile].args.Add(val);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public static void UpdateProfiles(TrackListEntry tle)
|
public static bool UpdateProfiles(TrackListEntry tle)
|
||||||
{
|
{
|
||||||
if (DoNotDownload)
|
if (I.DoNotDownload)
|
||||||
return;
|
return false;
|
||||||
if (!HasAutoProfiles)
|
if (!I.HasAutoProfiles)
|
||||||
return;
|
return false;
|
||||||
|
|
||||||
var newProfiles = ApplyAutoProfiles(tle);
|
bool needUpdate = false;
|
||||||
|
var toApply = new List<(string name, List<string> args)>();
|
||||||
|
|
||||||
if (newProfiles.Count > 0)
|
foreach ((var key, var val) in I.configProfiles)
|
||||||
{
|
{
|
||||||
//appliedProfiles.Clear();
|
if (key == "default" || val.cond == null)
|
||||||
appliedProfiles.UnionWith(newProfiles);
|
continue;
|
||||||
ApplyProfiles(profile);
|
|
||||||
ProcessArgs(arguments);
|
bool condSatisfied = I.ProfileConditionSatisfied(val.cond, tle);
|
||||||
PostProcessArgs();
|
bool alreadyApplied = I.appliedProfiles.Contains(key);
|
||||||
|
|
||||||
|
if (condSatisfied && !alreadyApplied)
|
||||||
|
needUpdate = true;
|
||||||
|
if (!condSatisfied && alreadyApplied)
|
||||||
|
needUpdate = true;
|
||||||
|
|
||||||
|
if (condSatisfied)
|
||||||
|
toApply.Add((key, val.args));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!needUpdate)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// this means that auto profiles can't change --profile and --config
|
||||||
|
var profile = I.profile;
|
||||||
|
Instance = new Config(I.configProfiles, I.arguments);
|
||||||
|
I.ApplyDefaultConfig();
|
||||||
|
I.ApplyProfiles(profile);
|
||||||
|
|
||||||
|
foreach (var (name, args) in toApply)
|
||||||
|
{
|
||||||
|
Console.WriteLine($"Applying auto profile: {name}");
|
||||||
|
I.ProcessArgs(args);
|
||||||
|
I.appliedProfiles.Add(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
I.ProcessArgs(I.arguments);
|
||||||
|
I.PostProcessArgs();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void ApplyDefaultConfig()
|
||||||
|
{
|
||||||
|
if (configProfiles.ContainsKey("default"))
|
||||||
|
{
|
||||||
|
ProcessArgs(configProfiles["default"].args);
|
||||||
|
appliedProfiles.Add("default");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void ApplyProfiles(string names)
|
void ApplyProfiles(string names)
|
||||||
{
|
{
|
||||||
foreach (var name in names.Split(','))
|
foreach (var name in names.Split(','))
|
||||||
{
|
{
|
||||||
if (name.Length > 0 && name != "default")
|
if (name.Length > 0 && name != "default")
|
||||||
{
|
{
|
||||||
if (profiles.ContainsKey(name))
|
if (configProfiles.ContainsKey(name))
|
||||||
{
|
{
|
||||||
ProcessArgs(profiles[name].args);
|
ProcessArgs(configProfiles[name].args);
|
||||||
appliedProfiles.Add(name);
|
appliedProfiles.Add(name);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@ -335,31 +370,7 @@ static class Config
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static HashSet<string> ApplyAutoProfiles(TrackListEntry? tle = null)
|
object GetVarValue(string var, TrackListEntry? tle = null)
|
||||||
{
|
|
||||||
var applied = new HashSet<string>();
|
|
||||||
|
|
||||||
if (!HasAutoProfiles)
|
|
||||||
return applied;
|
|
||||||
|
|
||||||
foreach ((var key, var val) in profiles)
|
|
||||||
{
|
|
||||||
if (key == "default" || appliedProfiles.Contains(key))
|
|
||||||
continue;
|
|
||||||
if (val.cond != null && ProfileConditionSatisfied(val.cond, tle))
|
|
||||||
{
|
|
||||||
Console.WriteLine($"Applying auto profile: {key}");
|
|
||||||
ProcessArgs(val.args);
|
|
||||||
appliedProfiles.Add(key);
|
|
||||||
applied.Add(key);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return applied;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static object GetVarValue(string var, TrackListEntry? tle = null)
|
|
||||||
{
|
{
|
||||||
static string toKebab(string input)
|
static string toKebab(string input)
|
||||||
{
|
{
|
||||||
|
@ -380,7 +391,7 @@ static class Config
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public static bool ProfileConditionSatisfied(string cond, TrackListEntry? tle = null)
|
public bool ProfileConditionSatisfied(string cond, TrackListEntry? tle = null)
|
||||||
{
|
{
|
||||||
var tokens = new Queue<string>(Regex.Split(cond, @"(\s+|\(|\)|&&|\|\||==|!=|!|\"".*?\"")").Where(t => !string.IsNullOrWhiteSpace(t)));
|
var tokens = new Queue<string>(Regex.Split(cond, @"(\s+|\(|\)|&&|\|\||==|!=|!|\"".*?\"")").Where(t => !string.IsNullOrWhiteSpace(t)));
|
||||||
|
|
||||||
|
@ -456,10 +467,10 @@ static class Config
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void ListProfiles()
|
void ListProfiles()
|
||||||
{
|
{
|
||||||
Console.WriteLine("Available profiles:");
|
Console.WriteLine("Available profiles:");
|
||||||
foreach ((var key, var val) in profiles)
|
foreach ((var key, var val) in configProfiles)
|
||||||
{
|
{
|
||||||
if (key == "default")
|
if (key == "default")
|
||||||
continue;
|
continue;
|
||||||
|
@ -475,30 +486,25 @@ static class Config
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public static void AddTemporaryConditions(FileConditionsPatch? cond, FileConditionsPatch? prefCond)
|
public void AddTemporaryConditions(FileConditionsMod? cond, FileConditionsMod? prefCond)
|
||||||
{
|
{
|
||||||
if (cond != null)
|
if (cond != null)
|
||||||
{
|
undoTempConds = necessaryCond.ApplyMod(cond);
|
||||||
prevConds = necessaryCond;
|
|
||||||
necessaryCond = necessaryCond.With(cond);
|
|
||||||
}
|
|
||||||
if (prefCond != null)
|
if (prefCond != null)
|
||||||
{
|
undoTempPrefConds = preferredCond.ApplyMod(prefCond);
|
||||||
prevPrefConds = preferredCond;
|
|
||||||
preferredCond = preferredCond.With(prefCond);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void RestoreConditions()
|
|
||||||
|
public void RestoreConditions()
|
||||||
{
|
{
|
||||||
if (prevConds != null)
|
if (undoTempConds != null)
|
||||||
necessaryCond = prevConds;
|
necessaryCond.ApplyMod(undoTempConds);
|
||||||
if (prevPrefConds != null)
|
if (undoTempPrefConds != null)
|
||||||
preferredCond = prevPrefConds;
|
preferredCond.ApplyMod(undoTempPrefConds);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public static FileConditionsPatch ParseConditions(string input)
|
public static FileConditionsMod ParseConditions(string input)
|
||||||
{
|
{
|
||||||
static void UpdateMinMax(string value, string condition, ref int? min, ref int? max)
|
static void UpdateMinMax(string value, string condition, ref int? min, ref int? max)
|
||||||
{
|
{
|
||||||
|
@ -514,7 +520,7 @@ static class Config
|
||||||
min = max = int.Parse(value);
|
min = max = int.Parse(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
var cond = new FileConditionsPatch();
|
var cond = new FileConditionsMod();
|
||||||
|
|
||||||
var tr = StringSplitOptions.TrimEntries | StringSplitOptions.RemoveEmptyEntries;
|
var tr = StringSplitOptions.TrimEntries | StringSplitOptions.RemoveEmptyEntries;
|
||||||
string[] conditions = input.Split(';', tr);
|
string[] conditions = input.Split(';', tr);
|
||||||
|
@ -583,7 +589,7 @@ static class Config
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void ProcessArgs(IReadOnlyList<string> args)
|
void ProcessArgs(IReadOnlyList<string> args)
|
||||||
{
|
{
|
||||||
void setFlag(ref bool flag, ref int i, bool trueVal = true)
|
void setFlag(ref bool flag, ref int i, bool trueVal = true)
|
||||||
{
|
{
|
||||||
|
@ -1068,15 +1074,14 @@ static class Config
|
||||||
case "--accept-no-length":
|
case "--accept-no-length":
|
||||||
setFlag(ref necessaryCond.AcceptNoLength, ref i);
|
setFlag(ref necessaryCond.AcceptNoLength, ref i);
|
||||||
break;
|
break;
|
||||||
case "--c":
|
|
||||||
case "--cond":
|
case "--cond":
|
||||||
case "--conditions":
|
case "--conditions":
|
||||||
necessaryCond.With(ParseConditions(args[++i]));
|
necessaryCond.ApplyMod(ParseConditions(args[++i]));
|
||||||
break;
|
break;
|
||||||
case "--pc":
|
case "--pc":
|
||||||
case "--pref":
|
case "--pref":
|
||||||
case "--preferred-conditions":
|
case "--preferred-conditions":
|
||||||
preferredCond.With(ParseConditions(args[++i]));
|
preferredCond.ApplyMod(ParseConditions(args[++i]));
|
||||||
break;
|
break;
|
||||||
case "--nmsc":
|
case "--nmsc":
|
||||||
case "--no-modify-share-count":
|
case "--no-modify-share-count":
|
||||||
|
|
|
@ -107,8 +107,8 @@ namespace Data
|
||||||
public bool needSkipExistingAfterSearch = false;
|
public bool needSkipExistingAfterSearch = false;
|
||||||
public bool gotoNextAfterSearch = false;
|
public bool gotoNextAfterSearch = false;
|
||||||
public string? defaultFolderName = null;
|
public string? defaultFolderName = null;
|
||||||
public FileConditionsPatch? additionalConds = null;
|
public FileConditionsMod? additionalConds = null;
|
||||||
public FileConditionsPatch? additionalPrefConds = null;
|
public FileConditionsMod? additionalPrefConds = null;
|
||||||
|
|
||||||
public TrackListEntry(TrackType trackType)
|
public TrackListEntry(TrackType trackType)
|
||||||
{
|
{
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
using Soulseek;
|
using Soulseek;
|
||||||
using System.Collections.Concurrent;
|
using System.Collections.Concurrent;
|
||||||
using System.Text.RegularExpressions;
|
using System.Text.RegularExpressions;
|
||||||
|
using System.Diagnostics;
|
||||||
|
|
||||||
using Data;
|
using Data;
|
||||||
using Enums;
|
using Enums;
|
||||||
|
@ -17,9 +18,9 @@ using SlDictionary = System.Collections.Concurrent.ConcurrentDictionary<string,
|
||||||
|
|
||||||
static class Download
|
static class Download
|
||||||
{
|
{
|
||||||
public static async Task DownloadFile(SearchResponse response, Soulseek.File file, string filePath, Track track, ProgressBar progress, CancellationTokenSource cts, CancellationTokenSource? searchCts = null)
|
public static async Task DownloadFile(SearchResponse response, Soulseek.File file, string filePath, Track track, ProgressBar progress, CancellationToken? ct = null, CancellationTokenSource? searchCts = null)
|
||||||
{
|
{
|
||||||
if (Config.DoNotDownload)
|
if (Config.I.DoNotDownload)
|
||||||
throw new Exception();
|
throw new Exception();
|
||||||
|
|
||||||
await Program.WaitForLogin();
|
await Program.WaitForLogin();
|
||||||
|
@ -42,8 +43,12 @@ static class Download
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
using var downloadCts = ct != null ?
|
||||||
|
CancellationTokenSource.CreateLinkedTokenSource((CancellationToken)ct) :
|
||||||
|
new CancellationTokenSource();
|
||||||
|
|
||||||
using var outputStream = new FileStream(filePath, FileMode.Create);
|
using var outputStream = new FileStream(filePath, FileMode.Create);
|
||||||
var wrapper = new DownloadWrapper(origPath, response, file, track, cts, progress);
|
var wrapper = new DownloadWrapper(origPath, response, file, track, downloadCts, progress);
|
||||||
downloads.TryAdd(file.Filename, wrapper);
|
downloads.TryAdd(file.Filename, wrapper);
|
||||||
|
|
||||||
int maxRetries = 3;
|
int maxRetries = 3;
|
||||||
|
@ -55,7 +60,7 @@ static class Download
|
||||||
await client.DownloadAsync(response.Username, file.Filename,
|
await client.DownloadAsync(response.Username, file.Filename,
|
||||||
() => Task.FromResult((Stream)outputStream),
|
() => Task.FromResult((Stream)outputStream),
|
||||||
file.Size, startOffset: outputStream.Position,
|
file.Size, startOffset: outputStream.Position,
|
||||||
options: transferOptions, cancellationToken: cts.Token);
|
options: transferOptions, cancellationToken: downloadCts.Token);
|
||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -156,7 +161,12 @@ public class DownloadWrapper
|
||||||
else if (transfer != null)
|
else if (transfer != null)
|
||||||
{
|
{
|
||||||
if (queued)
|
if (queued)
|
||||||
state = "Queued";
|
{
|
||||||
|
if ((transfer.State & TransferStates.Remotely) != 0)
|
||||||
|
state = "Queued (R)";
|
||||||
|
else
|
||||||
|
state = "Queued (L)";
|
||||||
|
}
|
||||||
else if ((transfer.State & TransferStates.Initializing) != 0)
|
else if ((transfer.State & TransferStates.Initializing) != 0)
|
||||||
state = "Initialize";
|
state = "Initialize";
|
||||||
else if ((transfer.State & TransferStates.Completed) != 0)
|
else if ((transfer.State & TransferStates.Completed) != 0)
|
||||||
|
|
|
@ -77,15 +77,15 @@ namespace Extractors
|
||||||
var track = new Track() { Artist = artist, Album = name, Type = TrackType.Album };
|
var track = new Track() { Artist = artist, Album = name, Type = TrackType.Album };
|
||||||
trackLists.AddEntry(new TrackListEntry(track));
|
trackLists.AddEntry(new TrackListEntry(track));
|
||||||
|
|
||||||
if (Config.setAlbumMinTrackCount || Config.setAlbumMaxTrackCount)
|
if (Config.I.setAlbumMinTrackCount || Config.I.setAlbumMaxTrackCount)
|
||||||
{
|
{
|
||||||
var trackTable = doc.DocumentNode.SelectSingleNode("//*[@id='track_table']");
|
var trackTable = doc.DocumentNode.SelectSingleNode("//*[@id='track_table']");
|
||||||
int n = trackTable.SelectNodes(".//tr").Count;
|
int n = trackTable.SelectNodes(".//tr").Count;
|
||||||
|
|
||||||
if (Config.setAlbumMinTrackCount)
|
if (Config.I.setAlbumMinTrackCount)
|
||||||
track.MinAlbumTrackCount = n;
|
track.MinAlbumTrackCount = n;
|
||||||
|
|
||||||
if (Config.setAlbumMaxTrackCount)
|
if (Config.I.setAlbumMaxTrackCount)
|
||||||
track.MaxAlbumTrackCount = n;
|
track.MaxAlbumTrackCount = n;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,12 +19,12 @@ namespace Extractors
|
||||||
public async Task<TrackLists> GetTracks(string input, int maxTracks, int offset, bool reverse)
|
public async Task<TrackLists> GetTracks(string input, int maxTracks, int offset, bool reverse)
|
||||||
{
|
{
|
||||||
if (!File.Exists(input))
|
if (!File.Exists(input))
|
||||||
throw new FileNotFoundException("CSV file not found");
|
throw new FileNotFoundException($"CSV file '{input}' not found");
|
||||||
|
|
||||||
csvFilePath = input;
|
csvFilePath = input;
|
||||||
|
|
||||||
var tracks = await ParseCsvIntoTrackInfo(input, Config.artistCol, Config.trackCol, Config.lengthCol,
|
var tracks = await ParseCsvIntoTrackInfo(input, Config.I.artistCol, Config.I.trackCol, Config.I.lengthCol,
|
||||||
Config.albumCol, Config.descCol, Config.ytIdCol, Config.trackCountCol, Config.timeUnit, Config.ytParse);
|
Config.I.albumCol, Config.I.descCol, Config.I.ytIdCol, Config.I.trackCountCol, Config.I.timeUnit, Config.I.ytParse);
|
||||||
|
|
||||||
if (reverse)
|
if (reverse)
|
||||||
tracks.Reverse();
|
tracks.Reverse();
|
||||||
|
|
|
@ -22,7 +22,7 @@ namespace Extractors
|
||||||
public async Task<TrackLists> GetTracks(string input, int maxTracks, int offset, bool reverse)
|
public async Task<TrackLists> GetTracks(string input, int maxTracks, int offset, bool reverse)
|
||||||
{
|
{
|
||||||
if (!File.Exists(input))
|
if (!File.Exists(input))
|
||||||
throw new FileNotFoundException("List file not found");
|
throw new FileNotFoundException($"List file '{input}' not found");
|
||||||
|
|
||||||
listFilePath = input;
|
listFilePath = input;
|
||||||
|
|
||||||
|
@ -49,12 +49,22 @@ namespace Extractors
|
||||||
if (added >= maxTracks)
|
if (added >= maxTracks)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
bool savedVal = Config.I.album;
|
||||||
|
|
||||||
|
if (line.StartsWith("a:"))
|
||||||
|
{
|
||||||
|
line = line[2..];
|
||||||
|
Config.I.album = true;
|
||||||
|
}
|
||||||
|
|
||||||
var fields = ParseLine(line);
|
var fields = ParseLine(line);
|
||||||
|
|
||||||
var (_, ex) = ExtractorRegistry.GetMatchingExtractor(fields[0]);
|
var (_, ex) = ExtractorRegistry.GetMatchingExtractor(fields[0]);
|
||||||
|
|
||||||
var tl = await ex.GetTracks(fields[0], int.MaxValue, 0, false);
|
var tl = await ex.GetTracks(fields[0], int.MaxValue, 0, false);
|
||||||
|
|
||||||
|
Config.I.album = savedVal;
|
||||||
|
|
||||||
foreach (var tle in tl.lists)
|
foreach (var tle in tl.lists)
|
||||||
{
|
{
|
||||||
if (fields.Count >= 2)
|
if (fields.Count >= 2)
|
||||||
|
|
|
@ -25,22 +25,24 @@ namespace Extractors
|
||||||
int max = reverse ? int.MaxValue : maxTracks;
|
int max = reverse ? int.MaxValue : maxTracks;
|
||||||
int off = reverse ? 0 : offset;
|
int off = reverse ? 0 : offset;
|
||||||
|
|
||||||
bool needLogin = input == "spotify-likes" || Config.removeTracksFromSource;
|
bool needLogin = input == "spotify-likes" || Config.I.removeTracksFromSource;
|
||||||
var tle = new TrackListEntry(TrackType.Normal);
|
|
||||||
|
|
||||||
if (needLogin && Config.spotifyToken.Length == 0 && (Config.spotifyId.Length == 0 || Config.spotifySecret.Length == 0))
|
if (needLogin && Config.I.spotifyToken.Length == 0 && (Config.I.spotifyId.Length == 0 || Config.I.spotifySecret.Length == 0))
|
||||||
{
|
{
|
||||||
Console.WriteLine("Error: Credentials are required when downloading liked music or removing from source playlists.");
|
Console.WriteLine("Error: Credentials are required when downloading liked music or removing from source playlists.");
|
||||||
Environment.Exit(1);
|
Environment.Exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
spotifyClient = new Spotify(Config.spotifyId, Config.spotifySecret, Config.spotifyToken, Config.spotifyRefresh);
|
spotifyClient = new Spotify(Config.I.spotifyId, Config.I.spotifySecret, Config.I.spotifyToken, Config.I.spotifyRefresh);
|
||||||
await spotifyClient.Authorize(needLogin, Config.removeTracksFromSource);
|
await spotifyClient.Authorize(needLogin, Config.I.removeTracksFromSource);
|
||||||
|
|
||||||
|
TrackListEntry? tle = null;
|
||||||
|
|
||||||
if (input == "spotify-likes")
|
if (input == "spotify-likes")
|
||||||
{
|
{
|
||||||
Console.WriteLine("Loading Spotify likes..");
|
Console.WriteLine("Loading Spotify likes..");
|
||||||
var tracks = await spotifyClient.GetLikes(max, off);
|
var tracks = await spotifyClient.GetLikes(max, off);
|
||||||
|
tle = new TrackListEntry(TrackType.Normal);
|
||||||
tle.defaultFolderName = "Spotify Likes";
|
tle.defaultFolderName = "Spotify Likes";
|
||||||
tle.list.Add(tracks);
|
tle.list.Add(tracks);
|
||||||
}
|
}
|
||||||
|
@ -48,12 +50,13 @@ namespace Extractors
|
||||||
{
|
{
|
||||||
Console.WriteLine("Loading Spotify album..");
|
Console.WriteLine("Loading Spotify album..");
|
||||||
(var source, var tracks) = await spotifyClient.GetAlbum(input);
|
(var source, var tracks) = await spotifyClient.GetAlbum(input);
|
||||||
|
tle = new TrackListEntry(TrackType.Album);
|
||||||
tle.source = source;
|
tle.source = source;
|
||||||
|
|
||||||
if (Config.setAlbumMinTrackCount)
|
if (Config.I.setAlbumMinTrackCount)
|
||||||
source.MinAlbumTrackCount = tracks.Count;
|
source.MinAlbumTrackCount = tracks.Count;
|
||||||
|
|
||||||
if (Config.setAlbumMaxTrackCount)
|
if (Config.I.setAlbumMaxTrackCount)
|
||||||
source.MaxAlbumTrackCount = tracks.Count;
|
source.MaxAlbumTrackCount = tracks.Count;
|
||||||
}
|
}
|
||||||
else if (input.Contains("/artist/"))
|
else if (input.Contains("/artist/"))
|
||||||
|
@ -65,6 +68,7 @@ namespace Extractors
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
var tracks = new List<Track>();
|
var tracks = new List<Track>();
|
||||||
|
tle = new TrackListEntry(TrackType.Normal);
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
@ -76,7 +80,7 @@ namespace Extractors
|
||||||
{
|
{
|
||||||
if (!needLogin && !spotifyClient.UsedDefaultCredentials)
|
if (!needLogin && !spotifyClient.UsedDefaultCredentials)
|
||||||
{
|
{
|
||||||
await spotifyClient.Authorize(true, Config.removeTracksFromSource);
|
await spotifyClient.Authorize(true, Config.I.removeTracksFromSource);
|
||||||
(var playlistName, playlistUri, tracks) = await spotifyClient.GetPlaylist(input, max, off);
|
(var playlistName, playlistUri, tracks) = await spotifyClient.GetPlaylist(input, max, off);
|
||||||
tle.defaultFolderName = playlistName;
|
tle.defaultFolderName = playlistName;
|
||||||
}
|
}
|
||||||
|
@ -219,7 +223,7 @@ namespace Extractors
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
Console.WriteLine($"Could not make an API call with existing token: {ex}");
|
Console.WriteLine($"Could not make an API call with existing token: {ex.Message}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (_clientRefreshToken.Length != 0)
|
if (_clientRefreshToken.Length != 0)
|
||||||
|
|
|
@ -14,10 +14,10 @@ namespace Extractors
|
||||||
public async Task<TrackLists> GetTracks(string input, int maxTracks, int offset, bool reverse)
|
public async Task<TrackLists> GetTracks(string input, int maxTracks, int offset, bool reverse)
|
||||||
{
|
{
|
||||||
var trackLists = new TrackLists();
|
var trackLists = new TrackLists();
|
||||||
var music = ParseTrackArg(input, Config.album);
|
var music = ParseTrackArg(input, Config.I.album);
|
||||||
TrackListEntry tle;
|
TrackListEntry tle;
|
||||||
|
|
||||||
if (Config.album || (music.Title.Length == 0 && music.Album.Length > 0))
|
if (Config.I.album || (music.Title.Length == 0 && music.Album.Length > 0))
|
||||||
{
|
{
|
||||||
music.Type = TrackType.Album;
|
music.Type = TrackType.Album;
|
||||||
tle = new TrackListEntry(music);
|
tle = new TrackListEntry(music);
|
||||||
|
|
|
@ -26,19 +26,19 @@ namespace Extractors
|
||||||
var trackLists = new TrackLists();
|
var trackLists = new TrackLists();
|
||||||
int max = reverse ? int.MaxValue : maxTracks;
|
int max = reverse ? int.MaxValue : maxTracks;
|
||||||
int off = reverse ? 0 : offset;
|
int off = reverse ? 0 : offset;
|
||||||
YouTube.apiKey = Config.ytKey;
|
YouTube.apiKey = Config.I.ytKey;
|
||||||
|
|
||||||
string name;
|
string name;
|
||||||
List<Track>? deleted = null;
|
List<Track>? deleted = null;
|
||||||
List<Track> tracks = new();
|
List<Track> tracks = new();
|
||||||
|
|
||||||
if (Config.getDeleted)
|
if (Config.I.getDeleted)
|
||||||
{
|
{
|
||||||
Console.WriteLine("Getting deleted videos..");
|
Console.WriteLine("Getting deleted videos..");
|
||||||
var archive = new YouTube.YouTubeArchiveRetriever();
|
var archive = new YouTube.YouTubeArchiveRetriever();
|
||||||
deleted = await archive.RetrieveDeleted(input, printFailed: Config.deletedOnly);
|
deleted = await archive.RetrieveDeleted(input, printFailed: Config.I.deletedOnly);
|
||||||
}
|
}
|
||||||
if (!Config.deletedOnly)
|
if (!Config.I.deletedOnly)
|
||||||
{
|
{
|
||||||
if (YouTube.apiKey.Length > 0)
|
if (YouTube.apiKey.Length > 0)
|
||||||
{
|
{
|
||||||
|
|
|
@ -1,5 +1,4 @@
|
||||||
using System.Text.RegularExpressions;
|
|
||||||
|
|
||||||
using Data;
|
using Data;
|
||||||
|
|
||||||
using SearchResponse = Soulseek.SearchResponse;
|
using SearchResponse = Soulseek.SearchResponse;
|
||||||
|
@ -41,58 +40,90 @@ public class FileConditions
|
||||||
BannedUsers = other.BannedUsers.ToArray();
|
BannedUsers = other.BannedUsers.ToArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
public FileConditions With(FileConditionsPatch patch)
|
public FileConditionsMod ApplyMod(FileConditionsMod mod)
|
||||||
{
|
{
|
||||||
var cond = new FileConditions(this);
|
var undoMod = new FileConditionsMod();
|
||||||
|
|
||||||
if (patch.LengthTolerance != null)
|
if (mod.LengthTolerance != null)
|
||||||
cond.LengthTolerance = patch.LengthTolerance.Value;
|
{
|
||||||
|
undoMod.LengthTolerance = LengthTolerance;
|
||||||
|
LengthTolerance = mod.LengthTolerance.Value;
|
||||||
|
}
|
||||||
|
if (mod.MinBitrate != null)
|
||||||
|
{
|
||||||
|
undoMod.MinBitrate = MinBitrate;
|
||||||
|
MinBitrate = mod.MinBitrate.Value;
|
||||||
|
}
|
||||||
|
if (mod.MaxBitrate != null)
|
||||||
|
{
|
||||||
|
undoMod.MaxBitrate = MaxBitrate;
|
||||||
|
MaxBitrate = mod.MaxBitrate.Value;
|
||||||
|
}
|
||||||
|
if (mod.MinSampleRate != null)
|
||||||
|
{
|
||||||
|
undoMod.MinSampleRate = MinSampleRate;
|
||||||
|
MinSampleRate = mod.MinSampleRate.Value;
|
||||||
|
}
|
||||||
|
if (mod.MaxSampleRate != null)
|
||||||
|
{
|
||||||
|
undoMod.MaxSampleRate = MaxSampleRate;
|
||||||
|
MaxSampleRate = mod.MaxSampleRate.Value;
|
||||||
|
}
|
||||||
|
if (mod.MinBitDepth != null)
|
||||||
|
{
|
||||||
|
undoMod.MinBitDepth = MinBitDepth;
|
||||||
|
MinBitDepth = mod.MinBitDepth.Value;
|
||||||
|
}
|
||||||
|
if (mod.MaxBitDepth != null)
|
||||||
|
{
|
||||||
|
undoMod.MaxBitDepth = MaxBitDepth;
|
||||||
|
MaxBitDepth = mod.MaxBitDepth.Value;
|
||||||
|
}
|
||||||
|
if (mod.StrictTitle != null)
|
||||||
|
{
|
||||||
|
undoMod.StrictTitle = StrictTitle;
|
||||||
|
StrictTitle = mod.StrictTitle.Value;
|
||||||
|
}
|
||||||
|
if (mod.StrictArtist != null)
|
||||||
|
{
|
||||||
|
undoMod.StrictArtist = StrictArtist;
|
||||||
|
StrictArtist = mod.StrictArtist.Value;
|
||||||
|
}
|
||||||
|
if (mod.StrictAlbum != null)
|
||||||
|
{
|
||||||
|
undoMod.StrictAlbum = StrictAlbum;
|
||||||
|
StrictAlbum = mod.StrictAlbum.Value;
|
||||||
|
}
|
||||||
|
if (mod.Formats != null)
|
||||||
|
{
|
||||||
|
undoMod.Formats = Formats;
|
||||||
|
Formats = mod.Formats;
|
||||||
|
}
|
||||||
|
if (mod.BannedUsers != null)
|
||||||
|
{
|
||||||
|
undoMod.BannedUsers = BannedUsers;
|
||||||
|
BannedUsers = mod.BannedUsers;
|
||||||
|
}
|
||||||
|
if (mod.StrictStringDiacrRemove != null)
|
||||||
|
{
|
||||||
|
undoMod.StrictStringDiacrRemove = StrictStringDiacrRemove;
|
||||||
|
StrictStringDiacrRemove = mod.StrictStringDiacrRemove.Value;
|
||||||
|
}
|
||||||
|
if (mod.AcceptNoLength != null)
|
||||||
|
{
|
||||||
|
undoMod.AcceptNoLength = AcceptNoLength;
|
||||||
|
AcceptNoLength = mod.AcceptNoLength.Value;
|
||||||
|
}
|
||||||
|
if (mod.AcceptMissingProps != null)
|
||||||
|
{
|
||||||
|
undoMod.AcceptMissingProps = AcceptMissingProps;
|
||||||
|
AcceptMissingProps = mod.AcceptMissingProps.Value;
|
||||||
|
}
|
||||||
|
|
||||||
if (patch.MinBitrate != null)
|
return undoMod;
|
||||||
cond.MinBitrate = patch.MinBitrate.Value;
|
|
||||||
|
|
||||||
if (patch.MaxBitrate != null)
|
|
||||||
cond.MaxBitrate = patch.MaxBitrate.Value;
|
|
||||||
|
|
||||||
if (patch.MinSampleRate != null)
|
|
||||||
cond.MinSampleRate = patch.MinSampleRate.Value;
|
|
||||||
|
|
||||||
if (patch.MaxSampleRate != null)
|
|
||||||
cond.MaxSampleRate = patch.MaxSampleRate.Value;
|
|
||||||
|
|
||||||
if (patch.MinBitDepth != null)
|
|
||||||
cond.MinBitDepth = patch.MinBitDepth.Value;
|
|
||||||
|
|
||||||
if (patch.MaxBitDepth != null)
|
|
||||||
cond.MaxBitDepth = patch.MaxBitDepth.Value;
|
|
||||||
|
|
||||||
if (patch.StrictTitle != null)
|
|
||||||
cond.StrictTitle = patch.StrictTitle.Value;
|
|
||||||
|
|
||||||
if (patch.StrictArtist != null)
|
|
||||||
cond.StrictArtist = patch.StrictArtist.Value;
|
|
||||||
|
|
||||||
if (patch.StrictAlbum != null)
|
|
||||||
cond.StrictAlbum = patch.StrictAlbum.Value;
|
|
||||||
|
|
||||||
if (patch.Formats != null)
|
|
||||||
cond.Formats = patch.Formats;
|
|
||||||
|
|
||||||
if (patch.BannedUsers != null)
|
|
||||||
cond.BannedUsers = patch.BannedUsers;
|
|
||||||
|
|
||||||
if (patch.StrictStringDiacrRemove != null)
|
|
||||||
cond.StrictStringDiacrRemove = patch.StrictStringDiacrRemove.Value;
|
|
||||||
|
|
||||||
if (patch.AcceptNoLength != null)
|
|
||||||
cond.AcceptNoLength = patch.AcceptNoLength.Value;
|
|
||||||
|
|
||||||
if (patch.AcceptMissingProps != null)
|
|
||||||
cond.AcceptMissingProps = patch.AcceptMissingProps.Value;
|
|
||||||
|
|
||||||
return cond;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public override bool Equals(object obj)
|
public override bool Equals(object obj)
|
||||||
{
|
{
|
||||||
if (obj is FileConditions other)
|
if (obj is FileConditions other)
|
||||||
|
@ -298,7 +329,7 @@ public class FileConditions
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public class FileConditionsPatch
|
public class FileConditionsMod
|
||||||
{
|
{
|
||||||
public int? LengthTolerance = null;
|
public int? LengthTolerance = null;
|
||||||
public int? MinBitrate = null;
|
public int? MinBitrate = null;
|
||||||
|
|
|
@ -7,14 +7,13 @@ using System.Text.RegularExpressions;
|
||||||
|
|
||||||
using Data;
|
using Data;
|
||||||
using Enums;
|
using Enums;
|
||||||
using System.ComponentModel;
|
|
||||||
|
|
||||||
|
|
||||||
public class FileManager
|
public class FileManager
|
||||||
{
|
{
|
||||||
readonly TrackListEntry tle;
|
readonly TrackListEntry tle;
|
||||||
readonly HashSet<Track> organized = new();
|
readonly HashSet<Track> organized = new();
|
||||||
string? remoteCommonDir;
|
public string? remoteCommonDir { get; private set; }
|
||||||
|
|
||||||
public FileManager(TrackListEntry tle)
|
public FileManager(TrackListEntry tle)
|
||||||
{
|
{
|
||||||
|
@ -28,19 +27,16 @@ public class FileManager
|
||||||
|
|
||||||
public string GetSavePathNoExt(string sourceFname)
|
public string GetSavePathNoExt(string sourceFname)
|
||||||
{
|
{
|
||||||
string parent = Config.parentDir;
|
string parent = Config.I.parentDir;
|
||||||
string name = Utils.GetFileNameWithoutExtSlsk(sourceFname);
|
string name = Utils.GetFileNameWithoutExtSlsk(sourceFname);
|
||||||
|
|
||||||
if (tle.defaultFolderName != null)
|
if (tle.defaultFolderName != null)
|
||||||
{
|
{
|
||||||
parent = Path.Join(parent, tle.defaultFolderName.ReplaceInvalidChars(Config.invalidReplaceStr, removeSlash: false));
|
parent = Path.Join(parent, tle.defaultFolderName.ReplaceInvalidChars(Config.I.invalidReplaceStr, removeSlash: false));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (tle.source.Type == TrackType.Album)
|
if (tle.source.Type == TrackType.Album && !string.IsNullOrEmpty(remoteCommonDir))
|
||||||
{
|
{
|
||||||
if (remoteCommonDir == null)
|
|
||||||
throw new NullReferenceException("Remote common dir needs to be configured to organize album files");
|
|
||||||
|
|
||||||
string dirname = Path.GetFileName(remoteCommonDir);
|
string dirname = Path.GetFileName(remoteCommonDir);
|
||||||
string relpath = Path.GetRelativePath(remoteCommonDir, Utils.NormalizedPath(sourceFname));
|
string relpath = Path.GetRelativePath(remoteCommonDir, Utils.NormalizedPath(sourceFname));
|
||||||
parent = Path.Join(parent, dirname, Path.GetDirectoryName(relpath));
|
parent = Path.Join(parent, dirname, Path.GetDirectoryName(relpath));
|
||||||
|
@ -64,7 +60,7 @@ public class FileManager
|
||||||
OrganizeAudio(track, track.FirstDownload);
|
OrganizeAudio(track, track.FirstDownload);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool onlyAdditionalImages = Config.nameFormat.Length == 0;
|
bool onlyAdditionalImages = Config.I.nameFormat.Length == 0;
|
||||||
|
|
||||||
var nonAudioToOrganize = onlyAdditionalImages ? additionalImages : tracks.Where(t => t.IsNotAudio);
|
var nonAudioToOrganize = onlyAdditionalImages ? additionalImages : tracks.Where(t => t.IsNotAudio);
|
||||||
|
|
||||||
|
@ -88,14 +84,14 @@ public class FileManager
|
||||||
if (track.DownloadPath.Length == 0 || !Utils.IsMusicFile(track.DownloadPath))
|
if (track.DownloadPath.Length == 0 || !Utils.IsMusicFile(track.DownloadPath))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (Config.nameFormat.Length == 0)
|
if (Config.I.nameFormat.Length == 0)
|
||||||
{
|
{
|
||||||
organized.Add(track);
|
organized.Add(track);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
string pathPart = SubstituteValues(Config.nameFormat, track, file);
|
string pathPart = SubstituteValues(Config.I.nameFormat, track, file);
|
||||||
string newFilePath = Path.Join(Config.parentDir, pathPart + Path.GetExtension(track.DownloadPath));
|
string newFilePath = Path.Join(Config.I.parentDir, pathPart + Path.GetExtension(track.DownloadPath));
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
@ -140,7 +136,7 @@ public class FileManager
|
||||||
{
|
{
|
||||||
Directory.CreateDirectory(Path.GetDirectoryName(newPath));
|
Directory.CreateDirectory(Path.GetDirectoryName(newPath));
|
||||||
Utils.Move(oldPath, newPath);
|
Utils.Move(oldPath, newPath);
|
||||||
Utils.DeleteAncestorsIfEmpty(Path.GetDirectoryName(oldPath), Config.parentDir);
|
Utils.DeleteAncestorsIfEmpty(Path.GetDirectoryName(oldPath), Config.I.parentDir);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -184,7 +180,7 @@ public class FileManager
|
||||||
chosenOpt = Regex.Replace(chosenOpt, @"\([^()]*\)|[^()]+", match =>
|
chosenOpt = Regex.Replace(chosenOpt, @"\([^()]*\)|[^()]+", match =>
|
||||||
{
|
{
|
||||||
if (match.Value.StartsWith("(") && match.Value.EndsWith(")"))
|
if (match.Value.StartsWith("(") && match.Value.EndsWith(")"))
|
||||||
return match.Value[1..^1].ReplaceInvalidChars(Config.invalidReplaceStr, removeSlash: false);
|
return match.Value[1..^1].ReplaceInvalidChars(Config.I.invalidReplaceStr, removeSlash: false);
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
TryGetVarValue(match.Value, file, slfile, track, out string res);
|
TryGetVarValue(match.Value, file, slfile, track, out string res);
|
||||||
|
@ -205,7 +201,7 @@ public class FileManager
|
||||||
char dirsep = Path.DirectorySeparatorChar;
|
char dirsep = Path.DirectorySeparatorChar;
|
||||||
newName = newName.Replace('/', dirsep).Replace('\\', dirsep);
|
newName = newName.Replace('/', dirsep).Replace('\\', dirsep);
|
||||||
var x = newName.Split(dirsep, StringSplitOptions.RemoveEmptyEntries);
|
var x = newName.Split(dirsep, StringSplitOptions.RemoveEmptyEntries);
|
||||||
newName = string.Join(dirsep, x.Select(x => x.ReplaceInvalidChars(Config.invalidReplaceStr).Trim(' ', '.')));
|
newName = string.Join(dirsep, x.Select(x => x.ReplaceInvalidChars(Config.I.invalidReplaceStr).Trim(' ', '.')));
|
||||||
return newName;
|
return newName;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -257,12 +253,14 @@ public class FileManager
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
case "extractor":
|
case "extractor":
|
||||||
res = Config.inputType.ToString(); break;
|
res = Config.I.inputType.ToString(); break;
|
||||||
|
case "default-folder":
|
||||||
|
res = tle.defaultFolderName ?? tle.source.ToString(false); break;
|
||||||
default:
|
default:
|
||||||
res = x; return false;
|
res = x; return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
res = res.ReplaceInvalidChars(Config.invalidReplaceStr);
|
res = res.ReplaceInvalidChars(Config.I.invalidReplaceStr);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,7 +7,7 @@ namespace FileSkippers
|
||||||
{
|
{
|
||||||
public static class FileSkipperRegistry
|
public static class FileSkipperRegistry
|
||||||
{
|
{
|
||||||
public static FileSkipper GetChecker(SkipMode mode, string dir, FileConditions conditions, M3uEditor m3uEditor)
|
public static FileSkipper GetSkipper(SkipMode mode, string dir, FileConditions conditions, M3uEditor m3uEditor)
|
||||||
{
|
{
|
||||||
bool noConditions = conditions.Equals(new FileConditions());
|
bool noConditions = conditions.Equals(new FileConditions());
|
||||||
return mode switch
|
return mode switch
|
||||||
|
|
|
@ -29,7 +29,7 @@ public static class Help
|
||||||
--profile <names> Configuration profile(s) to use. See --help ""config"".
|
--profile <names> Configuration profile(s) to use. See --help ""config"".
|
||||||
--concurrent-downloads <num> Max concurrent downloads (default: 2)
|
--concurrent-downloads <num> Max concurrent downloads (default: 2)
|
||||||
--m3u <option> Create an m3u8 playlist file in the output directory
|
--m3u <option> Create an m3u8 playlist file in the output directory
|
||||||
'none' (default for single inputs): Do not create
|
'none' (default for string inputs): Do not create
|
||||||
'index' (default): Write a line indexing all downloaded
|
'index' (default): Write a line indexing all downloaded
|
||||||
files, required for skip-not-found or skip-existing=m3u
|
files, required for skip-not-found or skip-existing=m3u
|
||||||
'all': Write the index and a list of paths and fails
|
'all': Write the index and a list of paths and fails
|
||||||
|
@ -204,8 +204,8 @@ public static class Help
|
||||||
Path to a local CSV file: Use a csv file containing track info of the songs to download.
|
Path to a local CSV file: Use a csv file containing track info of the songs to download.
|
||||||
The names of the columns should be Artist, Title, Album, Length, although alternative names
|
The names of the columns should be Artist, Title, Album, Length, although alternative names
|
||||||
are usually detected as well. Only the title or album column is required, but extra info may
|
are usually detected as well. Only the title or album column is required, but extra info may
|
||||||
improve search results. Every row that does not have a title column text will be treated as an
|
improve search result ranking. Every row that does not have a title column text will be treated
|
||||||
album download.
|
as an album download.
|
||||||
|
|
||||||
YouTube
|
YouTube
|
||||||
A playlist url: Download songs from a youtube playlist.
|
A playlist url: Download songs from a youtube playlist.
|
||||||
|
@ -213,10 +213,6 @@ public static class Help
|
||||||
the ones which are unavailable. To get all video titles, you can use the official API by
|
the ones which are unavailable. To get all video titles, you can use the official API by
|
||||||
providing a key with --youtube-key. Get it here https://console.cloud.google.com. Create a
|
providing a key with --youtube-key. Get it here https://console.cloud.google.com. Create a
|
||||||
new project, click ""Enable Api"" and search for ""youtube data"", then follow the prompts.
|
new project, click ""Enable Api"" and search for ""youtube data"", then follow the prompts.
|
||||||
|
|
||||||
Tip: For playlists containing music videos, it may be better to remove all text in parentheses
|
|
||||||
(to remove (Lyrics), (Official), etc) and disable song duration checking:
|
|
||||||
--regex ""[\[\(].*?[\]\)]"" --pref-length-tol -1
|
|
||||||
|
|
||||||
Spotify
|
Spotify
|
||||||
A playlist/album url or 'spotify-likes': Download a spotify playlist, album, or your
|
A playlist/album url or 'spotify-likes': Download a spotify playlist, album, or your
|
||||||
|
@ -255,7 +251,7 @@ public static class Help
|
||||||
(like what you would enter into the soulseek search bar), or a comma-separated list of
|
(like what you would enter into the soulseek search bar), or a comma-separated list of
|
||||||
properties like 'title=Song Name, artist=Artist Name, length=215'.
|
properties like 'title=Song Name, artist=Artist Name, length=215'.
|
||||||
|
|
||||||
The following properties are allowed:
|
The following properties are accepted:
|
||||||
title
|
title
|
||||||
artist
|
artist
|
||||||
album
|
album
|
||||||
|
@ -423,6 +419,7 @@ public static class Help
|
||||||
filename Soulseek filename without extension
|
filename Soulseek filename without extension
|
||||||
foldername Soulseek folder name
|
foldername Soulseek folder name
|
||||||
extractor Name of the extractor used (CSV/Spotify/YouTube/etc)
|
extractor Name of the extractor used (CSV/Spotify/YouTube/etc)
|
||||||
|
default-folder Default sldl folder name (usually the playlist name)
|
||||||
";
|
";
|
||||||
|
|
||||||
const string skipExistingHelp = @"
|
const string skipExistingHelp = @"
|
||||||
|
|
|
@ -6,11 +6,11 @@ using System.Text;
|
||||||
public class M3uEditor
|
public class M3uEditor
|
||||||
{
|
{
|
||||||
public string path { get; private set; }
|
public string path { get; private set; }
|
||||||
|
public M3uOption option = M3uOption.Index;
|
||||||
string parent;
|
string parent;
|
||||||
List<string> lines;
|
List<string> lines;
|
||||||
bool needFirstUpdate = false;
|
bool needFirstUpdate = false;
|
||||||
readonly TrackLists trackLists;
|
readonly TrackLists trackLists;
|
||||||
readonly M3uOption option = M3uOption.Index;
|
|
||||||
readonly Dictionary<string, Track> previousRunData = new(); // { track.ToKey(), track }
|
readonly Dictionary<string, Track> previousRunData = new(); // { track.ToKey(), track }
|
||||||
|
|
||||||
public M3uEditor(TrackLists trackLists, M3uOption option)
|
public M3uEditor(TrackLists trackLists, M3uOption option)
|
||||||
|
@ -27,11 +27,12 @@ public class M3uEditor
|
||||||
|
|
||||||
public void SetPathAndLoad(string path)
|
public void SetPathAndLoad(string path)
|
||||||
{
|
{
|
||||||
if (this.path == path)
|
if (Utils.NormalizedPath(this.path) == Utils.NormalizedPath(path))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
this.path = Path.GetFullPath(path);
|
this.path = Path.GetFullPath(path);
|
||||||
parent = Utils.NormalizedPath(Path.GetDirectoryName(this.path));
|
parent = Utils.NormalizedPath(Path.GetDirectoryName(this.path));
|
||||||
|
|
||||||
lines = ReadAllLines().ToList();
|
lines = ReadAllLines().ToList();
|
||||||
LoadPreviousResults();
|
LoadPreviousResults();
|
||||||
}
|
}
|
||||||
|
@ -81,7 +82,7 @@ public class M3uEditor
|
||||||
if (field == 0)
|
if (field == 0)
|
||||||
{
|
{
|
||||||
if (x.StartsWith("./"))
|
if (x.StartsWith("./"))
|
||||||
x = System.IO.Path.Join(parent, x[2..]);
|
x = Path.Join(parent, x[2..]);
|
||||||
track.DownloadPath = x;
|
track.DownloadPath = x;
|
||||||
}
|
}
|
||||||
else if (field == 1)
|
else if (field == 1)
|
||||||
|
|
|
@ -164,21 +164,21 @@ public static class Printing
|
||||||
{
|
{
|
||||||
Console.WriteLine(new string('-', 60));
|
Console.WriteLine(new string('-', 60));
|
||||||
|
|
||||||
if (!Config.printOption.HasFlag(PrintOption.Full))
|
if (!Config.I.printOption.HasFlag(PrintOption.Full))
|
||||||
Console.WriteLine($"Result 1 of {tle.list.Count} for album {tle.source.ToString(true)}:");
|
Console.WriteLine($"Result 1 of {tle.list.Count} for album {tle.source.ToString(true)}:");
|
||||||
else
|
else
|
||||||
Console.WriteLine($"Results ({tle.list.Count}) for album {tle.source.ToString(true)}:");
|
Console.WriteLine($"Results ({tle.list.Count}) for album {tle.source.ToString(true)}:");
|
||||||
|
|
||||||
if (tle.list.Count > 0 && tle.list[0].Count > 0)
|
if (tle.list.Count > 0 && tle.list[0].Count > 0)
|
||||||
{
|
{
|
||||||
if (!Config.noBrowseFolder)
|
if (!Config.I.noBrowseFolder)
|
||||||
Console.WriteLine("[Skipping full folder retrieval]");
|
Console.WriteLine("[Skipping full folder retrieval]");
|
||||||
|
|
||||||
foreach (var ls in tle.list)
|
foreach (var ls in tle.list)
|
||||||
{
|
{
|
||||||
PrintAlbum(ls);
|
PrintAlbum(ls);
|
||||||
|
|
||||||
if (!Config.printOption.HasFlag(PrintOption.Full))
|
if (!Config.I.printOption.HasFlag(PrintOption.Full))
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -208,14 +208,14 @@ public static class Printing
|
||||||
|
|
||||||
public static void PrintTracksTbd(List<Track> toBeDownloaded, List<Track> existing, List<Track> notFound, TrackType type, bool summary = true)
|
public static void PrintTracksTbd(List<Track> toBeDownloaded, List<Track> existing, List<Track> notFound, TrackType type, bool summary = true)
|
||||||
{
|
{
|
||||||
if (type == TrackType.Normal && !Config.PrintTracks && toBeDownloaded.Count == 1 && existing.Count + notFound.Count == 0)
|
if (type == TrackType.Normal && !Config.I.PrintTracks && toBeDownloaded.Count == 1 && existing.Count + notFound.Count == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
string notFoundLastTime = notFound.Count > 0 ? $"{notFound.Count} not found" : "";
|
string notFoundLastTime = notFound.Count > 0 ? $"{notFound.Count} not found" : "";
|
||||||
string alreadyExist = existing.Count > 0 ? $"{existing.Count} already exist" : "";
|
string alreadyExist = existing.Count > 0 ? $"{existing.Count} already exist" : "";
|
||||||
notFoundLastTime = alreadyExist.Length > 0 && notFoundLastTime.Length > 0 ? ", " + notFoundLastTime : notFoundLastTime;
|
notFoundLastTime = alreadyExist.Length > 0 && notFoundLastTime.Length > 0 ? ", " + notFoundLastTime : notFoundLastTime;
|
||||||
string skippedTracks = alreadyExist.Length + notFoundLastTime.Length > 0 ? $" ({alreadyExist}{notFoundLastTime})" : "";
|
string skippedTracks = alreadyExist.Length + notFoundLastTime.Length > 0 ? $" ({alreadyExist}{notFoundLastTime})" : "";
|
||||||
bool full = Config.printOption.HasFlag(PrintOption.Full);
|
bool full = Config.I.printOption.HasFlag(PrintOption.Full);
|
||||||
bool allSkipped = existing.Count + notFound.Count > toBeDownloaded.Count;
|
bool allSkipped = existing.Count + notFound.Count > toBeDownloaded.Count;
|
||||||
|
|
||||||
if (summary && (type == TrackType.Normal || skippedTracks.Length > 0))
|
if (summary && (type == TrackType.Normal || skippedTracks.Length > 0))
|
||||||
|
@ -223,27 +223,26 @@ public static class Printing
|
||||||
|
|
||||||
if (toBeDownloaded.Count > 0)
|
if (toBeDownloaded.Count > 0)
|
||||||
{
|
{
|
||||||
bool showAll = type != TrackType.Normal || Config.PrintTracks || Config.PrintResults;
|
bool showAll = type != TrackType.Normal || Config.I.PrintTracks || Config.I.PrintResults;
|
||||||
PrintTracks(toBeDownloaded, showAll ? int.MaxValue : 10, full, infoFirst: Config.PrintTracks);
|
PrintTracks(toBeDownloaded, showAll ? int.MaxValue : 10, full, infoFirst: Config.I.PrintTracks);
|
||||||
|
|
||||||
if (full && (existing.Count > 0 || notFound.Count > 0))
|
if (full && (existing.Count > 0 || notFound.Count > 0))
|
||||||
Console.WriteLine("\n-----------------------------------------------\n");
|
Console.WriteLine("\n-----------------------------------------------\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Config.PrintTracks || Config.PrintResults)
|
if (Config.I.PrintTracks || Config.I.PrintResults)
|
||||||
{
|
{
|
||||||
if (existing.Count > 0)
|
if (existing.Count > 0)
|
||||||
{
|
{
|
||||||
Console.WriteLine($"\nThe following tracks already exist:");
|
Console.WriteLine($"\nThe following tracks already exist:");
|
||||||
PrintTracks(existing, fullInfo: full, infoFirst: Config.PrintTracks);
|
PrintTracks(existing, fullInfo: full, infoFirst: Config.I.PrintTracks);
|
||||||
}
|
}
|
||||||
if (notFound.Count > 0)
|
if (notFound.Count > 0)
|
||||||
{
|
{
|
||||||
Console.WriteLine($"\nThe following tracks were not found during a prior run:");
|
Console.WriteLine($"\nThe following tracks were not found during a prior run:");
|
||||||
PrintTracks(notFound, fullInfo: full, infoFirst: Config.PrintTracks);
|
PrintTracks(notFound, fullInfo: full, infoFirst: Config.I.PrintTracks);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Console.WriteLine();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -294,9 +293,7 @@ public static class Printing
|
||||||
|
|
||||||
res += $" / {totalFileSizeInMB:F2} MB]";
|
res += $" / {totalFileSizeInMB:F2} MB]";
|
||||||
|
|
||||||
string gcp = Utils.GreatestCommonDirectory(files.Select(x => x.Filename)).TrimEnd('\\');
|
string gcp = Utils.GreatestCommonDirectorySlsk(files.Select(x => x.Filename)).TrimEnd('\\');
|
||||||
|
|
||||||
var discPattern = new Regex(@"^(?i)(dis[c|k]|cd)\s*\d{1,2}$");
|
|
||||||
int lastIndex = gcp.LastIndexOf('\\');
|
int lastIndex = gcp.LastIndexOf('\\');
|
||||||
if (lastIndex != -1)
|
if (lastIndex != -1)
|
||||||
{
|
{
|
||||||
|
@ -315,14 +312,14 @@ public static class Printing
|
||||||
try { progress.Refresh(current, item); }
|
try { progress.Refresh(current, item); }
|
||||||
catch { }
|
catch { }
|
||||||
}
|
}
|
||||||
else if ((Config.displayMode == DisplayMode.Simple || Console.IsOutputRedirected) && print)
|
else if ((Config.I.displayMode == DisplayMode.Simple || Console.IsOutputRedirected) && print)
|
||||||
Console.WriteLine(item);
|
Console.WriteLine(item);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public static void WriteLine(string value, ConsoleColor color = ConsoleColor.Gray, bool safe = false, bool debugOnly = false)
|
public static void WriteLine(string value, ConsoleColor color = ConsoleColor.Gray, bool safe = false, bool debugOnly = false)
|
||||||
{
|
{
|
||||||
if (debugOnly && !Config.debugInfo)
|
if (debugOnly && !Config.I.debugInfo)
|
||||||
return;
|
return;
|
||||||
if (!safe)
|
if (!safe)
|
||||||
{
|
{
|
||||||
|
|
|
@ -15,17 +15,13 @@ using static Printing;
|
||||||
|
|
||||||
using Directory = System.IO.Directory;
|
using Directory = System.IO.Directory;
|
||||||
using File = System.IO.File;
|
using File = System.IO.File;
|
||||||
using ProgressBar = Konsole.ProgressBar;
|
|
||||||
using SearchResponse = Soulseek.SearchResponse;
|
|
||||||
using SlFile = Soulseek.File;
|
using SlFile = Soulseek.File;
|
||||||
using SlResponse = Soulseek.SearchResponse;
|
|
||||||
|
|
||||||
|
|
||||||
static partial class Program
|
static partial class Program
|
||||||
{
|
{
|
||||||
public static bool skipUpdate = false;
|
public static bool skipUpdate = false;
|
||||||
public static bool initialized = false;
|
public static bool initialized = false;
|
||||||
public static Extractors.IExtractor? extractor;
|
public static IExtractor? extractor;
|
||||||
public static FileSkipper? outputDirSkipper;
|
public static FileSkipper? outputDirSkipper;
|
||||||
public static FileSkipper? musicDirSkipper;
|
public static FileSkipper? musicDirSkipper;
|
||||||
public static SoulseekClient? client;
|
public static SoulseekClient? client;
|
||||||
|
@ -48,33 +44,28 @@ static partial class Program
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool doContinue = Config.ParseArgsAndReadConfig(args);
|
Config.I.Load(args);
|
||||||
|
|
||||||
if (!doContinue)
|
if (Config.I.input.Length == 0)
|
||||||
return;
|
|
||||||
|
|
||||||
if (Config.input.Length == 0)
|
|
||||||
throw new ArgumentException($"No input provided");
|
throw new ArgumentException($"No input provided");
|
||||||
|
|
||||||
(Config.inputType, extractor) = ExtractorRegistry.GetMatchingExtractor(Config.input, Config.inputType);
|
(Config.I.inputType, extractor) = ExtractorRegistry.GetMatchingExtractor(Config.I.input, Config.I.inputType);
|
||||||
|
|
||||||
WriteLine($"Using extractor: {Config.inputType}", debugOnly: true);
|
WriteLine($"Using extractor: {Config.I.inputType}", debugOnly: true);
|
||||||
|
|
||||||
trackLists = await extractor.GetTracks(Config.input, Config.maxTracks, Config.offset, Config.reverse);
|
trackLists = await extractor.GetTracks(Config.I.input, Config.I.maxTracks, Config.I.offset, Config.I.reverse);
|
||||||
|
|
||||||
WriteLine("Got tracks", debugOnly: true);
|
WriteLine("Got tracks", debugOnly: true);
|
||||||
|
|
||||||
trackLists.UpgradeListTypes(Config.aggregate, Config.album);
|
Config.I.PostProcessArgs();
|
||||||
|
|
||||||
|
trackLists.UpgradeListTypes(Config.I.aggregate, Config.I.album);
|
||||||
trackLists.SetListEntryOptions();
|
trackLists.SetListEntryOptions();
|
||||||
|
|
||||||
Config.PostProcessArgs();
|
m3uEditor = new M3uEditor(trackLists, Config.I.m3uOption);
|
||||||
|
|
||||||
m3uEditor = new M3uEditor(trackLists, Config.m3uOption);
|
|
||||||
|
|
||||||
InitFileSkippers();
|
|
||||||
|
|
||||||
await MainLoop();
|
await MainLoop();
|
||||||
|
|
||||||
WriteLine("Mainloop done", debugOnly: true);
|
WriteLine("Mainloop done", debugOnly: true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -84,7 +75,7 @@ static partial class Program
|
||||||
if (initialized)
|
if (initialized)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
bool needLogin = !Config.PrintTracks;
|
bool needLogin = !Config.I.PrintTracks;
|
||||||
if (needLogin)
|
if (needLogin)
|
||||||
{
|
{
|
||||||
var connectionOptions = new ConnectionOptions(configureSocket: (socket) =>
|
var connectionOptions = new ConnectionOptions(configureSocket: (socket) =>
|
||||||
|
@ -98,17 +89,17 @@ static partial class Program
|
||||||
var clientOptions = new SoulseekClientOptions(
|
var clientOptions = new SoulseekClientOptions(
|
||||||
transferConnectionOptions: connectionOptions,
|
transferConnectionOptions: connectionOptions,
|
||||||
serverConnectionOptions: connectionOptions,
|
serverConnectionOptions: connectionOptions,
|
||||||
listenPort: Config.listenPort
|
listenPort: Config.I.listenPort
|
||||||
);
|
);
|
||||||
|
|
||||||
client = new SoulseekClient(clientOptions);
|
client = new SoulseekClient(clientOptions);
|
||||||
|
|
||||||
if (!Config.useRandomLogin && (string.IsNullOrEmpty(Config.username) || string.IsNullOrEmpty(Config.password)))
|
if (!Config.I.useRandomLogin && (string.IsNullOrEmpty(Config.I.username) || string.IsNullOrEmpty(Config.I.password)))
|
||||||
throw new ArgumentException("No soulseek username or password");
|
throw new ArgumentException("No soulseek username or password");
|
||||||
|
|
||||||
await Login(Config.useRandomLogin);
|
await Login(Config.I.useRandomLogin);
|
||||||
|
|
||||||
Search.searchSemaphore = new RateLimitedSemaphore(Config.searchesPerTime, TimeSpan.FromSeconds(Config.searchRenewTime));
|
Search.searchSemaphore = new RateLimitedSemaphore(Config.I.searchesPerTime, TimeSpan.FromSeconds(Config.I.searchRenewTime));
|
||||||
}
|
}
|
||||||
|
|
||||||
bool needUpdate = needLogin;
|
bool needUpdate = needLogin;
|
||||||
|
@ -124,19 +115,18 @@ static partial class Program
|
||||||
|
|
||||||
static void InitFileSkippers()
|
static void InitFileSkippers()
|
||||||
{
|
{
|
||||||
if (Config.skipExisting)
|
if (Config.I.skipExisting)
|
||||||
{
|
{
|
||||||
var cond = Config.skipExistingPrefCond ? Config.preferredCond : Config.necessaryCond;
|
var cond = Config.I.skipExistingPrefCond ? Config.I.preferredCond : Config.I.necessaryCond;
|
||||||
|
|
||||||
if (Config.musicDir.Length == 0 || !Config.parentDir.StartsWith(Config.musicDir, StringComparison.OrdinalIgnoreCase))
|
outputDirSkipper = FileSkipperRegistry.GetSkipper(Config.I.skipMode, Config.I.parentDir, cond, m3uEditor);
|
||||||
outputDirSkipper = FileSkipperRegistry.GetChecker(Config.skipMode, Config.parentDir, cond, m3uEditor);
|
|
||||||
|
|
||||||
if (Config.musicDir.Length > 0)
|
if (Config.I.musicDir.Length > 0)
|
||||||
{
|
{
|
||||||
if (!Directory.Exists(Config.musicDir))
|
if (!Directory.Exists(Config.I.musicDir))
|
||||||
Console.WriteLine("Error: Music directory does not exist");
|
Console.WriteLine("Error: Music directory does not exist");
|
||||||
else
|
else
|
||||||
musicDirSkipper = FileSkipperRegistry.GetChecker(Config.skipModeMusicDir, Config.musicDir, cond, m3uEditor);
|
musicDirSkipper = FileSkipperRegistry.GetSkipper(Config.I.skipModeMusicDir, Config.I.musicDir, cond, m3uEditor);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -144,9 +134,10 @@ static partial class Program
|
||||||
|
|
||||||
static void PreprocessTracks(TrackListEntry tle)
|
static void PreprocessTracks(TrackListEntry tle)
|
||||||
{
|
{
|
||||||
|
PreprocessTrack(tle.source);
|
||||||
|
|
||||||
for (int k = 0; k < tle.list.Count; k++)
|
for (int k = 0; k < tle.list.Count; k++)
|
||||||
{
|
{
|
||||||
PreprocessTrack(tle.source);
|
|
||||||
foreach (var ls in tle.list)
|
foreach (var ls in tle.list)
|
||||||
{
|
{
|
||||||
for (int i = 0; i < ls.Count; i++)
|
for (int i = 0; i < ls.Count; i++)
|
||||||
|
@ -160,22 +151,22 @@ static partial class Program
|
||||||
|
|
||||||
static void PreprocessTrack(Track track)
|
static void PreprocessTrack(Track track)
|
||||||
{
|
{
|
||||||
if (Config.removeFt)
|
if (Config.I.removeFt)
|
||||||
{
|
{
|
||||||
track.Title = track.Title.RemoveFt();
|
track.Title = track.Title.RemoveFt();
|
||||||
track.Artist = track.Artist.RemoveFt();
|
track.Artist = track.Artist.RemoveFt();
|
||||||
}
|
}
|
||||||
if (Config.removeBrackets)
|
if (Config.I.removeBrackets)
|
||||||
{
|
{
|
||||||
track.Title = track.Title.RemoveSquareBrackets();
|
track.Title = track.Title.RemoveSquareBrackets();
|
||||||
}
|
}
|
||||||
if (Config.regexToReplace.Title.Length + Config.regexToReplace.Artist.Length + Config.regexToReplace.Album.Length > 0)
|
if (Config.I.regexToReplace.Title.Length + Config.I.regexToReplace.Artist.Length + Config.I.regexToReplace.Album.Length > 0)
|
||||||
{
|
{
|
||||||
track.Title = Regex.Replace(track.Title, Config.regexToReplace.Title, Config.regexReplaceBy.Title);
|
track.Title = Regex.Replace(track.Title, Config.I.regexToReplace.Title, Config.I.regexReplaceBy.Title);
|
||||||
track.Artist = Regex.Replace(track.Artist, Config.regexToReplace.Artist, Config.regexReplaceBy.Artist);
|
track.Artist = Regex.Replace(track.Artist, Config.I.regexToReplace.Artist, Config.I.regexReplaceBy.Artist);
|
||||||
track.Album = Regex.Replace(track.Album, Config.regexToReplace.Album, Config.regexReplaceBy.Album);
|
track.Album = Regex.Replace(track.Album, Config.I.regexToReplace.Album, Config.I.regexReplaceBy.Album);
|
||||||
}
|
}
|
||||||
if (Config.artistMaybeWrong)
|
if (Config.I.artistMaybeWrong)
|
||||||
{
|
{
|
||||||
track.ArtistMaybeWrong = true;
|
track.ArtistMaybeWrong = true;
|
||||||
}
|
}
|
||||||
|
@ -186,22 +177,28 @@ static partial class Program
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void PrepareListEntry(TrackListEntry tle)
|
static void PrepareListEntry(TrackListEntry tle, bool isFirstEntry)
|
||||||
{
|
{
|
||||||
Config.RestoreConditions();
|
Config.I.RestoreConditions();
|
||||||
|
|
||||||
Config.UpdateProfiles(tle);
|
bool changed = Config.UpdateProfiles(tle);
|
||||||
|
|
||||||
Config.AddTemporaryConditions(tle.additionalConds, tle.additionalPrefConds);
|
Config.I.AddTemporaryConditions(tle.additionalConds, tle.additionalPrefConds);
|
||||||
|
|
||||||
string m3uPath;
|
string m3uPath;
|
||||||
|
|
||||||
if (Config.m3uFilePath.Length > 0)
|
if (Config.I.m3uFilePath.Length > 0)
|
||||||
m3uPath = Config.m3uFilePath;
|
m3uPath = Config.I.m3uFilePath;
|
||||||
else
|
else
|
||||||
m3uPath = Path.Join(Config.parentDir, tle.defaultFolderName, "sldl.m3u");
|
m3uPath = Path.Join(Config.I.parentDir, tle.defaultFolderName, "sldl.m3u8");
|
||||||
|
|
||||||
m3uEditor.SetPathAndLoad(m3uPath);
|
m3uEditor.option = Config.I.m3uOption;
|
||||||
|
m3uEditor.SetPathAndLoad(m3uPath); // does nothing if the path is the same
|
||||||
|
|
||||||
|
if (changed || isFirstEntry)
|
||||||
|
{
|
||||||
|
InitFileSkippers(); // todo: only do this when a relevant config item changes
|
||||||
|
}
|
||||||
|
|
||||||
PreprocessTracks(tle);
|
PreprocessTracks(tle);
|
||||||
}
|
}
|
||||||
|
@ -211,16 +208,16 @@ static partial class Program
|
||||||
{
|
{
|
||||||
for (int i = 0; i < trackLists.lists.Count; i++)
|
for (int i = 0; i < trackLists.lists.Count; i++)
|
||||||
{
|
{
|
||||||
if (i > 0) Console.WriteLine();
|
Console.WriteLine();
|
||||||
|
|
||||||
var tle = trackLists[i];
|
var tle = trackLists[i];
|
||||||
|
|
||||||
PrepareListEntry(tle);
|
PrepareListEntry(tle, isFirstEntry: i == 0);
|
||||||
|
|
||||||
var existing = new List<Track>();
|
var existing = new List<Track>();
|
||||||
var notFound = new List<Track>();
|
var notFound = new List<Track>();
|
||||||
|
|
||||||
if (Config.skipNotFound && !Config.PrintResults)
|
if (Config.I.skipNotFound && !Config.I.PrintResults)
|
||||||
{
|
{
|
||||||
if (tle.sourceCanBeSkipped && SetNotFoundLastTime(tle.source))
|
if (tle.sourceCanBeSkipped && SetNotFoundLastTime(tle.source))
|
||||||
notFound.Add(tle.source);
|
notFound.Add(tle.source);
|
||||||
|
@ -232,7 +229,7 @@ static partial class Program
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Config.skipExisting && !Config.PrintResults && tle.source.State != TrackState.NotFoundLastTime)
|
if (Config.I.skipExisting && !Config.I.PrintResults && tle.source.State != TrackState.NotFoundLastTime)
|
||||||
{
|
{
|
||||||
if (tle.sourceCanBeSkipped && SetExisting(tle.source))
|
if (tle.sourceCanBeSkipped && SetExisting(tle.source))
|
||||||
existing.Add(tle.source);
|
existing.Add(tle.source);
|
||||||
|
@ -244,7 +241,7 @@ static partial class Program
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Config.PrintTracks)
|
if (Config.I.PrintTracks)
|
||||||
{
|
{
|
||||||
if (tle.source.Type == TrackType.Normal)
|
if (tle.source.Type == TrackType.Normal)
|
||||||
{
|
{
|
||||||
|
@ -286,12 +283,12 @@ static partial class Program
|
||||||
if (tle.source.Type == TrackType.Album)
|
if (tle.source.Type == TrackType.Album)
|
||||||
{
|
{
|
||||||
tle.list = await Search.GetAlbumDownloads(tle.source, responseData);
|
tle.list = await Search.GetAlbumDownloads(tle.source, responseData);
|
||||||
foundSomething = tle.list.Count > 0;
|
foundSomething = tle.list.Count > 0 && tle.list[0].Count > 0;
|
||||||
}
|
}
|
||||||
else if (tle.source.Type == TrackType.Aggregate)
|
else if (tle.source.Type == TrackType.Aggregate)
|
||||||
{
|
{
|
||||||
tle.list.Insert(0, await Search.GetAggregateTracks(tle.source, responseData));
|
tle.list.Insert(0, await Search.GetAggregateTracks(tle.source, responseData));
|
||||||
foundSomething = tle.list.Count > 0;
|
foundSomething = tle.list.Count > 0 && tle.list[0].Count > 0;
|
||||||
}
|
}
|
||||||
else if (tle.source.Type == TrackType.AlbumAggregate)
|
else if (tle.source.Type == TrackType.AlbumAggregate)
|
||||||
{
|
{
|
||||||
|
@ -313,7 +310,7 @@ static partial class Program
|
||||||
var lockedFiles = responseData.lockedFilesCount > 0 ? $" (Found {responseData.lockedFilesCount} locked files)" : "";
|
var lockedFiles = responseData.lockedFilesCount > 0 ? $" (Found {responseData.lockedFilesCount} locked files)" : "";
|
||||||
Console.WriteLine($"No results.{lockedFiles}");
|
Console.WriteLine($"No results.{lockedFiles}");
|
||||||
|
|
||||||
if (!Config.PrintResults)
|
if (!Config.I.PrintResults)
|
||||||
{
|
{
|
||||||
tle.source.State = TrackState.Failed;
|
tle.source.State = TrackState.Failed;
|
||||||
tle.source.FailureReason = FailureReason.NoSuitableFileFound;
|
tle.source.FailureReason = FailureReason.NoSuitableFileFound;
|
||||||
|
@ -323,7 +320,7 @@ static partial class Program
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Config.skipExisting && tle.needSkipExistingAfterSearch)
|
if (Config.I.skipExisting && tle.needSkipExistingAfterSearch)
|
||||||
{
|
{
|
||||||
foreach (var tracks in tle.list)
|
foreach (var tracks in tle.list)
|
||||||
existing.AddRange(DoSkipExisting(tracks));
|
existing.AddRange(DoSkipExisting(tracks));
|
||||||
|
@ -335,7 +332,7 @@ static partial class Program
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Config.PrintResults)
|
if (Config.I.PrintResults)
|
||||||
{
|
{
|
||||||
await PrintResults(tle, existing, notFound);
|
await PrintResults(tle, existing, notFound);
|
||||||
continue;
|
continue;
|
||||||
|
@ -369,7 +366,7 @@ static partial class Program
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!Config.DoNotDownload && (trackLists.lists.Count > 0 || trackLists.Flattened(false, false).Skip(1).Any()))
|
if (!Config.I.DoNotDownload && (trackLists.lists.Count > 0 || trackLists.Flattened(false, false).Skip(1).Any()))
|
||||||
{
|
{
|
||||||
PrintComplete(trackLists);
|
PrintComplete(trackLists);
|
||||||
}
|
}
|
||||||
|
@ -455,7 +452,7 @@ static partial class Program
|
||||||
{
|
{
|
||||||
var tracks = tle.list[0];
|
var tracks = tle.list[0];
|
||||||
|
|
||||||
var semaphore = new SemaphoreSlim(Config.concurrentProcesses);
|
var semaphore = new SemaphoreSlim(Config.I.concurrentProcesses);
|
||||||
|
|
||||||
var organizer = new FileManager(tle);
|
var organizer = new FileManager(tle);
|
||||||
|
|
||||||
|
@ -468,7 +465,7 @@ static partial class Program
|
||||||
|
|
||||||
await Task.WhenAll(downloadTasks);
|
await Task.WhenAll(downloadTasks);
|
||||||
|
|
||||||
if (Config.removeTracksFromSource && tracks.All(t => t.State == TrackState.Downloaded || t.State == TrackState.AlreadyExists))
|
if (Config.I.removeTracksFromSource && tracks.All(t => t.State == TrackState.Downloaded || t.State == TrackState.AlreadyExists))
|
||||||
await extractor.RemoveTrackFromSource(tle.source);
|
await extractor.RemoveTrackFromSource(tle.source);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -481,14 +478,14 @@ static partial class Program
|
||||||
bool succeeded = false;
|
bool succeeded = false;
|
||||||
string? soulseekDir = null;
|
string? soulseekDir = null;
|
||||||
|
|
||||||
while (tle.list.Count > 0 && !Config.albumArtOnly)
|
while (tle.list.Count > 0 && !Config.I.albumArtOnly)
|
||||||
{
|
{
|
||||||
int index = 0;
|
int index = 0;
|
||||||
bool wasInteractive = Config.interactiveMode;
|
bool wasInteractive = Config.I.interactiveMode;
|
||||||
|
|
||||||
if (Config.interactiveMode)
|
if (Config.I.interactiveMode)
|
||||||
{
|
{
|
||||||
index = await InteractiveModeAlbum(tle.list, !Config.noBrowseFolder, retrievedFolders);
|
index = await InteractiveModeAlbum(tle.list, !Config.I.noBrowseFolder, retrievedFolders);
|
||||||
if (index == -1) break;
|
if (index == -1) break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -498,20 +495,20 @@ static partial class Program
|
||||||
|
|
||||||
organizer.SetRemoteCommonDir(soulseekDir);
|
organizer.SetRemoteCommonDir(soulseekDir);
|
||||||
|
|
||||||
if (!Config.interactiveMode && !wasInteractive)
|
if (!Config.I.interactiveMode && !wasInteractive)
|
||||||
{
|
{
|
||||||
Console.WriteLine();
|
Console.WriteLine();
|
||||||
PrintAlbum(tracks);
|
PrintAlbum(tracks);
|
||||||
}
|
}
|
||||||
|
|
||||||
var semaphore = new SemaphoreSlim(Config.concurrentProcesses);
|
var semaphore = new SemaphoreSlim(Config.I.concurrentProcesses);
|
||||||
using var cts = new CancellationTokenSource();
|
using var cts = new CancellationTokenSource();
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await RunAlbumDownloads(tle, organizer, tracks, semaphore, cts);
|
await RunAlbumDownloads(tle, organizer, tracks, semaphore, cts);
|
||||||
|
|
||||||
if (!Config.noBrowseFolder && !retrievedFolders.Contains(soulseekDir))
|
if (!Config.I.noBrowseFolder && !retrievedFolders.Contains(soulseekDir))
|
||||||
{
|
{
|
||||||
Console.WriteLine("Getting all files in folder...");
|
Console.WriteLine("Getting all files in folder...");
|
||||||
|
|
||||||
|
@ -548,10 +545,10 @@ static partial class Program
|
||||||
|
|
||||||
List<Track>? additionalImages = null;
|
List<Track>? additionalImages = null;
|
||||||
|
|
||||||
if (Config.albumArtOnly || succeeded && Config.albumArtOption != AlbumArtOption.Default)
|
if (Config.I.albumArtOnly || succeeded && Config.I.albumArtOption != AlbumArtOption.Default)
|
||||||
{
|
{
|
||||||
Console.WriteLine($"\nDownloading additional images:");
|
Console.WriteLine($"\nDownloading additional images:");
|
||||||
additionalImages = await DownloadImages(tle.list, Config.albumArtOption, tracks, organizer);
|
additionalImages = await DownloadImages(tle.list, Config.I.albumArtOption, tracks, organizer);
|
||||||
tracks?.AddRange(additionalImages);
|
tracks?.AddRange(additionalImages);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -586,7 +583,7 @@ static partial class Program
|
||||||
tle.source.State = TrackState.Downloaded;
|
tle.source.State = TrackState.Downloaded;
|
||||||
tle.source.DownloadPath = Utils.GreatestCommonDirectory(downloadedAudio.Select(t => t.DownloadPath));
|
tle.source.DownloadPath = Utils.GreatestCommonDirectory(downloadedAudio.Select(t => t.DownloadPath));
|
||||||
|
|
||||||
if (Config.removeTracksFromSource)
|
if (Config.I.removeTracksFromSource)
|
||||||
{
|
{
|
||||||
await extractor.RemoveTrackFromSource(tle.source);
|
await extractor.RemoveTrackFromSource(tle.source);
|
||||||
}
|
}
|
||||||
|
@ -596,7 +593,7 @@ static partial class Program
|
||||||
|
|
||||||
static void OnAlbumFail(List<Track>? tracks)
|
static void OnAlbumFail(List<Track>? tracks)
|
||||||
{
|
{
|
||||||
if (tracks == null || Config.IgnoreAlbumFail)
|
if (tracks == null || Config.I.IgnoreAlbumFail)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
foreach (var track in tracks)
|
foreach (var track in tracks)
|
||||||
|
@ -605,18 +602,18 @@ static partial class Program
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (Config.DeleteAlbumOnFail)
|
if (Config.I.DeleteAlbumOnFail)
|
||||||
{
|
{
|
||||||
File.Delete(track.DownloadPath);
|
File.Delete(track.DownloadPath);
|
||||||
}
|
}
|
||||||
else if (Config.failedAlbumPath.Length > 0)
|
else if (Config.I.failedAlbumPath.Length > 0)
|
||||||
{
|
{
|
||||||
var newPath = Path.Join(Config.failedAlbumPath, Path.GetRelativePath(Config.parentDir, track.DownloadPath));
|
var newPath = Path.Join(Config.I.failedAlbumPath, Path.GetRelativePath(Config.I.parentDir, track.DownloadPath));
|
||||||
Directory.CreateDirectory(Path.GetDirectoryName(newPath));
|
Directory.CreateDirectory(Path.GetDirectoryName(newPath));
|
||||||
Utils.Move(track.DownloadPath, newPath);
|
Utils.Move(track.DownloadPath, newPath);
|
||||||
}
|
}
|
||||||
|
|
||||||
Utils.DeleteAncestorsIfEmpty(Path.GetDirectoryName(track.DownloadPath), Config.parentDir);
|
Utils.DeleteAncestorsIfEmpty(Path.GetDirectoryName(track.DownloadPath), Config.I.parentDir);
|
||||||
}
|
}
|
||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
|
@ -698,9 +695,9 @@ static partial class Program
|
||||||
while (albumArtLists.Count > 0)
|
while (albumArtLists.Count > 0)
|
||||||
{
|
{
|
||||||
int index = 0;
|
int index = 0;
|
||||||
bool wasInteractive = Config.interactiveMode;
|
bool wasInteractive = Config.I.interactiveMode;
|
||||||
|
|
||||||
if (Config.interactiveMode)
|
if (Config.I.interactiveMode)
|
||||||
{
|
{
|
||||||
index = await InteractiveModeAlbum(albumArtLists, false, null);
|
index = await InteractiveModeAlbum(albumArtLists, false, null);
|
||||||
if (index == -1) break;
|
if (index == -1) break;
|
||||||
|
@ -715,12 +712,17 @@ static partial class Program
|
||||||
return downloadedImages;
|
return downloadedImages;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!Config.interactiveMode && !wasInteractive)
|
if (!Config.I.interactiveMode && !wasInteractive)
|
||||||
{
|
{
|
||||||
Console.WriteLine();
|
Console.WriteLine();
|
||||||
PrintAlbum(tracks);
|
PrintAlbum(tracks);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (fileManager.remoteCommonDir == null)
|
||||||
|
{
|
||||||
|
fileManager.SetRemoteCommonDir(Utils.GreatestCommonDirectorySlsk(tracks.Select(t => t.FirstDownload.Filename)));
|
||||||
|
}
|
||||||
|
|
||||||
bool allSucceeded = true;
|
bool allSucceeded = true;
|
||||||
var semaphore = new SemaphoreSlim(1);
|
var semaphore = new SemaphoreSlim(1);
|
||||||
|
|
||||||
|
@ -743,14 +745,14 @@ static partial class Program
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static async Task DownloadTask(TrackListEntry? tle, Track track, SemaphoreSlim semaphore, FileManager organizer, CancellationTokenSource cts, bool cancelOnFail, bool removeFromSource, bool organize)
|
static async Task DownloadTask(TrackListEntry? tle, Track track, SemaphoreSlim semaphore, FileManager organizer, CancellationTokenSource? cts, bool cancelOnFail, bool removeFromSource, bool organize)
|
||||||
{
|
{
|
||||||
if (track.State != TrackState.Initial)
|
if (track.State != TrackState.Initial)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
await semaphore.WaitAsync(cts.Token);
|
await semaphore.WaitAsync(cts.Token);
|
||||||
|
|
||||||
int tries = Config.unknownErrorRetries;
|
int tries = Config.I.unknownErrorRetries;
|
||||||
string savedFilePath = "";
|
string savedFilePath = "";
|
||||||
SlFile? chosenFile = null;
|
SlFile? chosenFile = null;
|
||||||
|
|
||||||
|
@ -809,7 +811,7 @@ static partial class Program
|
||||||
track.DownloadPath = savedFilePath;
|
track.DownloadPath = savedFilePath;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (removeFromSource && Config.removeTracksFromSource)
|
if (removeFromSource && Config.I.removeTracksFromSource)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
@ -827,9 +829,9 @@ static partial class Program
|
||||||
organizer?.OrganizeAudio(track, chosenFile);
|
organizer?.OrganizeAudio(track, chosenFile);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Config.onComplete.Length > 0)
|
if (Config.I.onComplete.Length > 0)
|
||||||
{
|
{
|
||||||
OnComplete(Config.onComplete, track);
|
OnComplete(Config.I.onComplete, track);
|
||||||
}
|
}
|
||||||
|
|
||||||
semaphore.Release();
|
semaphore.Release();
|
||||||
|
@ -890,9 +892,11 @@ static partial class Program
|
||||||
case "s":
|
case "s":
|
||||||
return -1;
|
return -1;
|
||||||
case "q":
|
case "q":
|
||||||
Config.interactiveMode = false;
|
Config.I.interactiveMode = false;
|
||||||
return aidx;
|
return aidx;
|
||||||
case "r":
|
case "r":
|
||||||
|
if (!retrieveFolder)
|
||||||
|
break;
|
||||||
var folder = Utils.GreatestCommonDirectorySlsk(tracks.Select(t => t.FirstDownload.Filename));
|
var folder = Utils.GreatestCommonDirectorySlsk(tracks.Select(t => t.FirstDownload.Filename));
|
||||||
if (retrieveFolder && !retrievedFolders.Contains(username + '\\' + folder))
|
if (retrieveFolder && !retrievedFolders.Contains(username + '\\' + folder))
|
||||||
{
|
{
|
||||||
|
@ -939,7 +943,7 @@ static partial class Program
|
||||||
{
|
{
|
||||||
lock (val)
|
lock (val)
|
||||||
{
|
{
|
||||||
if ((DateTime.Now - val.UpdateLastChangeTime()).TotalMilliseconds > Config.maxStaleTime)
|
if ((DateTime.Now - val.UpdateLastChangeTime()).TotalMilliseconds > Config.I.maxStaleTime)
|
||||||
{
|
{
|
||||||
val.stalled = true;
|
val.stalled = true;
|
||||||
val.UpdateText();
|
val.UpdateText();
|
||||||
|
@ -966,10 +970,10 @@ static partial class Program
|
||||||
&& !client.State.HasFlag(SoulseekClientStates.Connecting))
|
&& !client.State.HasFlag(SoulseekClientStates.Connecting))
|
||||||
{
|
{
|
||||||
WriteLine($"\nDisconnected, logging in\n", ConsoleColor.DarkYellow, true);
|
WriteLine($"\nDisconnected, logging in\n", ConsoleColor.DarkYellow, true);
|
||||||
try { await Login(Config.useRandomLogin); }
|
try { await Login(Config.I.useRandomLogin); }
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
string banMsg = Config.useRandomLogin ? "" : " (possibly a 30-minute ban caused by frequent searches)";
|
string banMsg = Config.I.useRandomLogin ? "" : " (possibly a 30-minute ban caused by frequent searches)";
|
||||||
WriteLine($"{ex.Message}{banMsg}", ConsoleColor.DarkYellow, true);
|
WriteLine($"{ex.Message}{banMsg}", ConsoleColor.DarkYellow, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -989,14 +993,14 @@ static partial class Program
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
await Task.Delay(Config.updateDelay);
|
await Task.Delay(Config.I.updateDelay);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static async Task Login(bool random = false, int tries = 3)
|
static async Task Login(bool random = false, int tries = 3)
|
||||||
{
|
{
|
||||||
string user = Config.username, pass = Config.password;
|
string user = Config.I.username, pass = Config.I.password;
|
||||||
if (random)
|
if (random)
|
||||||
{
|
{
|
||||||
var r = new Random();
|
var r = new Random();
|
||||||
|
@ -1013,7 +1017,7 @@ static partial class Program
|
||||||
{
|
{
|
||||||
WriteLine($"Connecting {user}", debugOnly: true);
|
WriteLine($"Connecting {user}", debugOnly: true);
|
||||||
await client.ConnectAsync(user, pass);
|
await client.ConnectAsync(user, pass);
|
||||||
if (!Config.noModifyShareCount)
|
if (!Config.I.noModifyShareCount)
|
||||||
{
|
{
|
||||||
WriteLine($"Setting share count", debugOnly: true);
|
WriteLine($"Setting share count", debugOnly: true);
|
||||||
await client.SetSharedCountsAsync(20, 100);
|
await client.SetSharedCountsAsync(20, 100);
|
||||||
|
@ -1074,7 +1078,7 @@ static partial class Program
|
||||||
.Replace("{failure-reason}", track.FailureReason.ToString())
|
.Replace("{failure-reason}", track.FailureReason.ToString())
|
||||||
.Replace("{path}", track.DownloadPath)
|
.Replace("{path}", track.DownloadPath)
|
||||||
.Replace("{state}", track.State.ToString())
|
.Replace("{state}", track.State.ToString())
|
||||||
.Replace("{extractor}", Config.inputType.ToString())
|
.Replace("{extractor}", Config.I.inputType.ToString())
|
||||||
.Trim();
|
.Trim();
|
||||||
|
|
||||||
if (onComplete[0] == '"')
|
if (onComplete[0] == '"')
|
||||||
|
|
|
@ -15,19 +15,38 @@ using SlFile = Soulseek.File;
|
||||||
using SlDictionary = System.Collections.Concurrent.ConcurrentDictionary<string, (Soulseek.SearchResponse, Soulseek.File)>;
|
using SlDictionary = System.Collections.Concurrent.ConcurrentDictionary<string, (Soulseek.SearchResponse, Soulseek.File)>;
|
||||||
|
|
||||||
|
|
||||||
|
using System.Diagnostics;
|
||||||
|
|
||||||
|
class TimerReporter
|
||||||
|
{
|
||||||
|
private Stopwatch stopwatch;
|
||||||
|
|
||||||
|
public TimerReporter()
|
||||||
|
{
|
||||||
|
stopwatch = new Stopwatch();
|
||||||
|
stopwatch.Start();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Report(string message = "")
|
||||||
|
{
|
||||||
|
Console.WriteLine($"Time elapsed: {stopwatch.ElapsedMilliseconds} ms. {message}");
|
||||||
|
stopwatch.Restart();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static class Search
|
static class Search
|
||||||
{
|
{
|
||||||
public static RateLimitedSemaphore? searchSemaphore;
|
public static RateLimitedSemaphore? searchSemaphore;
|
||||||
|
|
||||||
// very messy function that does everything
|
// very messy function that does everything
|
||||||
public static async Task<(string, SlFile?)> SearchAndDownload(Track track, FileManager organizer, CancellationTokenSource cts)
|
public static async Task<(string, SlFile?)> SearchAndDownload(Track track, FileManager organizer, CancellationTokenSource? cts = null)
|
||||||
{
|
{
|
||||||
if (Config.DoNotDownload)
|
if (Config.I.DoNotDownload)
|
||||||
throw new Exception();
|
throw new Exception();
|
||||||
|
|
||||||
IEnumerable<(SlResponse response, SlFile file)>? orderedResults = null;
|
IEnumerable<(SlResponse response, SlFile file)>? orderedResults = null;
|
||||||
var responseData = new ResponseData();
|
var responseData = new ResponseData();
|
||||||
var progress = Printing.GetProgressBar(Config.displayMode);
|
var progress = Printing.GetProgressBar(Config.I.displayMode);
|
||||||
var results = new SlDictionary();
|
var results = new SlDictionary();
|
||||||
var fsResults = new SlDictionary();
|
var fsResults = new SlDictionary();
|
||||||
using var searchCts = new CancellationTokenSource();
|
using var searchCts = new CancellationTokenSource();
|
||||||
|
@ -62,7 +81,7 @@ static class Search
|
||||||
saveFilePath = organizer.GetSavePath(f.Filename);
|
saveFilePath = organizer.GetSavePath(f.Filename);
|
||||||
fsUser = r.Username;
|
fsUser = r.Username;
|
||||||
chosenFile = f;
|
chosenFile = f;
|
||||||
downloadTask = Download.DownloadFile(r, f, saveFilePath, track, progress, cts, searchCts);
|
downloadTask = Download.DownloadFile(r, f, saveFilePath, track, progress, cts?.Token, searchCts);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -76,17 +95,17 @@ static class Search
|
||||||
foreach (var file in r.Files)
|
foreach (var file in r.Files)
|
||||||
results.TryAdd(r.Username + '\\' + file.Filename, (r, file));
|
results.TryAdd(r.Username + '\\' + file.Filename, (r, file));
|
||||||
|
|
||||||
if (Config.fastSearch && userSuccessCount.GetValueOrDefault(r.Username, 0) > Config.downrankOn)
|
if (Config.I.fastSearch && userSuccessCount.GetValueOrDefault(r.Username, 0) > Config.I.downrankOn)
|
||||||
{
|
{
|
||||||
var f = r.Files.First();
|
var f = r.Files.First();
|
||||||
|
|
||||||
if (r.HasFreeUploadSlot && r.UploadSpeed / 1024.0 / 1024.0 >= Config.fastSearchMinUpSpeed
|
if (r.HasFreeUploadSlot && r.UploadSpeed / 1024.0 / 1024.0 >= Config.I.fastSearchMinUpSpeed
|
||||||
&& FileConditions.BracketCheck(track, InferTrack(f.Filename, track)) && Config.preferredCond.FileSatisfies(f, track, r))
|
&& FileConditions.BracketCheck(track, InferTrack(f.Filename, track)) && Config.I.preferredCond.FileSatisfies(f, track, r))
|
||||||
{
|
{
|
||||||
fsResults.TryAdd(r.Username + '\\' + f.Filename, (r, f));
|
fsResults.TryAdd(r.Username + '\\' + f.Filename, (r, f));
|
||||||
if (Interlocked.Exchange(ref fsResultsStarted, 1) == 0)
|
if (Interlocked.Exchange(ref fsResultsStarted, 1) == 0)
|
||||||
{
|
{
|
||||||
Task.Delay(Config.fastSearchDelay).ContinueWith(tt => fastSearchDownload());
|
Task.Delay(Config.I.fastSearchDelay).ContinueWith(tt => fastSearchDownload());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -98,8 +117,8 @@ static class Search
|
||||||
return new SearchOptions(
|
return new SearchOptions(
|
||||||
minimumResponseFileCount: 1,
|
minimumResponseFileCount: 1,
|
||||||
minimumPeerUploadSpeed: 1,
|
minimumPeerUploadSpeed: 1,
|
||||||
searchTimeout: Config.searchTimeout,
|
searchTimeout: Config.I.searchTimeout,
|
||||||
removeSingleCharacterSearchTerms: Config.removeSingleCharacterSearchTerms,
|
removeSingleCharacterSearchTerms: Config.I.removeSingleCharacterSearchTerms,
|
||||||
responseFilter: (response) =>
|
responseFilter: (response) =>
|
||||||
{
|
{
|
||||||
return response.UploadSpeed > 0 && necCond.BannedUsersSatisfies(response);
|
return response.UploadSpeed > 0 && necCond.BannedUsersSatisfies(response);
|
||||||
|
@ -117,7 +136,7 @@ static class Search
|
||||||
searchEnded = true;
|
searchEnded = true;
|
||||||
lock (fsDownloadLock) { }
|
lock (fsDownloadLock) { }
|
||||||
|
|
||||||
if (downloading == 0 && results.IsEmpty && !Config.useYtdlp)
|
if (downloading == 0 && results.IsEmpty && !Config.I.useYtdlp)
|
||||||
{
|
{
|
||||||
notFound = true;
|
notFound = true;
|
||||||
}
|
}
|
||||||
|
@ -151,7 +170,7 @@ static class Search
|
||||||
if (orderedResults == null)
|
if (orderedResults == null)
|
||||||
orderedResults = OrderedResults(results, track, useInfer: true);
|
orderedResults = OrderedResults(results, track, useInfer: true);
|
||||||
|
|
||||||
int trackTries = Config.maxRetriesPerTrack;
|
int trackTries = Config.I.maxRetriesPerTrack;
|
||||||
async Task<bool> process(SlResponse response, SlFile file)
|
async Task<bool> process(SlResponse response, SlFile file)
|
||||||
{
|
{
|
||||||
saveFilePath = organizer.GetSavePath(file.Filename);
|
saveFilePath = organizer.GetSavePath(file.Filename);
|
||||||
|
@ -159,7 +178,7 @@ static class Search
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
downloading = 1;
|
downloading = 1;
|
||||||
await Download.DownloadFile(response, file, saveFilePath, track, progress, cts);
|
await Download.DownloadFile(response, file, saveFilePath, track, progress, cts?.Token);
|
||||||
userSuccessCount.AddOrUpdate(response.Username, 1, (k, v) => v + 1);
|
userSuccessCount.AddOrUpdate(response.Username, 1, (k, v) => v + 1);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -172,7 +191,7 @@ static class Search
|
||||||
if (!IsConnectedAndLoggedIn())
|
if (!IsConnectedAndLoggedIn())
|
||||||
throw;
|
throw;
|
||||||
|
|
||||||
Printing.WriteLine("Error: " + e.Message, ConsoleColor.DarkYellow, true);
|
Printing.WriteLine("Error: Download Error: " + e.Message, ConsoleColor.DarkYellow, debugOnly: true);
|
||||||
|
|
||||||
userSuccessCount.AddOrUpdate(response.Username, -1, (k, v) => v - 1);
|
userSuccessCount.AddOrUpdate(response.Username, -1, (k, v) => v - 1);
|
||||||
if (--trackTries <= 0)
|
if (--trackTries <= 0)
|
||||||
|
@ -194,7 +213,7 @@ static class Search
|
||||||
fr = orderedResults.Skip(1).FirstOrDefault();
|
fr = orderedResults.Skip(1).FirstOrDefault();
|
||||||
if (fr != default)
|
if (fr != default)
|
||||||
{
|
{
|
||||||
if (userSuccessCount.GetValueOrDefault(fr.response.Username, 0) > Config.ignoreOn)
|
if (userSuccessCount.GetValueOrDefault(fr.response.Username, 0) > Config.I.ignoreOn)
|
||||||
{
|
{
|
||||||
success = await process(fr.response, fr.file);
|
success = await process(fr.response, fr.file);
|
||||||
}
|
}
|
||||||
|
@ -202,7 +221,7 @@ static class Search
|
||||||
{
|
{
|
||||||
foreach (var (response, file) in orderedResults.Skip(2))
|
foreach (var (response, file) in orderedResults.Skip(2))
|
||||||
{
|
{
|
||||||
if (userSuccessCount.GetValueOrDefault(response.Username, 0) <= Config.ignoreOn)
|
if (userSuccessCount.GetValueOrDefault(response.Username, 0) <= Config.I.ignoreOn)
|
||||||
continue;
|
continue;
|
||||||
success = await process(response, file);
|
success = await process(response, file);
|
||||||
if (success) break;
|
if (success) break;
|
||||||
|
@ -212,7 +231,7 @@ static class Search
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (downloading == 0 && Config.useYtdlp)
|
if (downloading == 0 && Config.I.useYtdlp)
|
||||||
{
|
{
|
||||||
notFound = false;
|
notFound = false;
|
||||||
try
|
try
|
||||||
|
@ -224,12 +243,12 @@ static class Search
|
||||||
{
|
{
|
||||||
foreach (var (length, id, title) in ytResults)
|
foreach (var (length, id, title) in ytResults)
|
||||||
{
|
{
|
||||||
if (Config.necessaryCond.LengthToleranceSatisfies(length, track.Length))
|
if (Config.I.necessaryCond.LengthToleranceSatisfies(length, track.Length))
|
||||||
{
|
{
|
||||||
string saveFilePathNoExt = organizer.GetSavePathNoExt(title);
|
string saveFilePathNoExt = organizer.GetSavePathNoExt(title);
|
||||||
downloading = 1;
|
downloading = 1;
|
||||||
Printing.RefreshOrPrint(progress, 0, $"yt-dlp download: {track}", true);
|
Printing.RefreshOrPrint(progress, 0, $"yt-dlp download: {track}", true);
|
||||||
saveFilePath = await Extractors.YouTube.YtdlpDownload(id, saveFilePathNoExt, Config.ytdlpArgument);
|
saveFilePath = await Extractors.YouTube.YtdlpDownload(id, saveFilePathNoExt, Config.I.ytdlpArgument);
|
||||||
Printing.RefreshOrPrint(progress, 100, $"Succeded: yt-dlp completed download for {track}", true);
|
Printing.RefreshOrPrint(progress, 100, $"Succeded: yt-dlp completed download for {track}", true);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -271,7 +290,7 @@ static class Search
|
||||||
new SearchOptions(
|
new SearchOptions(
|
||||||
minimumResponseFileCount: 1,
|
minimumResponseFileCount: 1,
|
||||||
minimumPeerUploadSpeed: 1,
|
minimumPeerUploadSpeed: 1,
|
||||||
removeSingleCharacterSearchTerms: Config.removeSingleCharacterSearchTerms,
|
removeSingleCharacterSearchTerms: Config.I.removeSingleCharacterSearchTerms,
|
||||||
searchTimeout: timeout,
|
searchTimeout: timeout,
|
||||||
responseFilter: (response) =>
|
responseFilter: (response) =>
|
||||||
{
|
{
|
||||||
|
@ -354,10 +373,10 @@ static class Search
|
||||||
}
|
}
|
||||||
|
|
||||||
int min, max;
|
int min, max;
|
||||||
if (Config.minAlbumTrackCount > -1 || Config.maxAlbumTrackCount > -1)
|
if (Config.I.minAlbumTrackCount > -1 || Config.I.maxAlbumTrackCount > -1)
|
||||||
{
|
{
|
||||||
min = Config.minAlbumTrackCount;
|
min = Config.I.minAlbumTrackCount;
|
||||||
max = Config.maxAlbumTrackCount;
|
max = Config.I.maxAlbumTrackCount;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -410,7 +429,7 @@ static class Search
|
||||||
new(
|
new(
|
||||||
minimumResponseFileCount: 1,
|
minimumResponseFileCount: 1,
|
||||||
minimumPeerUploadSpeed: 1,
|
minimumPeerUploadSpeed: 1,
|
||||||
removeSingleCharacterSearchTerms: Config.removeSingleCharacterSearchTerms,
|
removeSingleCharacterSearchTerms: Config.I.removeSingleCharacterSearchTerms,
|
||||||
searchTimeout: timeout,
|
searchTimeout: timeout,
|
||||||
responseFilter: (response) =>
|
responseFilter: (response) =>
|
||||||
{
|
{
|
||||||
|
@ -442,7 +461,7 @@ static class Search
|
||||||
var equivalentFiles = EquivalentFiles(track, results.Select(x => x.Value))
|
var equivalentFiles = EquivalentFiles(track, results.Select(x => x.Value))
|
||||||
.Select(x => (x.Item1, OrderedResults(x.Item2, track, false, false, false))).ToList();
|
.Select(x => (x.Item1, OrderedResults(x.Item2, track, false, false, false))).ToList();
|
||||||
|
|
||||||
if (!Config.relax)
|
if (!Config.I.relax)
|
||||||
{
|
{
|
||||||
equivalentFiles = equivalentFiles
|
equivalentFiles = equivalentFiles
|
||||||
.Where(x => FileConditions.StrictString(x.Item1.Title, track.Title, ignoreCase: true)
|
.Where(x => FileConditions.StrictString(x.Item1.Title, track.Title, ignoreCase: true)
|
||||||
|
@ -458,14 +477,14 @@ static class Search
|
||||||
kvp.Item1.Downloads = kvp.Item2.ToList();
|
kvp.Item1.Downloads = kvp.Item2.ToList();
|
||||||
return kvp.Item1;
|
return kvp.Item1;
|
||||||
}).ToList();
|
}).ToList();
|
||||||
|
|
||||||
return tracks;
|
return tracks;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public static async Task<List<List<List<Track>>>> GetAggregateAlbums(Track track, ResponseData responseData)
|
public static async Task<List<List<List<Track>>>> GetAggregateAlbums(Track track, ResponseData responseData)
|
||||||
{
|
{
|
||||||
int maxDiff = Config.aggregateLengthTol;
|
int maxDiff = Config.I.aggregateLengthTol;
|
||||||
|
|
||||||
bool lengthsAreSimilar(int[] sorted1, int[] sorted2)
|
bool lengthsAreSimilar(int[] sorted1, int[] sorted2)
|
||||||
{
|
{
|
||||||
|
@ -547,7 +566,7 @@ static class Search
|
||||||
}
|
}
|
||||||
|
|
||||||
res = res.Select((x, i) => (x, i))
|
res = res.Select((x, i) => (x, i))
|
||||||
.Where(x => usernamesList[x.i].Count >= Config.minSharesAggregate)
|
.Where(x => usernamesList[x.i].Count >= Config.I.minSharesAggregate)
|
||||||
.OrderByDescending(x => usernamesList[x.i].Count)
|
.OrderByDescending(x => usernamesList[x.i].Count)
|
||||||
.Select(x => x.x)
|
.Select(x => x.x)
|
||||||
.ToList();
|
.ToList();
|
||||||
|
@ -619,7 +638,7 @@ static class Search
|
||||||
IEnumerable<(SlResponse, SlFile)> fileResponses, int minShares = -1)
|
IEnumerable<(SlResponse, SlFile)> fileResponses, int minShares = -1)
|
||||||
{
|
{
|
||||||
if (minShares == -1)
|
if (minShares == -1)
|
||||||
minShares = Config.minSharesAggregate;
|
minShares = Config.I.minSharesAggregate;
|
||||||
|
|
||||||
Track inferTrack((SearchResponse r, Soulseek.File f) x)
|
Track inferTrack((SearchResponse r, Soulseek.File f) x)
|
||||||
{
|
{
|
||||||
|
@ -629,7 +648,7 @@ static class Search
|
||||||
}
|
}
|
||||||
|
|
||||||
var groups = fileResponses
|
var groups = fileResponses
|
||||||
.GroupBy(inferTrack, new TrackComparer(ignoreCase: true, Config.aggregateLengthTol))
|
.GroupBy(inferTrack, new TrackComparer(ignoreCase: true, Config.I.aggregateLengthTol))
|
||||||
.Select(x => (x, x.Select(y => y.Item1.Username).Distinct().Count()))
|
.Select(x => (x, x.Select(y => y.Item1.Username).Distinct().Count()))
|
||||||
.Where(x => x.Item2 >= minShares)
|
.Where(x => x.Item2 >= minShares)
|
||||||
.OrderByDescending(x => x.Item2)
|
.OrderByDescending(x => x.Item2)
|
||||||
|
@ -691,22 +710,22 @@ static class Search
|
||||||
|
|
||||||
var random = new Random();
|
var random = new Random();
|
||||||
return results.Select(x => (response: x.Item1, file: x.Item2))
|
return results.Select(x => (response: x.Item1, file: x.Item2))
|
||||||
.Where(x => userSuccessCount.GetValueOrDefault(x.response.Username, 0) > Config.ignoreOn)
|
.Where(x => userSuccessCount.GetValueOrDefault(x.response.Username, 0) > Config.I.ignoreOn)
|
||||||
.OrderByDescending(x => userSuccessCount.GetValueOrDefault(x.response.Username, 0) > Config.downrankOn)
|
.OrderByDescending(x => userSuccessCount.GetValueOrDefault(x.response.Username, 0) > Config.I.downrankOn)
|
||||||
.ThenByDescending(x => Config.necessaryCond.FileSatisfies(x.file, track, x.response))
|
.ThenByDescending(x => Config.I.necessaryCond.FileSatisfies(x.file, track, x.response))
|
||||||
.ThenByDescending(x => Config.preferredCond.BannedUsersSatisfies(x.response))
|
.ThenByDescending(x => Config.I.preferredCond.BannedUsersSatisfies(x.response))
|
||||||
.ThenByDescending(x => (x.file.Length != null && x.file.Length > 0) || Config.preferredCond.AcceptNoLength)
|
.ThenByDescending(x => (x.file.Length != null && x.file.Length > 0) || Config.I.preferredCond.AcceptNoLength)
|
||||||
.ThenByDescending(x => !useBracketCheck || FileConditions.BracketCheck(track, inferredTrack(x).Item1)) // downrank result if it contains '(' or '[' and the title does not (avoid remixes)
|
.ThenByDescending(x => !useBracketCheck || FileConditions.BracketCheck(track, inferredTrack(x).Item1)) // downrank result if it contains '(' or '[' and the title does not (avoid remixes)
|
||||||
.ThenByDescending(x => Config.preferredCond.StrictTitleSatisfies(x.file.Filename, track.Title))
|
.ThenByDescending(x => Config.I.preferredCond.StrictTitleSatisfies(x.file.Filename, track.Title))
|
||||||
.ThenByDescending(x => !albumMode || Config.preferredCond.StrictAlbumSatisfies(x.file.Filename, track.Album))
|
.ThenByDescending(x => !albumMode || Config.I.preferredCond.StrictAlbumSatisfies(x.file.Filename, track.Album))
|
||||||
.ThenByDescending(x => Config.preferredCond.StrictArtistSatisfies(x.file.Filename, track.Title))
|
.ThenByDescending(x => Config.I.preferredCond.StrictArtistSatisfies(x.file.Filename, track.Title))
|
||||||
.ThenByDescending(x => Config.preferredCond.LengthToleranceSatisfies(x.file, track.Length))
|
.ThenByDescending(x => Config.I.preferredCond.LengthToleranceSatisfies(x.file, track.Length))
|
||||||
.ThenByDescending(x => Config.preferredCond.FormatSatisfies(x.file.Filename))
|
.ThenByDescending(x => Config.I.preferredCond.FormatSatisfies(x.file.Filename))
|
||||||
.ThenByDescending(x => albumMode || Config.preferredCond.StrictAlbumSatisfies(x.file.Filename, track.Album))
|
.ThenByDescending(x => albumMode || Config.I.preferredCond.StrictAlbumSatisfies(x.file.Filename, track.Album))
|
||||||
.ThenByDescending(x => Config.preferredCond.BitrateSatisfies(x.file))
|
.ThenByDescending(x => Config.I.preferredCond.BitrateSatisfies(x.file))
|
||||||
.ThenByDescending(x => Config.preferredCond.SampleRateSatisfies(x.file))
|
.ThenByDescending(x => Config.I.preferredCond.SampleRateSatisfies(x.file))
|
||||||
.ThenByDescending(x => Config.preferredCond.BitDepthSatisfies(x.file))
|
.ThenByDescending(x => Config.I.preferredCond.BitDepthSatisfies(x.file))
|
||||||
.ThenByDescending(x => Config.preferredCond.FileSatisfies(x.file, track, x.response))
|
.ThenByDescending(x => Config.I.preferredCond.FileSatisfies(x.file, track, x.response))
|
||||||
.ThenByDescending(x => x.response.HasFreeUploadSlot)
|
.ThenByDescending(x => x.response.HasFreeUploadSlot)
|
||||||
.ThenByDescending(x => x.response.UploadSpeed / 1024 / 650)
|
.ThenByDescending(x => x.response.UploadSpeed / 1024 / 650)
|
||||||
.ThenByDescending(x => albumMode || FileConditions.StrictString(x.file.Filename, track.Title))
|
.ThenByDescending(x => albumMode || FileConditions.StrictString(x.file.Filename, track.Title))
|
||||||
|
@ -730,7 +749,7 @@ static class Search
|
||||||
string search = GetSearchString(track);
|
string search = GetSearchString(track);
|
||||||
var searchTasks = new List<Task>();
|
var searchTasks = new List<Task>();
|
||||||
|
|
||||||
var defaultSearchOpts = getSearchOptions(Config.searchTimeout, Config.necessaryCond, Config.preferredCond);
|
var defaultSearchOpts = getSearchOptions(Config.I.searchTimeout, Config.I.necessaryCond, Config.I.preferredCond);
|
||||||
searchTasks.Add(DoSearch(search, defaultSearchOpts, responseHandler, ct, onSearch));
|
searchTasks.Add(DoSearch(search, defaultSearchOpts, responseHandler, ct, onSearch));
|
||||||
|
|
||||||
if (search.RemoveDiacriticsIfExist(out string noDiacrSearch) && !track.ArtistMaybeWrong)
|
if (search.RemoveDiacriticsIfExist(out string noDiacrSearch) && !track.ArtistMaybeWrong)
|
||||||
|
@ -742,15 +761,15 @@ static class Search
|
||||||
|
|
||||||
if (results.IsEmpty && track.ArtistMaybeWrong && title)
|
if (results.IsEmpty && track.ArtistMaybeWrong && title)
|
||||||
{
|
{
|
||||||
var cond = new FileConditions(Config.necessaryCond);
|
var cond = new FileConditions(Config.I.necessaryCond);
|
||||||
var infTrack = InferTrack(track.Title, new Track());
|
var infTrack = InferTrack(track.Title, new Track());
|
||||||
cond.StrictTitle = infTrack.Title == track.Title;
|
cond.StrictTitle = infTrack.Title == track.Title;
|
||||||
cond.StrictArtist = false;
|
cond.StrictArtist = false;
|
||||||
var opts = getSearchOptions(Math.Min(Config.searchTimeout, 5000), cond, Config.preferredCond);
|
var opts = getSearchOptions(Math.Min(Config.I.searchTimeout, 5000), cond, Config.I.preferredCond);
|
||||||
searchTasks.Add(DoSearch($"{infTrack.Artist} {infTrack.Title}", opts, responseHandler, ct, onSearch));
|
searchTasks.Add(DoSearch($"{infTrack.Artist} {infTrack.Title}", opts, responseHandler, ct, onSearch));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Config.desperateSearch)
|
if (Config.I.desperateSearch)
|
||||||
{
|
{
|
||||||
await Task.WhenAll(searchTasks);
|
await Task.WhenAll(searchTasks);
|
||||||
|
|
||||||
|
@ -758,23 +777,23 @@ static class Search
|
||||||
{
|
{
|
||||||
if (artist && album && title)
|
if (artist && album && title)
|
||||||
{
|
{
|
||||||
var cond = new FileConditions(Config.necessaryCond)
|
var cond = new FileConditions(Config.I.necessaryCond)
|
||||||
{
|
{
|
||||||
StrictTitle = true,
|
StrictTitle = true,
|
||||||
StrictAlbum = true
|
StrictAlbum = true
|
||||||
};
|
};
|
||||||
var opts = getSearchOptions(Math.Min(Config.searchTimeout, 5000), cond, Config.preferredCond);
|
var opts = getSearchOptions(Math.Min(Config.I.searchTimeout, 5000), cond, Config.I.preferredCond);
|
||||||
searchTasks.Add(DoSearch($"{track.Artist} {track.Album}", opts, responseHandler, ct, onSearch));
|
searchTasks.Add(DoSearch($"{track.Artist} {track.Album}", opts, responseHandler, ct, onSearch));
|
||||||
}
|
}
|
||||||
if (artist && title && track.Length != -1 && Config.necessaryCond.LengthTolerance != -1)
|
if (artist && title && track.Length != -1 && Config.I.necessaryCond.LengthTolerance != -1)
|
||||||
{
|
{
|
||||||
var cond = new FileConditions(Config.necessaryCond)
|
var cond = new FileConditions(Config.I.necessaryCond)
|
||||||
{
|
{
|
||||||
LengthTolerance = -1,
|
LengthTolerance = -1,
|
||||||
StrictTitle = true,
|
StrictTitle = true,
|
||||||
StrictArtist = true
|
StrictArtist = true
|
||||||
};
|
};
|
||||||
var opts = getSearchOptions(Math.Min(Config.searchTimeout, 5000), cond, Config.preferredCond);
|
var opts = getSearchOptions(Math.Min(Config.I.searchTimeout, 5000), cond, Config.I.preferredCond);
|
||||||
searchTasks.Add(DoSearch($"{track.Artist} {track.Title}", opts, responseHandler, ct, onSearch));
|
searchTasks.Add(DoSearch($"{track.Artist} {track.Title}", opts, responseHandler, ct, onSearch));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -787,36 +806,36 @@ static class Search
|
||||||
|
|
||||||
if (track.Album.Length > 3 && album)
|
if (track.Album.Length > 3 && album)
|
||||||
{
|
{
|
||||||
var cond = new FileConditions(Config.necessaryCond)
|
var cond = new FileConditions(Config.I.necessaryCond)
|
||||||
{
|
{
|
||||||
StrictAlbum = true,
|
StrictAlbum = true,
|
||||||
StrictTitle = !track.ArtistMaybeWrong,
|
StrictTitle = !track.ArtistMaybeWrong,
|
||||||
StrictArtist = !track.ArtistMaybeWrong,
|
StrictArtist = !track.ArtistMaybeWrong,
|
||||||
LengthTolerance = -1
|
LengthTolerance = -1
|
||||||
};
|
};
|
||||||
var opts = getSearchOptions(Math.Min(Config.searchTimeout, 5000), cond, Config.preferredCond);
|
var opts = getSearchOptions(Math.Min(Config.I.searchTimeout, 5000), cond, Config.I.preferredCond);
|
||||||
searchTasks.Add(DoSearch($"{track.Album}", opts, responseHandler, ct, onSearch));
|
searchTasks.Add(DoSearch($"{track.Album}", opts, responseHandler, ct, onSearch));
|
||||||
}
|
}
|
||||||
if (track2.Title.Length > 3 && artist)
|
if (track2.Title.Length > 3 && artist)
|
||||||
{
|
{
|
||||||
var cond = new FileConditions(Config.necessaryCond)
|
var cond = new FileConditions(Config.I.necessaryCond)
|
||||||
{
|
{
|
||||||
StrictTitle = !track.ArtistMaybeWrong,
|
StrictTitle = !track.ArtistMaybeWrong,
|
||||||
StrictArtist = !track.ArtistMaybeWrong,
|
StrictArtist = !track.ArtistMaybeWrong,
|
||||||
LengthTolerance = -1
|
LengthTolerance = -1
|
||||||
};
|
};
|
||||||
var opts = getSearchOptions(Math.Min(Config.searchTimeout, 5000), cond, Config.preferredCond);
|
var opts = getSearchOptions(Math.Min(Config.I.searchTimeout, 5000), cond, Config.I.preferredCond);
|
||||||
searchTasks.Add(DoSearch($"{track2.Title}", opts, responseHandler, ct, onSearch));
|
searchTasks.Add(DoSearch($"{track2.Title}", opts, responseHandler, ct, onSearch));
|
||||||
}
|
}
|
||||||
if (track2.Artist.Length > 3 && title)
|
if (track2.Artist.Length > 3 && title)
|
||||||
{
|
{
|
||||||
var cond = new FileConditions(Config.necessaryCond)
|
var cond = new FileConditions(Config.I.necessaryCond)
|
||||||
{
|
{
|
||||||
StrictTitle = !track.ArtistMaybeWrong,
|
StrictTitle = !track.ArtistMaybeWrong,
|
||||||
StrictArtist = !track.ArtistMaybeWrong,
|
StrictArtist = !track.ArtistMaybeWrong,
|
||||||
LengthTolerance = -1
|
LengthTolerance = -1
|
||||||
};
|
};
|
||||||
var opts = getSearchOptions(Math.Min(Config.searchTimeout, 5000), cond, Config.preferredCond);
|
var opts = getSearchOptions(Math.Min(Config.I.searchTimeout, 5000), cond, Config.I.preferredCond);
|
||||||
searchTasks.Add(DoSearch($"{track2.Artist}", opts, responseHandler, ct, onSearch));
|
searchTasks.Add(DoSearch($"{track2.Artist}", opts, responseHandler, ct, onSearch));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -851,15 +870,15 @@ static class Search
|
||||||
return new SearchOptions(
|
return new SearchOptions(
|
||||||
minimumResponseFileCount: 1,
|
minimumResponseFileCount: 1,
|
||||||
minimumPeerUploadSpeed: 1,
|
minimumPeerUploadSpeed: 1,
|
||||||
searchTimeout: Config.searchTimeout,
|
searchTimeout: Config.I.searchTimeout,
|
||||||
removeSingleCharacterSearchTerms: Config.removeSingleCharacterSearchTerms,
|
removeSingleCharacterSearchTerms: Config.I.removeSingleCharacterSearchTerms,
|
||||||
responseFilter: (response) =>
|
responseFilter: (response) =>
|
||||||
{
|
{
|
||||||
return response.UploadSpeed > 0 && necCond.BannedUsersSatisfies(response);
|
return response.UploadSpeed > 0 && necCond.BannedUsersSatisfies(response);
|
||||||
},
|
},
|
||||||
fileFilter: (file) =>
|
fileFilter: (file) =>
|
||||||
{
|
{
|
||||||
return Utils.IsMusicFile(file.Filename) && (necCond.FileSatisfies(file, track, null) || Config.PrintResultsFull);
|
return Utils.IsMusicFile(file.Filename) && (necCond.FileSatisfies(file, track, null) || Config.I.PrintResultsFull);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -876,7 +895,7 @@ static class Search
|
||||||
|
|
||||||
await RunSearches(track, results, getSearchOptions, responseHandler);
|
await RunSearches(track, results, getSearchOptions, responseHandler);
|
||||||
|
|
||||||
if (Config.DoNotDownload && results.IsEmpty)
|
if (Config.I.DoNotDownload && results.IsEmpty)
|
||||||
{
|
{
|
||||||
Printing.WriteLine($"No results", ConsoleColor.Yellow);
|
Printing.WriteLine($"No results", ConsoleColor.Yellow);
|
||||||
}
|
}
|
||||||
|
@ -888,8 +907,8 @@ static class Search
|
||||||
foreach (var (response, file) in orderedResults)
|
foreach (var (response, file) in orderedResults)
|
||||||
{
|
{
|
||||||
Console.WriteLine(Printing.DisplayString(track, file, response,
|
Console.WriteLine(Printing.DisplayString(track, file, response,
|
||||||
Config.PrintResultsFull ? Config.necessaryCond : null, Config.PrintResultsFull ? Config.preferredCond : null,
|
Config.I.PrintResultsFull ? Config.I.necessaryCond : null, Config.I.PrintResultsFull ? Config.I.preferredCond : null,
|
||||||
fullpath: Config.PrintResultsFull, infoFirst: true, showSpeed: Config.PrintResultsFull));
|
fullpath: Config.I.PrintResultsFull, infoFirst: true, showSpeed: Config.I.PrintResultsFull));
|
||||||
count += 1;
|
count += 1;
|
||||||
}
|
}
|
||||||
Printing.WriteLine($"Total: {count}\n", ConsoleColor.Yellow);
|
Printing.WriteLine($"Total: {count}\n", ConsoleColor.Yellow);
|
||||||
|
@ -924,7 +943,7 @@ static class Search
|
||||||
static string CleanSearchString(string str)
|
static string CleanSearchString(string str)
|
||||||
{
|
{
|
||||||
string old;
|
string old;
|
||||||
if (!Config.noRemoveSpecialChars)
|
if (!Config.I.noRemoveSpecialChars)
|
||||||
{
|
{
|
||||||
old = str;
|
old = str;
|
||||||
str = str.ReplaceSpecialChars(" ").Trim().RemoveConsecutiveWs();
|
str = str.ReplaceSpecialChars(" ").Trim().RemoveConsecutiveWs();
|
||||||
|
|
|
@ -79,15 +79,13 @@ namespace Test
|
||||||
{
|
{
|
||||||
SetCurrentTest("TestAutoProfiles");
|
SetCurrentTest("TestAutoProfiles");
|
||||||
|
|
||||||
ResetProfiles();
|
ResetConfig();
|
||||||
Config.inputType = InputType.YouTube;
|
Config.I.inputType = InputType.YouTube;
|
||||||
Config.interactiveMode = true;
|
Config.I.interactiveMode = true;
|
||||||
Config.album = true;
|
Config.I.aggregate = false;
|
||||||
Config.aggregate = false;
|
Config.I.maxStaleTime = 50000;
|
||||||
Config.maxStaleTime = 500000;
|
|
||||||
|
|
||||||
string path = Path.Join(Directory.GetCurrentDirectory(), "test_conf.conf");
|
string path = Path.Join(Directory.GetCurrentDirectory(), "test_conf.conf");
|
||||||
Config.confPath = path;
|
|
||||||
|
|
||||||
string content =
|
string content =
|
||||||
"max-stale-time = 5" +
|
"max-stale-time = 5" +
|
||||||
|
@ -111,18 +109,19 @@ namespace Test
|
||||||
|
|
||||||
File.WriteAllText(path, content);
|
File.WriteAllText(path, content);
|
||||||
|
|
||||||
Config.ParseArgsAndReadConfig(new string[] { });
|
Config.I.Load(new string[] { "-c", path });
|
||||||
|
|
||||||
//Config.PostProcessArgs();
|
var tle = new TrackListEntry(TrackType.Album);
|
||||||
|
Config.UpdateProfiles(tle);
|
||||||
|
|
||||||
Assert(Config.maxStaleTime == 10 && !Config.fastSearch && Config.necessaryCond.Formats[0] == "flac");
|
Assert(Config.I.maxStaleTime == 10 && !Config.I.fastSearch && Config.I.necessaryCond.Formats[0] == "flac");
|
||||||
|
|
||||||
ResetProfiles();
|
ResetConfig();
|
||||||
Config.inputType = InputType.CSV;
|
Config.I.inputType = InputType.CSV;
|
||||||
Config.album = true;
|
Config.I.album = true;
|
||||||
Config.interactiveMode = true;
|
Config.I.interactiveMode = true;
|
||||||
Config.useYtdlp = false;
|
Config.I.useYtdlp = false;
|
||||||
Config.maxStaleTime = 50000;
|
Config.I.maxStaleTime = 50000;
|
||||||
content =
|
content =
|
||||||
"\n[no-stale]" +
|
"\n[no-stale]" +
|
||||||
"\nprofile-cond = interactive && download-mode == \"album\"" +
|
"\nprofile-cond = interactive && download-mode == \"album\"" +
|
||||||
|
@ -133,16 +132,17 @@ namespace Test
|
||||||
|
|
||||||
File.WriteAllText(path, content);
|
File.WriteAllText(path, content);
|
||||||
|
|
||||||
Config.ParseArgsAndReadConfig(new string[] { });
|
|
||||||
|
Config.I.Load(new string[] { "-c", path });
|
||||||
|
Config.UpdateProfiles(tle);
|
||||||
|
Assert(Config.I.maxStaleTime == 999999 && !Config.I.useYtdlp);
|
||||||
|
|
||||||
Assert(Config.maxStaleTime == 999999 && !Config.useYtdlp);
|
ResetConfig();
|
||||||
|
Config.I.inputType = InputType.YouTube;
|
||||||
ResetProfiles();
|
Config.I.album = false;
|
||||||
Config.inputType = InputType.YouTube;
|
Config.I.interactiveMode = true;
|
||||||
Config.album = false;
|
Config.I.useYtdlp = false;
|
||||||
Config.interactiveMode = true;
|
Config.I.maxStaleTime = 50000;
|
||||||
Config.useYtdlp = false;
|
|
||||||
Config.maxStaleTime = 50000;
|
|
||||||
content =
|
content =
|
||||||
"\n[no-stale]" +
|
"\n[no-stale]" +
|
||||||
"\nprofile-cond = interactive && download-mode == \"album\"" +
|
"\nprofile-cond = interactive && download-mode == \"album\"" +
|
||||||
|
@ -152,10 +152,10 @@ namespace Test
|
||||||
"\nyt-dlp = true";
|
"\nyt-dlp = true";
|
||||||
|
|
||||||
File.WriteAllText(path, content);
|
File.WriteAllText(path, content);
|
||||||
|
Config.I.Load(new string[] { "-c", path });
|
||||||
|
Config.UpdateProfiles(new TrackListEntry(TrackType.Normal));
|
||||||
|
|
||||||
Config.ParseArgsAndReadConfig(new string[] { });
|
Assert(Config.I.maxStaleTime == 50000 && Config.I.useYtdlp);
|
||||||
|
|
||||||
Assert(Config.maxStaleTime == 50000 && Config.useYtdlp);
|
|
||||||
|
|
||||||
if (File.Exists(path))
|
if (File.Exists(path))
|
||||||
File.Delete(path);
|
File.Delete(path);
|
||||||
|
@ -167,13 +167,15 @@ namespace Test
|
||||||
{
|
{
|
||||||
SetCurrentTest("TestProfileConditions");
|
SetCurrentTest("TestProfileConditions");
|
||||||
|
|
||||||
Config.inputType = InputType.YouTube;
|
Config.I.inputType = InputType.YouTube;
|
||||||
Config.interactiveMode = true;
|
Config.I.interactiveMode = true;
|
||||||
Config.album = true;
|
Config.I.album = true;
|
||||||
Config.aggregate = false;
|
Config.I.aggregate = false;
|
||||||
|
|
||||||
var conds = new (bool, string)[]
|
var conds = new (bool, string)[]
|
||||||
{
|
{
|
||||||
|
(true, "input-type == \"youtube\""),
|
||||||
|
(true, "download-mode == \"album\""),
|
||||||
(false, "aggregate"),
|
(false, "aggregate"),
|
||||||
(true, "interactive"),
|
(true, "interactive"),
|
||||||
(true, "album"),
|
(true, "album"),
|
||||||
|
@ -190,7 +192,7 @@ namespace Test
|
||||||
foreach ((var b, var c) in conds)
|
foreach ((var b, var c) in conds)
|
||||||
{
|
{
|
||||||
Console.WriteLine(c);
|
Console.WriteLine(c);
|
||||||
Assert(b == Config.ProfileConditionSatisfied(c));
|
Assert(b == Config.I.ProfileConditionSatisfied(c));
|
||||||
}
|
}
|
||||||
|
|
||||||
Passed();
|
Passed();
|
||||||
|
@ -247,29 +249,29 @@ namespace Test
|
||||||
|
|
||||||
var extractor = new Extractors.StringExtractor();
|
var extractor = new Extractors.StringExtractor();
|
||||||
|
|
||||||
Config.aggregate = false;
|
Config.I.aggregate = false;
|
||||||
Config.album = false;
|
Config.I.album = false;
|
||||||
|
|
||||||
Console.WriteLine("Testing songs: ");
|
Console.WriteLine("Testing songs: ");
|
||||||
for (int i = 0; i < strings.Count; i++)
|
for (int i = 0; i < strings.Count; i++)
|
||||||
{
|
{
|
||||||
Config.input = strings[i];
|
Config.I.input = strings[i];
|
||||||
Console.WriteLine(Config.input);
|
Console.WriteLine(Config.I.input);
|
||||||
var res = await extractor.GetTracks(Config.input, 0, 0, false);
|
var res = await extractor.GetTracks(Config.I.input, 0, 0, false);
|
||||||
var t = res[0].list[0][0];
|
var t = res[0].list[0][0];
|
||||||
Assert(Extractors.StringExtractor.InputMatches(Config.input));
|
Assert(Extractors.StringExtractor.InputMatches(Config.I.input));
|
||||||
Assert(t.ToKey() == tracks[i].ToKey());
|
Assert(t.ToKey() == tracks[i].ToKey());
|
||||||
}
|
}
|
||||||
|
|
||||||
Console.WriteLine();
|
Console.WriteLine();
|
||||||
Console.WriteLine("Testing albums");
|
Console.WriteLine("Testing albums");
|
||||||
Config.album = true;
|
Config.I.album = true;
|
||||||
for (int i = 0; i < strings.Count; i++)
|
for (int i = 0; i < strings.Count; i++)
|
||||||
{
|
{
|
||||||
Config.input = strings[i];
|
Config.I.input = strings[i];
|
||||||
Console.WriteLine(Config.input);
|
Console.WriteLine(Config.I.input);
|
||||||
var t = (await extractor.GetTracks(Config.input, 0, 0, false))[0].source;
|
var t = (await extractor.GetTracks(Config.I.input, 0, 0, false))[0].source;
|
||||||
Assert(Extractors.StringExtractor.InputMatches(Config.input));
|
Assert(Extractors.StringExtractor.InputMatches(Config.I.input));
|
||||||
Assert(t.ToKey() == albums[i].ToKey());
|
Assert(t.ToKey() == albums[i].ToKey());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -280,11 +282,11 @@ namespace Test
|
||||||
{
|
{
|
||||||
SetCurrentTest("TestM3uEditor");
|
SetCurrentTest("TestM3uEditor");
|
||||||
|
|
||||||
Config.m3uOption = M3uOption.All;
|
Config.I.m3uOption = M3uOption.All;
|
||||||
Config.skipMode = SkipMode.M3u;
|
Config.I.skipMode = SkipMode.M3u;
|
||||||
Config.musicDir = "";
|
Config.I.musicDir = "";
|
||||||
Config.printOption = PrintOption.Tracks | PrintOption.Full;
|
Config.I.printOption = PrintOption.Tracks | PrintOption.Full;
|
||||||
Config.skipExisting = true;
|
Config.I.skipExisting = true;
|
||||||
|
|
||||||
string path = Path.Join(Directory.GetCurrentDirectory(), "test_m3u.m3u8");
|
string path = Path.Join(Directory.GetCurrentDirectory(), "test_m3u.m3u8");
|
||||||
|
|
||||||
|
@ -325,7 +327,7 @@ namespace Test
|
||||||
foreach (var t in toBeDownloadedInitial)
|
foreach (var t in toBeDownloadedInitial)
|
||||||
trackLists.AddTrackToLast(t);
|
trackLists.AddTrackToLast(t);
|
||||||
|
|
||||||
Program.m3uEditor = new M3uEditor(path, trackLists, Config.m3uOption);
|
Program.m3uEditor = new M3uEditor(path, trackLists, Config.I.m3uOption);
|
||||||
|
|
||||||
Program.outputDirSkipper = new M3uSkipper(Program.m3uEditor, false);
|
Program.outputDirSkipper = new M3uSkipper(Program.m3uEditor, false);
|
||||||
|
|
||||||
|
@ -337,15 +339,15 @@ namespace Test
|
||||||
Assert(existing.SequenceEqualUpToPermutation(existingInitial));
|
Assert(existing.SequenceEqualUpToPermutation(existingInitial));
|
||||||
Assert(toBeDownloaded.SequenceEqualUpToPermutation(toBeDownloadedInitial));
|
Assert(toBeDownloaded.SequenceEqualUpToPermutation(toBeDownloadedInitial));
|
||||||
|
|
||||||
ProgramInvoke("PrintTracksTbd", new object[] { toBeDownloaded, existing, notFound, TrackType.Normal });
|
Printing.PrintTracksTbd(toBeDownloaded, existing, notFound, TrackType.Normal);
|
||||||
|
|
||||||
Program.m3uEditor.Update();
|
Program.m3uEditor.Update();
|
||||||
string output = File.ReadAllText(path);
|
string output = File.ReadAllText(path);
|
||||||
string need =
|
string need =
|
||||||
"#SLDL:./file1.5,\"Artist, 1.5\",,\"Title, , 1.5\",-1,0,3,0;path/to/file1,\"Artist, 1\",,\"Title, , 1\",-1,0,3,0;path/to/file2,\"Artist, 2\",,Title2,-1,0,3,0;,\"Artist; ,3\",,Title3 ;a,-1,0,4,0;,\"Artist,,, ;4\",,Title4,-1,0,4,3;,,,,-1,0,0,0;" +
|
"#SLDL:./file1.5,\"Artist, 1.5\",,\"Title, , 1.5\",-1,0,3,0;path/to/file1,\"Artist, 1\",,\"Title, , 1\",-1,0,3,0;path/to/file2,\"Artist, 2\",,Title2,-1,0,3,0;,\"Artist; ,3\",,Title3 ;a,-1,0,4,0;,\"Artist,,, ;4\",,Title4,-1,0,4,3;,,,,-1,0,0,0;" +
|
||||||
"\n" +
|
"\n" +
|
||||||
"\n# Failed: Artist; ,3 - Title3 ;a [NoSuitableFileFound]" +
|
"\n#FAIL: Artist; ,3 - Title3 ;a [NoSuitableFileFound]" +
|
||||||
"\n# Failed: Artist,,, ;4 - Title4 [NoSuitableFileFound]" +
|
"\n#FAIL: Artist,,, ;4 - Title4 [NoSuitableFileFound]" +
|
||||||
"\npath/to/file1" +
|
"\npath/to/file1" +
|
||||||
"\nfile1.5" +
|
"\nfile1.5" +
|
||||||
"\npath/to/file2" +
|
"\npath/to/file2" +
|
||||||
|
@ -364,20 +366,20 @@ namespace Test
|
||||||
"#SLDL:/other/new/file/path,\"Artist, 1.5\",,\"Title, , 1.5\",-1,0,3,0;path/to/file1,\"Artist, 1\",,\"Title, , 1\",-1,0,3,0;path/to/file2,\"Artist, 2\",,Title2,-1,0,3,0;,\"Artist; ,3\",,Title3 ;a,-1,0,4,0;,\"Artist,,, ;4\",,Title4,-1,0,4,3;" +
|
"#SLDL:/other/new/file/path,\"Artist, 1.5\",,\"Title, , 1.5\",-1,0,3,0;path/to/file1,\"Artist, 1\",,\"Title, , 1\",-1,0,3,0;path/to/file2,\"Artist, 2\",,Title2,-1,0,3,0;,\"Artist; ,3\",,Title3 ;a,-1,0,4,0;,\"Artist,,, ;4\",,Title4,-1,0,4,3;" +
|
||||||
",,,,-1,0,0,0;new/file/path,ArtistA,Albumm,TitleA,-1,0,1,0;,ArtistB,Albumm,TitleB,-1,0,2,3;" +
|
",,,,-1,0,0,0;new/file/path,ArtistA,Albumm,TitleA,-1,0,1,0;,ArtistB,Albumm,TitleB,-1,0,2,3;" +
|
||||||
"\n" +
|
"\n" +
|
||||||
"\n# Failed: Artist; ,3 - Title3 ;a [NoSuitableFileFound]" +
|
"\n#FAIL: Artist; ,3 - Title3 ;a [NoSuitableFileFound]" +
|
||||||
"\n# Failed: Artist,,, ;4 - Title4 [NoSuitableFileFound]" +
|
"\n#FAIL: Artist,,, ;4 - Title4 [NoSuitableFileFound]" +
|
||||||
"\npath/to/file1" +
|
"\npath/to/file1" +
|
||||||
"\n/other/new/file/path" +
|
"\n/other/new/file/path" +
|
||||||
"\npath/to/file2" +
|
"\npath/to/file2" +
|
||||||
"\nnew/file/path" +
|
"\nnew/file/path" +
|
||||||
"\n# Failed: ArtistB - TitleB [NoSuitableFileFound]" +
|
"\n#FAIL: ArtistB - TitleB [NoSuitableFileFound]" +
|
||||||
"\n";
|
"\n";
|
||||||
Assert(output == need);
|
Assert(output == need);
|
||||||
|
|
||||||
Console.WriteLine();
|
Console.WriteLine();
|
||||||
Console.WriteLine(output);
|
Console.WriteLine(output);
|
||||||
|
|
||||||
Program.m3uEditor = new M3uEditor(path, trackLists, Config.m3uOption);
|
Program.m3uEditor = new M3uEditor(path, trackLists, Config.I.m3uOption);
|
||||||
|
|
||||||
foreach (var t in trackLists.Flattened(false, false))
|
foreach (var t in trackLists.Flattened(false, false))
|
||||||
{
|
{
|
||||||
|
@ -406,8 +408,8 @@ namespace Test
|
||||||
trackLists.AddEntry(new TrackListEntry(t));
|
trackLists.AddEntry(new TrackListEntry(t));
|
||||||
|
|
||||||
File.WriteAllText(path, "");
|
File.WriteAllText(path, "");
|
||||||
Config.m3uOption = M3uOption.Index;
|
Config.I.m3uOption = M3uOption.Index;
|
||||||
Program.m3uEditor = new M3uEditor(path, trackLists, Config.m3uOption);
|
Program.m3uEditor = new M3uEditor(path, trackLists, Config.I.m3uOption);
|
||||||
Program.m3uEditor.Update();
|
Program.m3uEditor.Update();
|
||||||
|
|
||||||
Assert(File.ReadAllText(path) == "");
|
Assert(File.ReadAllText(path) == "");
|
||||||
|
@ -420,7 +422,7 @@ namespace Test
|
||||||
|
|
||||||
Program.m3uEditor.Update();
|
Program.m3uEditor.Update();
|
||||||
|
|
||||||
Program.m3uEditor = new M3uEditor(path, trackLists, Config.m3uOption);
|
Program.m3uEditor = new M3uEditor(path, trackLists, Config.I.m3uOption);
|
||||||
|
|
||||||
foreach (var t in test)
|
foreach (var t in test)
|
||||||
{
|
{
|
||||||
|
@ -470,12 +472,13 @@ namespace Test
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void ResetProfiles()
|
public static void ResetConfig()
|
||||||
{
|
{
|
||||||
var type = typeof(Config);
|
var singletonType = typeof(Config);
|
||||||
var field = type.GetField("profiles", BindingFlags.NonPublic | BindingFlags.Static);
|
var instanceField = singletonType.GetField("Instance", BindingFlags.Static | BindingFlags.NonPublic);
|
||||||
var value = (Dictionary<string, (List<string> args, string? cond)>)field.GetValue(null);
|
var constructor = singletonType.GetConstructor(BindingFlags.Instance | BindingFlags.NonPublic, null, Type.EmptyTypes, null);
|
||||||
value.Clear();
|
var newInstance = constructor.Invoke(null);
|
||||||
|
instanceField.SetValue(null, newInstance);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void Passed()
|
public static void Passed()
|
||||||
|
|
|
@ -89,13 +89,12 @@ public static class Utils
|
||||||
return path;
|
return path;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (path.StartsWith('~'))
|
path = path.Trim();
|
||||||
|
|
||||||
|
if (path.Length > 0 && path[0] == '~' && (path.Length == 1 || path[1] == '\\' || path[1] == '/'))
|
||||||
{
|
{
|
||||||
string homeDirectory = Environment.GetFolderPath(Environment.SpecialFolder.UserProfile);
|
string home = Environment.GetFolderPath(Environment.SpecialFolder.UserProfile);
|
||||||
path = Path.Join(homeDirectory, path.Substring(1).TrimStart('/').TrimStart('\\'));
|
path = Path.Join(home, path[1..].TrimStart('/').TrimStart('\\'));
|
||||||
|
|
||||||
if (path.Length > 0)
|
|
||||||
path = Path.GetFullPath(path);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return path;
|
return path;
|
||||||
|
|
Loading…
Reference in a new issue