Original code by Jeffrey Sharkey <jsharkey@android.com>. Modifications: - Remove .classpath, .project, and default.properties. - Remove generated files (/gen). - Create Android.mk and _index.html. - Cleaned up whitespace and converted to match AOSP style guide. - Renamed SimpleWiktionary to WiktionarySimple to keep both samples next to each other in the directory listing. - Removed the android:text attribute in the BulletPoint due to localization issues.
209 lines
8.0 KiB
Java
209 lines
8.0 KiB
Java
/*
|
|
* Copyright (C) 2009 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
package com.example.android.wiktionary;
|
|
|
|
import org.apache.http.HttpEntity;
|
|
import org.apache.http.HttpResponse;
|
|
import org.apache.http.StatusLine;
|
|
import org.apache.http.client.HttpClient;
|
|
import org.apache.http.client.methods.HttpGet;
|
|
import org.apache.http.impl.client.DefaultHttpClient;
|
|
import org.json.JSONArray;
|
|
import org.json.JSONException;
|
|
import org.json.JSONObject;
|
|
|
|
import android.content.Context;
|
|
import android.content.pm.PackageInfo;
|
|
import android.content.pm.PackageManager;
|
|
import android.content.pm.PackageManager.NameNotFoundException;
|
|
import android.net.Uri;
|
|
import android.util.Log;
|
|
|
|
import java.io.ByteArrayOutputStream;
|
|
import java.io.IOException;
|
|
import java.io.InputStream;
|
|
|
|
/**
|
|
* Helper methods to simplify talking with and parsing responses from a
|
|
* lightweight Wiktionary API. Before making any requests, you should call
|
|
* {@link #prepareUserAgent(Context)} to generate a User-Agent string based on
|
|
* your application package name and version.
|
|
*/
|
|
public class SimpleWikiHelper {
|
|
private static final String TAG = "SimpleWikiHelper";
|
|
|
|
/**
|
|
* Partial URL to use when requesting the detailed entry for a specific
|
|
* Wiktionary page. Use {@link String#format(String, Object...)} to insert
|
|
* the desired page title after escaping it as needed.
|
|
*/
|
|
private static final String WIKTIONARY_PAGE =
|
|
"http://en.wiktionary.org/w/api.php?action=query&prop=revisions&titles=%s&" +
|
|
"rvprop=content&format=json%s";
|
|
|
|
/**
|
|
* Partial URL to append to {@link #WIKTIONARY_PAGE} when you want to expand
|
|
* any templates found on the requested page. This is useful when browsing
|
|
* full entries, but may use more network bandwidth.
|
|
*/
|
|
private static final String WIKTIONARY_EXPAND_TEMPLATES =
|
|
"&rvexpandtemplates=true";
|
|
|
|
/**
|
|
* {@link StatusLine} HTTP status code when no server error has occurred.
|
|
*/
|
|
private static final int HTTP_STATUS_OK = 200;
|
|
|
|
/**
|
|
* Shared buffer used by {@link #getUrlContent(String)} when reading results
|
|
* from an API request.
|
|
*/
|
|
private static byte[] sBuffer = new byte[512];
|
|
|
|
/**
|
|
* User-agent string to use when making requests. Should be filled using
|
|
* {@link #prepareUserAgent(Context)} before making any other calls.
|
|
*/
|
|
private static String sUserAgent = null;
|
|
|
|
/**
|
|
* Thrown when there were problems contacting the remote API server, either
|
|
* because of a network error, or the server returned a bad status code.
|
|
*/
|
|
public static class ApiException extends Exception {
|
|
public ApiException(String detailMessage, Throwable throwable) {
|
|
super(detailMessage, throwable);
|
|
}
|
|
|
|
public ApiException(String detailMessage) {
|
|
super(detailMessage);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Thrown when there were problems parsing the response to an API call,
|
|
* either because the response was empty, or it was malformed.
|
|
*/
|
|
public static class ParseException extends Exception {
|
|
public ParseException(String detailMessage, Throwable throwable) {
|
|
super(detailMessage, throwable);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Prepare the internal User-Agent string for use. This requires a
|
|
* {@link Context} to pull the package name and version number for this
|
|
* application.
|
|
*/
|
|
public static void prepareUserAgent(Context context) {
|
|
try {
|
|
// Read package name and version number from manifest
|
|
PackageManager manager = context.getPackageManager();
|
|
PackageInfo info = manager.getPackageInfo(context.getPackageName(), 0);
|
|
sUserAgent = String.format(context.getString(R.string.template_user_agent),
|
|
info.packageName, info.versionName);
|
|
|
|
} catch(NameNotFoundException e) {
|
|
Log.e(TAG, "Couldn't find package information in PackageManager", e);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Read and return the content for a specific Wiktionary page. This makes a
|
|
* lightweight API call, and trims out just the page content returned.
|
|
* Because this call blocks until results are available, it should not be
|
|
* run from a UI thread.
|
|
*
|
|
* @param title The exact title of the Wiktionary page requested.
|
|
* @param expandTemplates If true, expand any wiki templates found.
|
|
* @return Exact content of page.
|
|
* @throws ApiException If any connection or server error occurs.
|
|
* @throws ParseException If there are problems parsing the response.
|
|
*/
|
|
public static String getPageContent(String title, boolean expandTemplates)
|
|
throws ApiException, ParseException {
|
|
// Encode page title and expand templates if requested
|
|
String encodedTitle = Uri.encode(title);
|
|
String expandClause = expandTemplates ? WIKTIONARY_EXPAND_TEMPLATES : "";
|
|
|
|
// Query the API for content
|
|
String content = getUrlContent(String.format(WIKTIONARY_PAGE,
|
|
encodedTitle, expandClause));
|
|
try {
|
|
// Drill into the JSON response to find the content body
|
|
JSONObject response = new JSONObject(content);
|
|
JSONObject query = response.getJSONObject("query");
|
|
JSONObject pages = query.getJSONObject("pages");
|
|
JSONObject page = pages.getJSONObject((String) pages.keys().next());
|
|
JSONArray revisions = page.getJSONArray("revisions");
|
|
JSONObject revision = revisions.getJSONObject(0);
|
|
return revision.getString("*");
|
|
} catch (JSONException e) {
|
|
throw new ParseException("Problem parsing API response", e);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Pull the raw text content of the given URL. This call blocks until the
|
|
* operation has completed, and is synchronized because it uses a shared
|
|
* buffer {@link #sBuffer}.
|
|
*
|
|
* @param url The exact URL to request.
|
|
* @return The raw content returned by the server.
|
|
* @throws ApiException If any connection or server error occurs.
|
|
*/
|
|
protected static synchronized String getUrlContent(String url) throws ApiException {
|
|
if (sUserAgent == null) {
|
|
throw new ApiException("User-Agent string must be prepared");
|
|
}
|
|
|
|
// Create client and set our specific user-agent string
|
|
HttpClient client = new DefaultHttpClient();
|
|
HttpGet request = new HttpGet(url);
|
|
request.setHeader("User-Agent", sUserAgent);
|
|
|
|
try {
|
|
HttpResponse response = client.execute(request);
|
|
|
|
// Check if server response is valid
|
|
StatusLine status = response.getStatusLine();
|
|
if (status.getStatusCode() != HTTP_STATUS_OK) {
|
|
throw new ApiException("Invalid response from server: " +
|
|
status.toString());
|
|
}
|
|
|
|
// Pull content stream from response
|
|
HttpEntity entity = response.getEntity();
|
|
InputStream inputStream = entity.getContent();
|
|
|
|
ByteArrayOutputStream content = new ByteArrayOutputStream();
|
|
|
|
// Read response into a buffered stream
|
|
int readBytes = 0;
|
|
while ((readBytes = inputStream.read(sBuffer)) != -1) {
|
|
content.write(sBuffer, 0, readBytes);
|
|
}
|
|
|
|
// Return result from buffered stream
|
|
return new String(content.toByteArray());
|
|
} catch (IOException e) {
|
|
throw new ApiException("Problem communicating with API", e);
|
|
}
|
|
}
|
|
|
|
}
|