Ответ 1
Во-первых, вам нужно убедиться, что вы правильно настроили mimetype при работе с мультимедийными файлами.
Во-вторых, вы не будете очень далеко читать файл MP3 по строкам, используя FileReader
, вместо этого вы должны предоставить NanoHTTPD с InputStream
.
Ниже приведена рабочая измененная версия вашего кода, которая служит файлу MP3. Установив mimetype на audio/mpeg
, вы разрешите браузеру решить, что делать с этим контентом. Например, в Chrome встроен музыкальный проигрыватель и воспроизводит файл.
public class StackOverflowMp3Server extends NanoHTTPD {
public StackOverflowMp3Server() {
super(8089);
}
@Override
public Response serve(String uri, Method method,
Map<String, String> header, Map<String, String> parameters,
Map<String, String> files) {
String answer = "";
FileInputStream fis = null;
try {
fis = new FileInputStream(Environment.getExternalStorageDirectory()
+ "/music/musicfile.mp3");
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return new NanoHTTPD.Response(Status.OK, "audio/mpeg", fis);
}
}
EDIT: многие люди спрашивали, как сделать аудиофайл доступным с использованием запросов диапазона, я продемонстрирую это ниже
Чтобы сделать аудиофайл доступным, используются запросы диапазона, которые позволяют HTTP-клиентам извлекать части аудиофайла в кусках. Убедитесь, что вы обслуживаете файл со статусом ответа PARTIAL_CONTENT (HTTP 206). Пример реализации можно найти в примере кода NanoHTTPD: SimpleWebserver.java
В моей реализации вместо того, чтобы возвращать ответ NanoHTTPD непосредственно в методе serve, я создаю другой метод, называемый "servefile", который я использую в качестве ответа для обработки запросов диапазона, как вы можете видеть ниже. Этот код является модифицированной реализацией SimpleWebServer.java, указанной выше.
@Override
public Response serve(String uri, Method method,
Map<String, String> header, Map<String, String> parameters,
Map<String, String> files) {
File f = new File(Environment.getExternalStorageDirectory()
+ "/music/musicfile.mp3");
String mimeType = "audio/mpeg";
return serveFile(uri, header, f, mimeType);
}
//Announce that the file server accepts partial content requests
private Response createResponse(Response.Status status, String mimeType,
InputStream message) {
Response res = new Response(status, mimeType, message);
res.addHeader("Accept-Ranges", "bytes");
return res;
}
/**
* Serves file from homeDir and its' subdirectories (only). Uses only URI,
* ignores all headers and HTTP parameters.
*/
private Response serveFile(String uri, Map<String, String> header,
File file, String mime) {
Response res;
try {
// Calculate etag
String etag = Integer.toHexString((file.getAbsolutePath()
+ file.lastModified() + "" + file.length()).hashCode());
// Support (simple) skipping:
long startFrom = 0;
long endAt = -1;
String range = header.get("range");
if (range != null) {
if (range.startsWith("bytes=")) {
range = range.substring("bytes=".length());
int minus = range.indexOf('-');
try {
if (minus > 0) {
startFrom = Long.parseLong(range
.substring(0, minus));
endAt = Long.parseLong(range.substring(minus + 1));
}
} catch (NumberFormatException ignored) {
}
}
}
// Change return code and add Content-Range header when skipping is
// requested
long fileLen = file.length();
if (range != null && startFrom >= 0) {
if (startFrom >= fileLen) {
res = createResponse(Response.Status.RANGE_NOT_SATISFIABLE,
NanoHTTPD.MIME_PLAINTEXT, "");
res.addHeader("Content-Range", "bytes 0-0/" + fileLen);
res.addHeader("ETag", etag);
} else {
if (endAt < 0) {
endAt = fileLen - 1;
}
long newLen = endAt - startFrom + 1;
if (newLen < 0) {
newLen = 0;
}
final long dataLen = newLen;
FileInputStream fis = new FileInputStream(file) {
@Override
public int available() throws IOException {
return (int) dataLen;
}
};
fis.skip(startFrom);
res = createResponse(Response.Status.PARTIAL_CONTENT, mime,
fis);
res.addHeader("Content-Length", "" + dataLen);
res.addHeader("Content-Range", "bytes " + startFrom + "-"
+ endAt + "/" + fileLen);
res.addHeader("ETag", etag);
}
} else {
if (etag.equals(header.get("if-none-match")))
res = createResponse(Response.Status.NOT_MODIFIED, mime, "");
else {
res = createResponse(Response.Status.OK, mime,
new FileInputStream(file));
res.addHeader("Content-Length", "" + fileLen);
res.addHeader("ETag", etag);
}
}
} catch (IOException ioe) {
res = createResponse(Response.Status.FORBIDDEN,
NanoHTTPD.MIME_PLAINTEXT, "FORBIDDEN: Reading file failed.");
}
return res;
}