Переделано воспроизведение аудио
All checks were successful
Release / pack-and-publish (release) Successful in 36s

This commit is contained in:
2026-04-21 11:14:36 +03:00
parent eb1eba0162
commit 526353d679
12 changed files with 107 additions and 47 deletions

View File

@@ -15,7 +15,7 @@ public static class YAlbumExtensions
if (album.Volumes != null)
return album;
var result = await album.Context.API.Album.GetAsync(album.Id);
var result = await album.Context.Api.Album.GetAsync(album.Id);
return result ?? album;
}
@@ -23,11 +23,11 @@ public static class YAlbumExtensions
/// Добавляет альбом в список лайкнутых.
/// </summary>
public static async Task<string?> AddLikeAsync(this YAlbum album)
=> await album.Context.API.Library.AddAlbumLikeAsync(album);
=> await album.Context.Api.Library.AddAlbumLikeAsync(album);
/// <summary>
/// Удаляет альбом из списка лайкнутых.
/// </summary>
public static async Task<string?> RemoveLikeAsync(this YAlbum album)
=> await album.Context.API.Library.RemoveAlbumLikeAsync(album);
=> await album.Context.Api.Library.RemoveAlbumLikeAsync(album);
}

View File

@@ -12,29 +12,29 @@ public static class YArtistExtensions
/// Получает расширенную информацию об исполнителе.
/// </summary>
public static async Task<YArtistBriefInfo?> BriefInfoAsync(this YArtist artist)
=> await artist.Context.API.Artist.GetAsync(artist.Id);
=> await artist.Context.Api.Artist.GetAsync(artist.Id);
/// <summary>
/// Получает страницу треков исполнителя.
/// </summary>
public static async Task<YTracksPage?> GetTracksAsync(this YArtist artist, int page = 0, int pageSize = 20)
=> await artist.Context.API.Artist.GetTracksAsync(artist.Id, page, pageSize);
=> await artist.Context.Api.Artist.GetTracksAsync(artist.Id, page, pageSize);
/// <summary>
/// Получает все треки исполнителя.
/// </summary>
public static async Task<List<YTrack>?> GetAllTracksAsync(this YArtist artist)
=> (await artist.Context.API.Artist.GetAllTracksAsync(artist.Id))?.Tracks;
=> (await artist.Context.Api.Artist.GetAllTracksAsync(artist.Id))?.Tracks;
/// <summary>
/// Добавляет исполнителя в список лайкнутых.
/// </summary>
public static async Task<string?> AddLikeAsync(this YArtist artist)
=> await artist.Context.API.Library.AddArtistLikeAsync(artist);
=> await artist.Context.Api.Library.AddArtistLikeAsync(artist);
/// <summary>
/// Удаляет исполнителя из списка лайкнутых.
/// </summary>
public static async Task<string?> RemoveLikeAsync(this YArtist artist)
=> await artist.Context.API.Library.RemoveArtistLikeAsync(artist);
=> await artist.Context.Api.Library.RemoveArtistLikeAsync(artist);
}

View File

