Added YaS Online download mirror.
URL: http://osu.yas-online.net/ Note that fetching the direct download URL for a beatmap requires an extra query, which is not currently done asynchronously (but it really should be). Signed-off-by: Jeffrey Han <itdelatrisu@gmail.com>
This commit is contained in:
parent
b9adf59d53
commit
dbb5eebf5c
204
src/itdelatrisu/opsu/downloads/servers/YaSOnlineServer.java
Normal file
204
src/itdelatrisu/opsu/downloads/servers/YaSOnlineServer.java
Normal file
|
@ -0,0 +1,204 @@
|
|||
/*
|
||||
* opsu! - an open-source osu! client
|
||||
* Copyright (C) 2014, 2015 Jeffrey Han
|
||||
*
|
||||
* opsu! 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.
|
||||
*
|
||||
* opsu! 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 opsu!. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
package itdelatrisu.opsu.downloads.servers;
|
||||
|
||||
import itdelatrisu.opsu.ErrorHandler;
|
||||
import itdelatrisu.opsu.Utils;
|
||||
import itdelatrisu.opsu.downloads.DownloadNode;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.io.UnsupportedEncodingException;
|
||||
import java.net.MalformedURLException;
|
||||
import java.net.URL;
|
||||
import java.net.URLEncoder;
|
||||
import java.text.DateFormat;
|
||||
import java.text.SimpleDateFormat;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Date;
|
||||
import java.util.Iterator;
|
||||
import java.util.List;
|
||||
|
||||
import org.json.JSONObject;
|
||||
|
||||
/**
|
||||
* Download server: http://osu.yas-online.net/
|
||||
*/
|
||||
public class YaSOnlineServer extends DownloadServer {
|
||||
/** Server name. */
|
||||
private static final String SERVER_NAME = "YaS Online";
|
||||
|
||||
/** Formatted download URL (returns JSON): {@code beatmapSetID} */
|
||||
private static final String DOWNLOAD_URL = "https://osu.yas-online.net/json.mapdata.php?mapId=%d";
|
||||
|
||||
/**
|
||||
* Formatted download fetch URL: {@code downloadLink}
|
||||
* (e.g. {@code /fetch/49125122158ef360a66a07bce2d0483596913843-m-10418.osz})
|
||||
*/
|
||||
private static final String DOWNLOAD_FETCH_URL = "https://osu.yas-online.net%s";
|
||||
|
||||
/** Maximum beatmaps displayed per page. */
|
||||
private static final int PAGE_LIMIT = 25;
|
||||
|
||||
/** Formatted home URL: {@code page} */
|
||||
private static final String HOME_URL = "https://osu.yas-online.net/json.maplist.php?o=%d";
|
||||
|
||||
/** Formatted search URL: {@code query} */
|
||||
private static final String SEARCH_URL = "https://osu.yas-online.net/json.search.php?searchQuery=%s";
|
||||
|
||||
/** Total result count from the last query. */
|
||||
private int totalResults = -1;
|
||||
|
||||
/** Max server download ID seen (for approximating total pages). */
|
||||
private int maxServerID = 0;
|
||||
|
||||
/** Constructor. */
|
||||
public YaSOnlineServer() {}
|
||||
|
||||
@Override
|
||||
public String getName() { return SERVER_NAME; }
|
||||
|
||||
@Override
|
||||
public String getDownloadURL(int beatmapSetID) {
|
||||
try {
|
||||
// TODO: do this asynchronously (will require lots of changes...)
|
||||
return getDownloadURLFromMapData(beatmapSetID);
|
||||
} catch (IOException e) {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the beatmap download URL by downloading its map data.
|
||||
* <p>
|
||||
* This is needed because there is no other way to find a beatmap's direct
|
||||
* download URL.
|
||||
* @param beatmapSetID the beatmap set ID
|
||||
* @return the URL string, or null if the address could not be determined
|
||||
* @throws IOException if any connection error occurred
|
||||
*/
|
||||
private String getDownloadURLFromMapData(int beatmapSetID) throws IOException {
|
||||
try {
|
||||
// read JSON
|
||||
String search = String.format(DOWNLOAD_URL, beatmapSetID);
|
||||
JSONObject json = Utils.readJsonObjectFromUrl(new URL(search));
|
||||
JSONObject results;
|
||||
if (json == null ||
|
||||
!json.getString("result").equals("success") ||
|
||||
(results = json.getJSONObject("success")).length() == 0) {
|
||||
return null;
|
||||
}
|
||||
|
||||
// parse result
|
||||
Iterator<?> keys = results.keys();
|
||||
if (!keys.hasNext())
|
||||
return null;
|
||||
String key = (String) keys.next();
|
||||
JSONObject item = results.getJSONObject(key);
|
||||
String downloadLink = item.getString("downloadLink");
|
||||
return String.format(DOWNLOAD_FETCH_URL, downloadLink);
|
||||
} catch (MalformedURLException | UnsupportedEncodingException e) {
|
||||
ErrorHandler.error(String.format("Problem retrieving download URL for beatmap '%d'.", beatmapSetID), e, true);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public DownloadNode[] resultList(String query, int page, boolean rankedOnly) throws IOException {
|
||||
DownloadNode[] nodes = null;
|
||||
try {
|
||||
// read JSON
|
||||
String search;
|
||||
boolean isSearch;
|
||||
if (query.isEmpty()) {
|
||||
isSearch = false;
|
||||
search = String.format(HOME_URL, (page - 1) * PAGE_LIMIT);
|
||||
} else {
|
||||
isSearch = true;
|
||||
search = String.format(SEARCH_URL, URLEncoder.encode(query, "UTF-8"));
|
||||
}
|
||||
JSONObject json = Utils.readJsonObjectFromUrl(new URL(search));
|
||||
if (json == null) {
|
||||
this.totalResults = -1;
|
||||
return null;
|
||||
}
|
||||
JSONObject results;
|
||||
if (!json.getString("result").equals("success") ||
|
||||
(results = json.getJSONObject("success")).length() == 0) {
|
||||
this.totalResults = 0;
|
||||
return new DownloadNode[0];
|
||||
}
|
||||
|
||||
// parse result list
|
||||
List<DownloadNode> nodeList = new ArrayList<DownloadNode>();
|
||||
for (Object obj : results.keySet()) {
|
||||
String key = (String) obj;
|
||||
JSONObject item = results.getJSONObject(key);
|
||||
|
||||
// parse title and artist
|
||||
String title, artist;
|
||||
String str = item.getString("map");
|
||||
int index = str.indexOf(" - ");
|
||||
if (index > -1) {
|
||||
title = str.substring(0, index);
|
||||
artist = str.substring(index + 3);
|
||||
} else { // should never happen...
|
||||
title = str;
|
||||
artist = "?";
|
||||
}
|
||||
|
||||
// only contains date added if part of a beatmap pack
|
||||
int added = item.getInt("added");
|
||||
String date = (added == 0) ? "?" : formatDate(added);
|
||||
|
||||
// approximate page count
|
||||
int serverID = item.getInt("id");
|
||||
if (serverID > maxServerID)
|
||||
maxServerID = serverID;
|
||||
|
||||
nodeList.add(new DownloadNode(item.getInt("mapid"), date, title, null, artist, null, ""));
|
||||
}
|
||||
nodes = nodeList.toArray(new DownloadNode[nodeList.size()]);
|
||||
|
||||
// store total result count
|
||||
if (isSearch)
|
||||
this.totalResults = nodes.length;
|
||||
else
|
||||
this.totalResults = maxServerID;
|
||||
} catch (MalformedURLException | UnsupportedEncodingException e) {
|
||||
ErrorHandler.error(String.format("Problem loading result list for query '%s'.", query), e, true);
|
||||
}
|
||||
return nodes;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int minQueryLength() { return 3; }
|
||||
|
||||
@Override
|
||||
public int totalResults() { return totalResults; }
|
||||
|
||||
/**
|
||||
* Returns a formatted date string from a raw date.
|
||||
* @param timestamp the UTC timestamp, in seconds
|
||||
* @return the formatted date
|
||||
*/
|
||||
private String formatDate(int timestamp) {
|
||||
Date d = new Date(timestamp * 1000L);
|
||||
DateFormat fmt = new SimpleDateFormat("d MMM yyyy HH:mm:ss");
|
||||
return fmt.format(d);
|
||||
}
|
||||
}
|
|
@ -35,6 +35,7 @@ import itdelatrisu.opsu.downloads.DownloadNode;
|
|||
import itdelatrisu.opsu.downloads.servers.BloodcatServer;
|
||||
import itdelatrisu.opsu.downloads.servers.DownloadServer;
|
||||
import itdelatrisu.opsu.downloads.servers.HexideServer;
|
||||
import itdelatrisu.opsu.downloads.servers.YaSOnlineServer;
|
||||
import itdelatrisu.opsu.ui.MenuButton;
|
||||
import itdelatrisu.opsu.ui.UI;
|
||||
|
||||
|
@ -73,7 +74,7 @@ public class DownloadsMenu extends BasicGameState {
|
|||
private static final int MIN_REQUEST_INTERVAL = 300;
|
||||
|
||||
/** Available beatmap download servers. */
|
||||
private static final DownloadServer[] SERVERS = { new BloodcatServer(), new HexideServer() };
|
||||
private static final DownloadServer[] SERVERS = { new BloodcatServer(), new HexideServer(), new YaSOnlineServer() };
|
||||
|
||||
/** The beatmap download server index. */
|
||||
private int serverIndex = 0;
|
||||
|
|
Loading…
Reference in New Issue
Block a user