/* This file is part of Subsonic. Subsonic is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. Subsonic is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with Subsonic. If not, see . Copyright 2009 (C) Sindre Mehus */ package net.nullsum.audinaut.service; import android.content.Context; import android.content.SharedPreferences; import android.graphics.Bitmap; import android.util.Log; import androidx.annotation.Nullable; import net.nullsum.audinaut.R; import net.nullsum.audinaut.domain.Genre; import net.nullsum.audinaut.domain.Indexes; import net.nullsum.audinaut.domain.MusicDirectory; import net.nullsum.audinaut.domain.MusicFolder; import net.nullsum.audinaut.domain.Playlist; import net.nullsum.audinaut.domain.SearchCritera; import net.nullsum.audinaut.domain.SearchResult; import net.nullsum.audinaut.fragments.MainFragment; import net.nullsum.audinaut.service.parser.EntryListParser; import net.nullsum.audinaut.service.parser.ErrorParser; import net.nullsum.audinaut.service.parser.GenreParser; import net.nullsum.audinaut.service.parser.IndexesParser; import net.nullsum.audinaut.service.parser.MusicDirectoryParser; import net.nullsum.audinaut.service.parser.MusicFoldersParser; import net.nullsum.audinaut.service.parser.PlaylistParser; import net.nullsum.audinaut.service.parser.PlaylistsParser; import net.nullsum.audinaut.service.parser.RandomSongsParser; import net.nullsum.audinaut.service.parser.SearchResult2Parser; import net.nullsum.audinaut.util.Constants; import net.nullsum.audinaut.util.FileUtil; import net.nullsum.audinaut.util.Pair; import net.nullsum.audinaut.util.ProgressListener; import net.nullsum.audinaut.util.SilentBackgroundTask; import net.nullsum.audinaut.util.SongDBHandler; import net.nullsum.audinaut.util.Util; import java.io.FileOutputStream; import java.io.InputStream; import java.io.OutputStream; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.concurrent.TimeUnit; import okhttp3.OkHttpClient; import okhttp3.Request; import okhttp3.Response; /** * @author Sindre Mehus */ public class RESTMusicService implements MusicService { private static final String TAG = RESTMusicService.class.getSimpleName(); private static final OkHttpClient client = new OkHttpClient(); private Integer instance; @Override public void ping(Context context, ProgressListener progressListener) throws Exception { String url = getRestUrl(context, "ping", null); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { new ErrorParser(context, getInstance(context)).parse(response.body().byteStream()); } } public List getMusicFolders(boolean refresh, Context context, ProgressListener progressListener) throws Exception { String url = getRestUrl(context, "getMusicFolders", null); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { return new MusicFoldersParser(context, getInstance(context)).parse(response.body().byteStream()); } } @Override public Indexes getIndexes(String musicFolderId, boolean refresh, Context context, ProgressListener progressListener) throws Exception { Map parameters = new HashMap<>(); if (musicFolderId != null) { parameters.put("musicFolderId", musicFolderId); } else { parameters = null; } String url = getRestUrl(context, "getArtists", parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { return new IndexesParser(context, getInstance(context)).parse(response.body().byteStream(), progressListener); } } @Override public MusicDirectory getMusicDirectory(String id, String name, boolean refresh, Context context, ProgressListener progressListener) throws Exception { SharedPreferences prefs = Util.getPreferences(context); String cacheLocn = prefs.getString(Constants.PREFERENCES_KEY_CACHE_LOCATION, null); if (cacheLocn != null && id.contains(cacheLocn)) { String search = Util.parseOfflineIDSearch(id, cacheLocn); SearchCritera critera = new SearchCritera(search, 1, 1, 0); SearchResult result = search(critera, context, progressListener); if (result.getArtists().size() == 1) { id = result.getArtists().get(0).getId(); } else if (result.getAlbums().size() == 1) { id = result.getAlbums().get(0).getId(); } } MusicDirectory dir = null; int index, start = 0; while ((index = id.indexOf(';', start)) != -1) { MusicDirectory extra = getMusicDirectoryImpl(id.substring(start, index), name, context); if (dir == null) { dir = extra; } else { dir.addChildren(extra.getChildren()); } start = index + 1; } MusicDirectory extra = getMusicDirectoryImpl(id.substring(start), name, context); if (dir == null) { dir = extra; } else { dir.addChildren(extra.getChildren()); } return dir; } private MusicDirectory getMusicDirectoryImpl(String id, String name, Context context) throws Exception { Map parameters = new HashMap<>(); parameters.put("id", id); String url = getRestUrl(context, "getMusicDirectory", parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { return new MusicDirectoryParser(context, getInstance(context)).parse(name, response.body().byteStream()); } } @Override public MusicDirectory getArtist(String id, String name, boolean refresh, Context context, ProgressListener progressListener) throws Exception { Map parameters = new HashMap<>(); parameters.put("id", id); String url = getRestUrl(context, "getArtist", parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { return new MusicDirectoryParser(context, getInstance(context)).parse(name, response.body().byteStream()); } } @Override public MusicDirectory getAlbum(String id, String name, boolean refresh, Context context, ProgressListener progressListener) throws Exception { Map parameters = new HashMap<>(); parameters.put("id", id); String url = getRestUrl(context, "getAlbum", parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { return new MusicDirectoryParser(context, getInstance(context)).parse(name, response.body().byteStream()); } } @Override public SearchResult search(SearchCritera critera, Context context, ProgressListener progressListener) throws Exception { Map parameters = new HashMap<>(); parameters.put("query", critera.getQuery()); parameters.put("artistCount", Integer.toString(critera.getArtistCount())); parameters.put("albumCount", Integer.toString(critera.getAlbumCount())); parameters.put("songCount", Integer.toString(critera.getSongCount())); String url = getRestUrl(context, "search3", parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { return new SearchResult2Parser(context, getInstance(context)).parse(response.body().byteStream()); } } @Override public MusicDirectory getPlaylist(boolean refresh, String id, String name, Context context, ProgressListener progressListener) throws Exception { Map parameters = new HashMap<>(); parameters.put("id", id); String url = getRestUrl(context, "getPlaylist", parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { return new PlaylistParser(context, getInstance(context)).parse(response.body().byteStream()); } } @Override public List getPlaylists(boolean refresh, Context context, ProgressListener progressListener) throws Exception { String url = getRestUrl(context, "getPlaylists", null); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { return new PlaylistsParser(context, getInstance(context)).parse(response.body().byteStream()); } } @Override public void createPlaylist(String id, String name, List entries, Context context, ProgressListener progressListener) throws Exception { Map parameters = new HashMap<>(); if (id != null) { parameters.put("playlistId", id); } if (name != null) { parameters.put("name", name); } for (MusicDirectory.Entry entry : entries) { parameters.put("songId", getOfflineSongId(entry.getId(), context, progressListener)); } String url = getRestUrl(context, "createPlaylist", parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { new ErrorParser(context, getInstance(context)).parse(response.body().byteStream()); } } @Override public void deletePlaylist(String id, Context context, ProgressListener progressListener) throws Exception { Map parameters = new HashMap<>(); parameters.put("id", id); String url = getRestUrl(context, "deletePlaylist", parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { new ErrorParser(context, getInstance(context)).parse(response.body().byteStream()); } } @Override public void addToPlaylist(String id, List toAdd, Context context, ProgressListener progressListener) throws Exception { Map parameters = new HashMap<>(); parameters.put("playlistId", id); for (MusicDirectory.Entry song : toAdd) { parameters.put("songIdToAdd", getOfflineSongId(song.getId(), context, progressListener)); } String url = getRestUrl(context, "updatePlaylist", parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { new ErrorParser(context, getInstance(context)).parse(response.body().byteStream()); } } @Override public void removeFromPlaylist(String id, List toRemove, Context context, ProgressListener progressListener) throws Exception { Map parameters = new HashMap<>(); parameters.put("playlistId", id); for (Integer song : toRemove) { parameters.put("songIndexToRemove", Integer.toString(song)); } String url = getRestUrl(context, "updatePlaylist", parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { new ErrorParser(context, getInstance(context)).parse(response.body().byteStream()); } } @Override public void overwritePlaylist(String id, String name, int toRemove, List toAdd, Context context, ProgressListener progressListener) throws Exception { Map parameters = new HashMap<>(); parameters.put("playlistId", id); parameters.put("name", name); for (MusicDirectory.Entry song : toAdd) { parameters.put("songIdToAdd", getOfflineSongId(song.getId(), context, progressListener)); } for (int i = 0; i < toRemove; i++) { parameters.put("songIndexToRemove", Integer.toString(i)); } String url = getRestUrl(context, "updatePlaylist", parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { new ErrorParser(context, getInstance(context)).parse(response.body().byteStream()); } } @Override public void updatePlaylist(String id, String name, String comment, boolean pub, Context context, ProgressListener progressListener) throws Exception { Map parameters = new HashMap<>(); parameters.put("playlistId", id); parameters.put("name", name); parameters.put("comment", comment); parameters.put("public", Boolean.toString(pub)); String url = getRestUrl(context, "updatePlaylist", parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { new ErrorParser(context, getInstance(context)).parse(response.body().byteStream()); } } @Override public MusicDirectory getAlbumList(String type, int size, int offset, boolean refresh, Context context, ProgressListener progressListener) throws Exception { Map parameters = new HashMap<>(); parameters.put("type", type); parameters.put("size", Integer.toString(size)); parameters.put("offset", Integer.toString(offset)); // Add folder if it was set and is non null int instance = getInstance(context); if (Util.getAlbumListsPerFolder(context, instance)) { String folderId = Util.getSelectedMusicFolderId(context, instance); if (folderId != null) { parameters.put("musicFolderId", folderId); } } String url = getRestUrl(context, "getAlbumList2", parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { return new EntryListParser(context, getInstance(context)).parse(response.body().byteStream()); } } @Override public MusicDirectory getAlbumList(String type, String extra, int size, int offset, boolean refresh, Context context, ProgressListener progressListener) throws Exception { Map parameters = new HashMap<>(); parameters.put("size", Integer.toString(size)); parameters.put("offset", Integer.toString(offset)); int instance = getInstance(context); if ("genres".equals(type)) { parameters.put("type", "byGenre"); parameters.put("genre", extra); } else if ("years".equals(type)) { int decade = Integer.parseInt(extra); parameters.put("type", "byYear"); parameters.put("fromYear", Integer.toString(decade + 9)); parameters.put("toYear", Integer.toString(decade)); } // Add folder if it was set and is non null if (Util.getAlbumListsPerFolder(context, instance)) { String folderId = Util.getSelectedMusicFolderId(context, instance); if (folderId != null) { parameters.put("musicFolderId", folderId); } } String url = getRestUrl(context, "getAlbumList2", parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { return new EntryListParser(context, getInstance(context)).parse(response.body().byteStream()); } } @Override public MusicDirectory getSongList(String type, int size, int offset, Context context, ProgressListener progressListener) throws Exception { Map parameters = new HashMap<>(); parameters.put("size", Integer.toString(size)); parameters.put("offset", Integer.toString(offset)); String method; switch (type) { case MainFragment.SONGS_NEWEST: method = "getNewaddedSongs"; break; case MainFragment.SONGS_TOP_PLAYED: method = "getTopplayedSongs"; break; case MainFragment.SONGS_RECENT: method = "getLastplayedSongs"; break; case MainFragment.SONGS_FREQUENT: method = "getMostplayedSongs"; break; default: method = "getNewaddedSongs"; } String url = getRestUrl(context, method, parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { return new EntryListParser(context, getInstance(context)).parse(response.body().byteStream()); } } @Override public MusicDirectory getRandomSongs(int size, String musicFolderId, String genre, String startYear, String endYear, Context context, ProgressListener progressListener) throws Exception { Map parameters = new HashMap<>(); parameters.put("size", Integer.toString(size)); if (genre != null && !"".equals(genre)) { parameters.put("genre", genre); } if (startYear != null && !"".equals(startYear)) { // Check to make sure user isn't doing 2015 -> 2010 since Subsonic will return no results if (endYear != null && !"".equals(endYear)) { try { int startYearInt = Integer.parseInt(startYear); int endYearInt = Integer.parseInt(endYear); if (startYearInt > endYearInt) { String tmp = startYear; startYear = endYear; endYear = tmp; } } catch (Exception e) { Log.w(TAG, "Failed to convert start/end year into ints", e); } } parameters.put("fromYear", startYear); } if (endYear != null && !"".equals(endYear)) { parameters.put("toYear", endYear); } String url = getRestUrl(context, "getRandomSongs", parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { return new RandomSongsParser(context, getInstance(context)).parse(response.body().byteStream()); } } @Override public Bitmap getCoverArt(Context context, MusicDirectory.Entry entry, int size, ProgressListener progressListener, SilentBackgroundTask task) throws Exception { // Synchronize on the entry so that we don't download concurrently for the same song. synchronized (entry) { // Use cached file, if existing. Bitmap bitmap = FileUtil.getAlbumArtBitmap(context, entry, size); if (bitmap != null) { return bitmap; } Map parameters = new HashMap<>(); parameters.put("id", entry.getCoverArt()); String url = getRestUrl(context, "getCoverArt", parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { InputStream in = response.body().byteStream(); byte[] bytes = Util.toByteArray(in); // Handle case where partial was downloaded before being cancelled if (task != null && task.isCancelled()) { return null; } OutputStream out = null; try { out = new FileOutputStream(FileUtil.getAlbumArtFile(context, entry)); out.write(bytes); } finally { Util.close(out); } // Size == 0 -> only want to download if (size == 0) { return null; } else { return FileUtil.getSampledBitmap(bytes, size); } } } } @Override public Response getDownloadInputStream(Context context, MusicDirectory.Entry song, long offset, int maxBitrate, SilentBackgroundTask task) throws Exception { OkHttpClient eagerClient = client.newBuilder() .readTimeout(Util.getNetworkTimeoutMs(context), TimeUnit.MILLISECONDS) .build(); Map parameters = new HashMap<>(); parameters.put("id", song.getId()); parameters.put("maxBitRate", Integer.toString(maxBitrate)); String url = getRestUrl(context, "stream", parameters); Request.Builder requestBuilder = new Request.Builder(); if (offset > 0) { requestBuilder.header("Range", "bytes=" + offset + "-"); } requestBuilder.url(url); Request request = requestBuilder.build(); return eagerClient.newCall(request).execute(); } @Override public List getGenres(boolean refresh, Context context, ProgressListener progressListener) throws Exception { String url = getRestUrl(context, "getGenres", null); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { return new GenreParser(context, getInstance(context)).parse(response.body().byteStream()); } } @Override public MusicDirectory getSongsByGenre(String genre, int count, int offset, Context context, ProgressListener progressListener) throws Exception { Map parameters = new HashMap<>(); parameters.put("genre", genre); parameters.put("count", Integer.toString(count)); parameters.put("offset", Integer.toString(offset)); // Add folder if it was set and is non null int instance = getInstance(context); if (Util.getAlbumListsPerFolder(context, instance)) { String folderId = Util.getSelectedMusicFolderId(context, instance); if (folderId != null) { parameters.put("musicFolderId", folderId); } } String url = getRestUrl(context, "getSongsByGenre", parameters); Request request = new Request.Builder() .url(url) .build(); try (Response response = client.newCall(request).execute()) { return new RandomSongsParser(context, getInstance(context)).parse(response.body().byteStream()); } } @Override public void startScan(Context context) throws Exception { String url = getRestUrl(context, "startScan", null); Request request = new Request.Builder().url(url).build(); try (Response response = client.newCall(request).execute()) { if (response.isSuccessful()) { Log.d(TAG, "Media scan started" + response.toString()); } else { Log.w(TAG, "media scan start failed" + response.toString()); Util.toast(context, R.string.settings_media_scan_start_failed); } } } private String getOfflineSongId(String id, Context context, ProgressListener progressListener) throws Exception { SharedPreferences prefs = Util.getPreferences(context); String cacheLocn = prefs.getString(Constants.PREFERENCES_KEY_CACHE_LOCATION, null); if (cacheLocn != null && id.contains(cacheLocn)) { Pair cachedSongId = SongDBHandler.getHandler(context).getIdFromPath(Util.getRestUrlHash(context, getInstance(context)), id); if (cachedSongId != null) { id = cachedSongId.getSecond(); } else { String searchCriteria = Util.parseOfflineIDSearch(id, cacheLocn); SearchCritera critera = new SearchCritera(searchCriteria, 0, 0, 1); SearchResult result = search(critera, context, progressListener); if (result.getSongs().size() == 1) { id = result.getSongs().get(0).getId(); } } } return id; } @Override public void setInstance(Integer instance) { this.instance = instance; } public int getInstance(Context context) { if (instance == null) { return Util.getActiveServer(context); } else { return instance; } } public String getRestUrl(Context context, String method, @Nullable Map parameters) { return getRestUrl(context, method, true, parameters); } public String getRestUrl(Context context, String method, boolean allowAltAddress, @Nullable Map parameters) { if (instance == null) { return Util.getRestUrl(context, method, allowAltAddress, parameters); } else { return Util.getRestUrl(context, method, instance, allowAltAddress, parameters); } } }