@@ -18,44 +18,44 @@ public static class YPlaylistExtensions
{
if (playlist.Tracks != null)
return playlist;
return await playlist.Context.API.Playlist.GetAsync(playlist);
return await playlist.Context.Api.Playlist.GetAsync(playlist);
}
/// <summary>
/// Добавляет плейлист в список лайкнутых.
/// </summary>
public static async Task<string?> AddLikeAsync(this YPlaylist playlist)
=> await playlist.Context.API.Library.AddPlaylistLikeAsync(playlist);
=> await playlist.Context.Api.Library.AddPlaylistLikeAsync(playlist);
/// <summary>
/// Удаляет плейлист из списка лайкнутых.
/// </summary>
public static async Task<string?> RemoveLikeAsync(this YPlaylist playlist)
=> await playlist.Context.API.Library.RemovePlaylistLikeAsync(playlist);
=> await playlist.Context.Api.Library.RemovePlaylistLikeAsync(playlist);
/// <summary>
/// Переименовывает плейлист (только для владельца).
/// </summary>
public static async Task<YPlaylist?> RenameAsync(this YPlaylist playlist, string newName)
=> IsOwner(playlist) ? await playlist.Context.API.Playlist.RenameAsync(playlist, newName) : playlist;
=> IsOwner(playlist) ? await playlist.Context.Api.Playlist.RenameAsync(playlist, newName) : playlist;
/// <summary>
/// Удаляет плейлист (только для владельца).
/// </summary>
public static async Task<bool> DeleteAsync(this YPlaylist playlist)
=> IsOwner(playlist) && await playlist.Context.API.Playlist.DeleteAsync(playlist);
=> IsOwner(playlist) && await playlist.Context.Api.Playlist.DeleteAsync(playlist);
/// <summary>
/// Вставляет треки в начало плейлиста (только для владельца).
/// </summary>
public static async Task<YPlaylist?> InsertTracksAsync(this YPlaylist playlist, params YTrack[] tracks)
=> IsOwner(playlist) ? await playlist.Context.API.Playlist.InsertTracksAsync(playlist, tracks) : playlist;
=> IsOwner(playlist) ? await playlist.Context.Api.Playlist.InsertTracksAsync(playlist, tracks) : playlist;
/// <summary>
/// Удаляет треки из плейлиста (только для владельца).
/// </summary>
public static async Task<YPlaylist?> RemoveTracksAsync(this YPlaylist playlist, params YTrack[] tracks)
=> IsOwner(playlist) ? await playlist.Context.API.Playlist.DeleteTracksAsync(playlist, tracks) : playlist;
=> IsOwner(playlist) ? await playlist.Context.Api.Playlist.DeleteTracksAsync(playlist, tracks) : playlist;
/// <summary>
/// Загружает трек в плейлист (только для владельца).
@@ -63,7 +63,7 @@ public static class YPlaylistExtensions
public static async Task<bool> UploadTrackAsync(this YPlaylist playlist, string filePath, string fileName)
{
if (!IsOwner(playlist)) return false;
var result = await playlist.Context.API.UserGeneratedContent.UploadTrackToPlaylistAsync(playlist, fileName, filePath);
var result = await playlist.Context.Api.UserGeneratedContent.UploadTrackToPlaylistAsync(playlist, fileName, filePath);
return result == "CREATED";
}
}

View File

@@ -12,17 +12,17 @@ public static class YStationResultExtensions
/// Получает список треков для радиостанции.
/// </summary>
public static async Task<List<YSequenceItem>?> GetTracksAsync(this YStation station, string prevTrackId = "")
=> (await station.Context.API.Radio.GetStationTracksAsync(station, prevTrackId))?.Sequence;
=> (await station.Context.Api.Radio.GetStationTracksAsync(station, prevTrackId))?.Sequence;
/// <summary>
/// Устанавливает настройки станции.
/// </summary>
public static async Task<string?> SetSettings2Async(this YStation station, YStationSettings2 settings)
=> await station.Context.API.Radio.SetStationSettings2Async(station, settings);
=> await station.Context.Api.Radio.SetStationSettings2Async(station, settings);
/// <summary>
/// Отправляет обратную связь о прослушивании.
/// </summary>
public static Task<string?> SendFeedbackAsync(this YStation station, YStationFeedbackType type, YTrack? track = null, string batchId = "", double totalPlayedSeconds = 0)
=> station.Context.API.Radio.SendStationFeedbackAsync(station, type, track, batchId, totalPlayedSeconds);
=> station.Context.Api.Radio.SendStationFeedbackAsync(station, type, track, batchId, totalPlayedSeconds);
}

View File

@@ -11,53 +11,53 @@ public static class YTrackExtensions
/// Получает прямую ссылку на скачивание трека.
/// </summary>
public static Task<string?> GetLinkAsync(this YTrack track)
=> track.Context.API.Track.GetFileLinkAsync(track);
=> track.Context.Api.Track.GetFileLinkAsync(track);
/// <summary>
/// Сохраняет трек в файл.
/// </summary>
public static Task SaveAsync(this YTrack track, string filePath)
=> track.Context.API.Track.ExtractToFileAsync(track, filePath);
=> track.Context.Api.Track.ExtractToFileAsync(track, filePath);
/// <summary>
/// Добавляет трек в список лайкнутых.
/// </summary>
public static async Task<int?> AddLikeAsync(this YTrack track)
=> await track.Context.API.Library.AddTrackLikeAsync(track);
=> await track.Context.Api.Library.AddTrackLikeAsync(track);
/// <summary>
/// Удаляет трек из списка лайкнутых.
/// </summary>
public static async Task<int?> RemoveLikeAsync(this YTrack track)
=> await track.Context.API.Library.RemoveTrackLikeAsync(track);
=> await track.Context.Api.Library.RemoveTrackLikeAsync(track);
/// <summary>
/// Добавляет трек в список дизлайкнутых.
/// </summary>
public static async Task<int?> AddDislikeAsync(this YTrack track)
=> await track.Context.API.Library.AddTrackDislikeAsync(track);
=> await track.Context.Api.Library.AddTrackDislikeAsync(track);
/// <summary>
/// Удаляет трек из списка дизлайкнутых.
/// </summary>
public static async Task<int?> RemoveDislikeAsync(this YTrack track)
=> await track.Context.API.Library.RemoveTrackDislikeAsync(track);
=> await track.Context.Api.Library.RemoveTrackDislikeAsync(track);
/// <summary>
/// Отправляет информацию о воспроизведении трека.
/// </summary>
public static Task<string?> SendPlayTrackInfoAsync(this YTrack track, string from, bool fromCache = false, string playId = "", string playlistId = "", double totalPlayedSeconds = 0, double endPositionSeconds = 0)
=> track.Context.API.Track.SendPlayTrackInfoAsync(track, from, fromCache, playId, playlistId, totalPlayedSeconds, endPositionSeconds);
=> track.Context.Api.Track.SendPlayTrackInfoAsync(track, from, fromCache, playId, playlistId, totalPlayedSeconds, endPositionSeconds);
/// <summary>
/// Получает дополнительную информацию о треке.
/// </summary>
public static async Task<YTrackSupplement?> SupplementAsync(this YTrack track)
=> await track.Context.API.Track.GetSupplementAsync(track);
=> await track.Context.Api.Track.GetSupplementAsync(track);
/// <summary>
/// Получает похожие треки.
/// </summary>
public static async Task<YTrackSimilar?> SimilarAsync(this YTrack track)
=> await track.Context.API.Track.GetSimilarAsync(track);
=> await track.Context.Api.Track.GetSimilarAsync(track);
}

View File

@@ -9,7 +9,7 @@ namespace YandexMusic.API.Models.Common;
public class YExecutionContext
{
/// <summary>Экземпляр основного API.</summary>
public YandexMusicApi API { get; internal set; } = null!;
public YandexMusicApi Api { get; internal set; } = null!;
/// <summary>Хранилище данных авторизации.</summary>
public AuthStorage Storage { get; internal set; } = null!;

View File

@@ -38,7 +38,7 @@ public class YExecutionContextConverter : JsonConverter<object>
var obj = JsonSerializer.Deserialize(ref reader, typeToConvert, innerOptions);
if (obj is YBaseModel baseModel)
{
baseModel.Context = new YExecutionContext { API = _api, Storage = _storage };
baseModel.Context = new YExecutionContext { Api = _api, Storage = _storage };
}
return obj;
}

View File

@@ -1,9 +1,16 @@
using System.Xml.Serialization;
namespace YandexMusic.API.Models.Common;
[XmlRoot("download-info")]
public class YStorageDownloadFile
{
[XmlElement("host")]
public string Host { get; set; }
[XmlElement("path")]
public string Path { get; set; }
[XmlElement("s")]
public string S { get; set; }
[XmlElement("ts")]
public string Ts { get; set; }
}

View File

@@ -10,7 +10,17 @@ namespace YandexMusic.API.Requests.Common;
/// </summary>
internal abstract class YJsonRequestBuilder<TResponse, TParams> : YRequestBuilder<TParams>
{
protected YJsonRequestBuilder(YandexMusicApi api) : base(api) { }
private readonly JsonSerializerOptions _jsonOptions;
protected YJsonRequestBuilder(YandexMusicApi api) : base(api)
{
_jsonOptions = new JsonSerializerOptions
{
PropertyNamingPolicy = JsonNamingPolicy.CamelCase,
DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull,
Converters = { new JsonStringEnumConverter(JsonNamingPolicy.CamelCase) }
};
}
protected virtual async Task<TResponse?> DeserializeAsync(HttpResponseMessage response)
{
@@ -43,6 +53,8 @@ internal abstract class YJsonRequestBuilder<TResponse, TParams> : YRequestBuilde
}
}
protected string SerializeJson(object data) => JsonSerializer.Serialize(data, _jsonOptions);
/// <summary>
/// Выполняет запрос и возвращает десериализованный объект типа TResponse.
/// </summary>

View File

@@ -2,8 +2,6 @@
using System.Net;
using System.Net.Http.Headers;
using System.Text;
using System.Text.Json;
using System.Text.Json.Serialization;
using System.Text.RegularExpressions;
using System.Web;
using YandexMusic.API.Common;
@@ -23,7 +21,8 @@ internal abstract class YRequestBuilder<TParams>
/// <summary>Шаблон пути (может содержать плейсхолдеры вида {id}).</summary>
protected abstract string PathTemplate { get; }
private readonly JsonSerializerOptions _jsonOptions;
/// <summary>Определяет, нужно ли добавлять заголовок Authorization для этого запроса.</summary>
protected virtual bool ShouldAddAuthorization => true;
/// <summary>Основной экземпляр API.</summary>
protected YandexMusicApi Api { get; }
@@ -34,12 +33,6 @@ internal abstract class YRequestBuilder<TParams>
protected YRequestBuilder(YandexMusicApi api)
{
Api = api;
_jsonOptions = new JsonSerializerOptions
{
PropertyNamingPolicy = JsonNamingPolicy.CamelCase,
DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull,
Converters = { new JsonStringEnumConverter(JsonNamingPolicy.CamelCase) }
};
}
private string FullUrl => $"{BaseUrl.TrimEnd('/')}/{PathTemplate.TrimStart('/')}";
@@ -66,7 +59,7 @@ internal abstract class YRequestBuilder<TParams>
};
msg.Headers.TryAddWithoutValidation(GetHeaderName(HttpRequestHeader.AcceptCharset), Encoding.UTF8.WebName);
msg.Headers.TryAddWithoutValidation(GetHeaderName(HttpRequestHeader.AcceptEncoding), "gzip");
if (!string.IsNullOrEmpty(Storage.Token))
if (ShouldAddAuthorization && !string.IsNullOrEmpty(Storage.Token))
msg.Headers.TryAddWithoutValidation(GetHeaderName(HttpRequestHeader.Authorization), $"OAuth {Storage.Token}");
SetCustomHeaders(msg.Headers);
return msg;
@@ -120,8 +113,6 @@ internal abstract class YRequestBuilder<TParams>
protected virtual HttpContent? GetContent(TParams parameters) => null;
protected virtual void SetCustomHeaders(HttpRequestHeaders headers) { }
protected string SerializeJson(object data) => JsonSerializer.Serialize(data, _jsonOptions);
/// <summary>Выполняет запрос и возвращает десериализованный ответ.</summary>
public async Task<HttpResponseMessage?> ExecuteRawAsync(TParams parameters)
{

View File

@@ -0,0 +1,47 @@
using System.Xml;
using System.Xml.Serialization;
namespace YandexMusic.API.Requests.Common;
/// <summary>
/// Строитель запросов с десериализацией XML-ответа в TResponse.
/// </summary>
internal abstract class YXmlRequestBuilder<TResponse, TParams> : YRequestBuilder<TParams>
{
protected YXmlRequestBuilder(YandexMusicApi api) : base(api) { }
/// <summary>
/// Десериализует XML-ответ в объект типа TResponse.
/// </summary>
protected virtual async Task<TResponse?> DeserializeAsync(HttpResponseMessage response)
{
var xml = await response.Content.ReadAsStringAsync();
if (!response.IsSuccessStatusCode)
{
// Для XML-ошибок можно создать отдельную модель, но для простоты выбрасываем исключение
throw new Exception($"Ошибка HTTP {response.StatusCode}: {xml}");
}
try
{
using var stringReader = new StringReader(xml);
using var xmlReader = XmlReader.Create(stringReader, new XmlReaderSettings { Async = true });
var serializer = new XmlSerializer(typeof(TResponse));
return (TResponse?)serializer.Deserialize(xmlReader);
}
catch (Exception ex)
{
throw new Exception($"Ошибка десериализации XML: {ex.Message}\nXML: {xml}", ex);
}
}
/// <summary>
/// Выполняет запрос и возвращает десериализованный объект типа TResponse.
/// </summary>
public async Task<TResponse?> ExecuteAsync(TParams parameters)
{
using var response = await ExecuteRawAsync(parameters);
return await DeserializeAsync(response);
}
}

View File

@@ -6,9 +6,11 @@ using YandexMusic.API.Requests.Common;
namespace YandexMusic.API.Requests.Track;
/// <summary>Особый запрос не к api.music.yandex.net, а к произвольному URL.</summary>
internal class YStorageDownloadFileBuilder : YJsonRequestBuilder<YStorageDownloadFile?, string>
internal class YStorageDownloadFileBuilder : YXmlRequestBuilder<YStorageDownloadFile?, string>
{
public YStorageDownloadFileBuilder(YandexMusicApi api) : base(api) { }
protected override bool ShouldAddAuthorization => false;
protected override string BaseUrl => "{src}"; // не используется, т.к. URL берётся из параметра
protected override string Method => WebRequestMethods.Http.Get;
@@ -16,10 +18,11 @@ internal class YStorageDownloadFileBuilder : YJsonRequestBuilder<YStorageDownloa
protected override string PathTemplate => "";
protected override Dictionary<string, string> GetSubstitutions(string src)
=> new() { { "src", src.Split('?')[0] } };
=> new() { { "src", src } };
protected override NameValueCollection GetQueryParams(string src)
{
var query = new NameValueCollection { { "format", "json" } };
var query = new NameValueCollection();
var parts = src.Split('?');
if (parts.Length > 1)
{