MediawikiConnection.java: Difference between revisions

From Knot Atlas
Jump to navigationJump to search
mNo edit summary
(despammifying)
 
(5 intermediate revisions by 3 users not shown)
Line 9: Line 9:
import java.io.IOException;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStream;
import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;


import org.apache.commons.httpclient.DefaultHttpMethodRetryHandler;
import org.apache.commons.httpclient.DefaultHttpMethodRetryHandler;
import org.apache.commons.httpclient.HttpClient;
import org.apache.commons.httpclient.HttpClient;
import org.apache.commons.httpclient.HttpException;
import org.apache.commons.httpclient.HttpException;
import org.apache.commons.httpclient.HttpMethodBase;
import org.apache.commons.httpclient.NameValuePair;
import org.apache.commons.httpclient.NameValuePair;
import org.apache.commons.httpclient.methods.GetMethod;
import org.apache.commons.httpclient.methods.GetMethod;
Line 21: Line 29:
import org.apache.commons.httpclient.methods.multipart.StringPart;
import org.apache.commons.httpclient.methods.multipart.StringPart;
import org.apache.commons.httpclient.params.HttpMethodParams;
import org.apache.commons.httpclient.params.HttpMethodParams;
import org.apache.commons.lang.StringEscapeUtils;
import org.jdom.Document;
import org.jdom.Document;
import org.jdom.Element;
import org.jdom.Element;
Line 32: Line 41:
private String baseURL;
private String baseURL;
private HttpClient client = new HttpClient();
private String username, password;
// public only for debugging!
public HttpClient client;
public HttpMethodBase currentMethod;
public MediawikiConnection(String baseURL) {
public MediawikiConnection(String baseURL) {
this.baseURL = baseURL;
initialise(baseURL, "", "");
}
}
public MediawikiConnection(String baseURL, String username, String password) {
public MediawikiConnection(String baseURL, String username, String password) {
initialise(baseURL, username, password);
}
private void initialise(String baseURL, String username, String password) {
this.baseURL = baseURL;
this.baseURL = baseURL;
doLogin(username, password);
this.username = username;
this.password = password;
renewClient();
}
private boolean renewClient() {
if(client == null) {
client = new HttpClient();
client.getHttpConnectionManager().getParams().setConnectionTimeout(5000);
return doLogin(username, password);
}
return true;
}
private void disposeClient() {
client = null;
}
private boolean forceRenewClient() {
disposeClient();
return renewClient();
}
private void sleep() {
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
// yawn!!! Better get back to work.
e.printStackTrace();
}
}
}
private boolean doLogin(String username, String password) {
private boolean doLogin(String username, String password) {
if(username == "") return true;
PostMethod post = new PostMethod(baseURL + "?title=Special:Userlogin&action=submit");
PostMethod post = new PostMethod(baseURL + "?title=Special:Userlogin&action=submit");
// post.getParams().setCookiePolicy(CookiePolicy.BROWSER_COMPATIBILITY);
// post.getParams().setCookiePolicy(CookiePolicy.BROWSER_COMPATIBILITY);
currentMethod = post;
// enable automatic retrying
// enable automatic retrying
Line 64: Line 112:
post.setRequestBody(data);
post.setRequestBody(data);
int status;
int status = 1;
try {
try {
status = client.executeMethod(post);
status = client.executeMethod(post);
} catch (HttpException e) {
} catch (HttpException e) {
// uhoh... the HttpClient object is probably in a bad state.
return false;
// deal with this!!! I can't just dispose here... or can I?
e.printStackTrace();
disposeClient();
sleep();
} catch (IOException e) {
} catch (IOException e) {
return false;
// who cares
e.printStackTrace();
disposeClient();
sleep();
} finally {
} finally {
post.releaseConnection();
post.releaseConnection();
currentMethod = null;
}
}
return (status == 302);
return (status == 302);
Line 78: Line 134:


public String getPageText(String title) throws HttpException, IOException, JDOMException {
public String getPageText(String title) throws HttpException, IOException, JDOMException {
renewClient();
String URL = baseURL + "?title=Special:Export/" + title;
String URL = baseURL + "?title=Special:Export/" + title;
GetMethod get = new GetMethod(URL);
GetMethod get = new GetMethod(URL);
currentMethod = get;
// enable automatic retrying
// enable automatic retrying
Line 86: Line 145:
client.executeMethod(get);
client.executeMethod(get);

InputStream is = get.getResponseBodyAsStream();
InputStream is = get.getResponseBodyAsStream();
Line 97: Line 157:
// finally, make sure we release the connection!
// finally, make sure we release the connection!
get.releaseConnection();
get.releaseConnection();
currentMethod = null;
return text;
return text;
}
}
private String[][] mapToArray(Map map) {
public void setPageTextAsync(String title, String text) throws IOException {
String[][] array = new String[map.size()][2];
setPageTextAsync(title, text, "");
Iterator iter = map.keySet().iterator();
String title;
int i = 0;
while(iter.hasNext()) {
title = (String)iter.next();
array[i][0] = title;
array[i][1] = (String)map.get(title);
i++;
}
return array;
}
}
private Map arrayToMap(String[][] array) {
public boolean setPageText(String title, String text) throws IOException {
Map map = new HashMap();
return setPageText(title, text, "");
for(int i = 0; i < array.length; i++) {
map.put(array[i][0], array[i][1]);
}
return map;
}
}

public String[][] getPageTexts(String[] titles) {
public void setPageTextAsync(final String title, final String text, final String summary) throws IOException {
return mapToArray(getPageTexts(new HashSet(Arrays.asList(titles))));
new Thread() {
}
public void run() {
try {
public Map getPageTexts(Set titles) {
setPageText(title, text, summary);
renewClient();
} catch (IOException e) {
// hehe... who cares? No one's listening anyway.
Map pageTexts = new HashMap();
}
String titlesString = "";
Iterator iter = titles.iterator();
while(iter.hasNext()) {
titlesString = titlesString + (String)iter.next() + "\r\n";
}
String URL = baseURL + "?title=Special:Export";
PostMethod post = new PostMethod(URL);
currentMethod = post;
post.getParams().setParameter(HttpMethodParams.RETRY_HANDLER,
new DefaultHttpMethodRetryHandler(3, false));
post.getParams().setIntParameter(HttpMethodParams.SO_TIMEOUT, 5000);
NameValuePair[] data = {
new NameValuePair("action", "submit"),
new NameValuePair("curonly", "true"),
new NameValuePair("pages", titlesString),
};
post.setRequestBody(data);
InputStream is = null;
int status = 1;
try {
status = client.executeMethod(post);
is = post.getResponseBodyAsStream();
} catch (HttpException e) {
// uhoh... the HttpClient object is probably in a bad state.
e.printStackTrace();
disposeClient();
post.releaseConnection();
return pageTexts;
} catch (IOException e) {
// who cares
e.printStackTrace();
disposeClient();
post.releaseConnection();
return pageTexts;
}
SAXBuilder builder = new SAXBuilder();
try {
Document doc = builder.build(is);
Element page;
String title;
String text;
List pages = doc.getRootElement().getChildren("page");
iter = pages.iterator();
while(iter.hasNext()) {
page = (Element)iter.next();
text = page.getChild("revision").getChild("text").getText();
title = page.getChild("title").getText();
pageTexts.put(title, text);
}
}
} catch (IOException e) {
}.start();
e.printStackTrace();
disposeClient();
} catch (JDOMException e) {
e.printStackTrace();
}
post.releaseConnection();
return pageTexts;
}
}
public String[][] filterSetPageTexts(String[][] edits) {
return mapToArray(filterSetPageTexts(arrayToMap(edits)));
}
public Map filterSetPageTexts(Map edits) {
Map pageTexts = getPageTexts(edits.keySet());
Iterator iter = pageTexts.keySet().iterator();
String title;
while(iter.hasNext()) {
title = (String)iter.next();
if( ((String)edits.get(title)).equals((String)pageTexts.get(title)) ) {
edits.remove(title);
}
}
return edits;
}
private Map setPageTextsOnce(Map edits) {
Map failedEdits = new HashMap();
Iterator iter = edits.keySet().iterator();
String title, text;
while(iter.hasNext()) {
title = (String)iter.next();
text = (String)edits.get(title);
if(!setPageText(title, text)) {
failedEdits.put(title, text);
}
}
return failedEdits;
}
private Map setFilteredPageTexts(Map edits) {
return setPageTextsOnce(filterSetPageTexts(edits));
}
public Map setPageTexts(Map edits) {
return setFilteredPageTexts(setFilteredPageTexts(edits));
}
public String[][] setPageTexts(String[][] edits) {
return mapToArray(setPageTexts(arrayToMap(edits)));
}
public boolean setPageText(String title, String text) {
return setPageText(title, text, "");
}
public boolean setPageText(String title, String text, String summary) {
renewClient();
public boolean setPageText(String title, String text, String summary) throws IOException {
String URL = baseURL + "?title=" + title + "&action=edit";
String URL = baseURL + "?title=" + title + "&action=edit";
Line 127: Line 314:
String editTime = "";
String editTime = "";
String editToken = "";
String editToken = "";
String mungedWikiSource = "";
GetMethod get = new GetMethod(URL);
GetMethod get = new GetMethod(URL);
currentMethod = get;
// enable automatic retrying
// enable automatic retrying
get.getParams().setParameter(HttpMethodParams.RETRY_HANDLER,
get.getParams().setParameter(HttpMethodParams.RETRY_HANDLER,
new DefaultHttpMethodRetryHandler(3, false));
new DefaultHttpMethodRetryHandler(3, false));
client.executeMethod(get);
String response = "";
boolean failure = false;
try {
client.executeMethod(get);
response = get.getResponseBodyAsString();
} catch (HttpException e) {
// uhoh... the HttpClient object is probably in a bad state.
e.printStackTrace();
failure = true;
} catch (IOException e) {
// who cares
e.printStackTrace();
failure = true;
} finally {
get.releaseConnection();
currentMethod = null;
if(failure) {
disposeClient();
return false;
}
}
// now sift through the response looking for the editime and edittoken
// now sift through the response looking for the editime and edittoken
String response = get.getResponseBodyAsString();
get.releaseConnection();
java.util.regex.Pattern regex1 = java.util.regex.Pattern.compile(
java.util.regex.Pattern regex1 = java.util.regex.Pattern.compile(
"value=\"([0-9]+)\" name=\"wpEdittime\"");
"value=\"([0-9]+)\" name=\"wpEdittime\"");
Line 145: Line 354:
java.util.regex.Matcher matcher2 = regex2.matcher(response);
java.util.regex.Matcher matcher2 = regex2.matcher(response);
if(matcher2.find()) editToken = matcher2.group(1);
if(matcher2.find()) editToken = matcher2.group(1);
// while we're at it, let's get the (html-munged) wiki source as well.
java.util.regex.Pattern regex3 = java.util.regex.Pattern.compile(
"<textarea .*? name=\"wpTextbox1\" .*?>(.*?)</textarea>", java.util.regex.Pattern.DOTALL);
java.util.regex.Matcher matcher3 = regex3.matcher(response);
if(matcher3.find()) mungedWikiSource = matcher3.group(1);
String wikiSource = StringEscapeUtils.unescapeHtml(mungedWikiSource).trim();
if(wikiSource.equals(text.trim())) {
// don't need to do anything
return true;
}
PostMethod post = new PostMethod(URL);
PostMethod post = new PostMethod(URL);
currentMethod = post;
post.getParams().setParameter(HttpMethodParams.RETRY_HANDLER,
new DefaultHttpMethodRetryHandler(3, false));
post.getParams().setIntParameter(HttpMethodParams.SO_TIMEOUT, 5000);
NameValuePair[] data = {
NameValuePair[] data = {
Line 157: Line 383:
post.setRequestBody(data);
post.setRequestBody(data);
int status;
int status = 1;
try {
try {
System.out.println("Starting post, name: " + title);
status = client.executeMethod(post);
status = client.executeMethod(post);
System.out.println("Finished setPageText post, status: " + status);
System.out.println("Finished setPageText post, status: " + status);
System.out.println(post.getResponseBodyAsString());
System.out.println(post.getResponseBodyAsString());
} catch (HttpException e) {
} catch (HttpException e) {
// uhoh... the HttpClient object is probably in a bad state.
return false;
e.printStackTrace();
disposeClient();
} catch (IOException e) {
} catch (IOException e) {
// who cares
return false;
e.printStackTrace();
disposeClient();
} finally {
} finally {
// debug!
post.releaseConnection();
post.releaseConnection();
currentMethod = null;
}
}
Line 178: Line 411:
public boolean uploadFile(File f, String description) throws IOException {
public boolean uploadFile(File f, String description) throws IOException {
renewClient();
String URL = baseURL + "?title=Special:Upload";
String URL = baseURL + "?title=Special:Upload";
PostMethod upload = new PostMethod(URL);
PostMethod upload = new PostMethod(URL);
currentMethod = upload;
Part[] parts = {
Part[] parts = {
new StringPart("wpUploadDescription", description),
new StringPart("wpUploadDescription", description),
Line 190: Line 426:
upload.setRequestEntity( new MultipartRequestEntity(parts, upload.getParams()) );
upload.setRequestEntity( new MultipartRequestEntity(parts, upload.getParams()) );


int status;
int status = 1;
try {
try {
status = client.executeMethod(upload);
status = client.executeMethod(upload);
} catch (HttpException e) {
} catch (HttpException e) {
e.printStackTrace();
return false;
disposeClient();
} catch (IOException e) {
} catch (IOException e) {
e.printStackTrace();
return false;
disposeClient();
} finally {
} finally {
upload.releaseConnection();
upload.releaseConnection();
currentMethod = null;
}
}
Line 212: Line 451:
String editToken = "";
String editToken = "";
GetMethod get = new GetMethod(URL1);
GetMethod get = new GetMethod(URL1);
currentMethod = get;
// enable automatic retrying
// enable automatic retrying
get.getParams().setParameter(HttpMethodParams.RETRY_HANDLER,
get.getParams().setParameter(HttpMethodParams.RETRY_HANDLER,
Line 220: Line 460:
String response = get.getResponseBodyAsString();
String response = get.getResponseBodyAsString();
get.releaseConnection();
get.releaseConnection();
currentMethod = null;
java.util.regex.Pattern regex1 = java.util.regex.Pattern.compile(
java.util.regex.Pattern regex1 = java.util.regex.Pattern.compile(
"value=\"([0-9]+)\" name=\"wpEdittime\"");
"value=\"([0-9]+)\" name=\"wpEdittime\"");
Line 232: Line 473:
String URL2 = baseURL + "?title=Image:" + filename + "&action=delete" + "&image=" + filename;
String URL2 = baseURL + "?title=Image:" + filename + "&action=delete" + "&image=" + filename;
PostMethod post = new PostMethod(URL2);
PostMethod post = new PostMethod(URL2);
currentMethod = post;
// post.getParams().setCookiePolicy(CookiePolicy.BROWSER_COMPATIBILITY);
// post.getParams().setCookiePolicy(CookiePolicy.BROWSER_COMPATIBILITY);
Line 244: Line 486:
post.setRequestBody(data);
post.setRequestBody(data);
int status;
int status = 1;
try {
try {
status = client.executeMethod(post);
status = client.executeMethod(post);
} catch (HttpException e) {
} catch (HttpException e) {
e.printStackTrace();
return false;
} catch (IOException e) {
} catch (IOException e) {
e.printStackTrace();
return false;
} finally {
} finally {
post.releaseConnection();
post.releaseConnection();
currentMethod = null;
}
}
Line 261: Line 504:
public static void main(String[] args) throws HttpException, IOException, JDOMException {
public static void main(String[] args) throws HttpException, IOException, JDOMException {
MediawikiConnection conn1 = new MediawikiConnection(
MediawikiConnection conn1 = new MediawikiConnection(
"http://katlas.math.toronto.edu/w/index.php", "TestRobot", "foobar");
"http://katlas.math.toronto.edu/w/index.php", "TestRobot", "******");
MediawikiConnection conn2 = new MediawikiConnection(
"http://katlas.math.toronto.edu/w/index.php", "TestRobot", "foobar");
MediawikiConnection conn3 = new MediawikiConnection(
"http://katlas.math.toronto.edu/w/index.php", "TestRobot", "foobar");
// MediawikiConnection conn2 = new MediawikiConnection(
// MediawikiConnection conn2 = new MediawikiConnection(
// "http://math.berkeley.edu/~scott/w/index.php", "TestRobot", "foobar");
// "http://katlas.math.toronto.edu/w/index.php", "TestRobot", "******");
// MediawikiConnection conn3 = new MediawikiConnection(
// "http://katlas.math.toronto.edu/w/index.php", "TestRobot", "******");
// MediawikiConnection conn2 = new MediawikiConnection(
// "http://math.berkeley.edu/~scott/w/index.php", "TestRobot", "******");
// MediawikiConnection conn3 = new MediawikiConnection(
// MediawikiConnection conn3 = new MediawikiConnection(
// "http://katlas.math.toronto.edu/w/index.php", "ScottSysopRobot", "******");
// "http://katlas.math.toronto.edu/w/index.php", "ScottSysopRobot", "******");
// System.out.println(conn.getPageText("Main_Page"));
// System.out.println(conn.getPageText("Main_Page"));
// System.out.println(conn.setPageText("foo", "crel3", "auto!"));
// System.out.println(conn.setPageText("foo", "crel3", "auto!"));
conn1.setPageTextAsync("foo","async1! "+Math.random());
System.out.println(conn1.setPageText("foo","<math>1+2</math> "+Math.random()));
conn1.setPageTextAsync("foo","async2! "+Math.random());
System.out.println(conn1.setPageText("foo","<math>1+2</math> "));
conn1.setPageTextAsync("foo","async3! "+Math.random());
System.out.println(conn1.setPageText("foo","<math>1+2</math> "));
// System.out.println(conn.deleteFile("K7a4_m.gif","redundant, see [[Image:K7a4.gif]]"));
// System.out.println(conn.deleteFile("K7a4_m.gif","redundant, see [[Image:K7a4.gif]]"));
}
}
}
}
}
}

/*
/*
</pre>
</pre>
[[Category:Source Code]]
*/
*/

Latest revision as of 20:26, 20 February 2007

/* This page contains the source code for MediawikiConnection.java, the java component of Scott Morrison's WikiLink` package.

*/
package wikilink;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;

import org.apache.commons.httpclient.DefaultHttpMethodRetryHandler;
import org.apache.commons.httpclient.HttpClient;
import org.apache.commons.httpclient.HttpException;
import org.apache.commons.httpclient.HttpMethodBase;
import org.apache.commons.httpclient.NameValuePair;
import org.apache.commons.httpclient.methods.GetMethod;
import org.apache.commons.httpclient.methods.PostMethod;
import org.apache.commons.httpclient.methods.multipart.FilePart;
import org.apache.commons.httpclient.methods.multipart.MultipartRequestEntity;
import org.apache.commons.httpclient.methods.multipart.Part;
import org.apache.commons.httpclient.methods.multipart.StringPart;
import org.apache.commons.httpclient.params.HttpMethodParams;
import org.apache.commons.lang.StringEscapeUtils;
import org.jdom.Document;
import org.jdom.Element;
import org.jdom.JDOMException;
import org.jdom.input.SAXBuilder;

/**
 * @author Scott Morrison
 */
public class MediawikiConnection {
	
	private String baseURL;
	private String username, password;
	// public only for debugging!
	public HttpClient client;
	public HttpMethodBase currentMethod;
	
	public MediawikiConnection(String baseURL) {
		initialise(baseURL, "", "");
	}
	
	public MediawikiConnection(String baseURL, String username, String password) {
		initialise(baseURL, username, password);
	}
	
	private void initialise(String baseURL, String username, String password) {
		this.baseURL = baseURL;
		this.username = username;
		this.password = password;
		renewClient();
	}
		
	private boolean renewClient() {
		if(client == null) {
			client = new HttpClient();
			client.getHttpConnectionManager().getParams().setConnectionTimeout(5000);
			return doLogin(username, password);
		}
		return true;
	}
	
	private void disposeClient() {
		client = null;
	}
	
	private boolean forceRenewClient() {
		disposeClient();
		return renewClient();
	}
	
	private void sleep() {
		try {
			Thread.sleep(5000);
		} catch (InterruptedException e) {
			// yawn!!! Better get back to work.
			e.printStackTrace();
		}
	}
	
	private boolean doLogin(String username, String password) {
		if(username == "") return true;
		
		PostMethod post = new PostMethod(baseURL + "?title=Special:Userlogin&action=submit");
		// post.getParams().setCookiePolicy(CookiePolicy.BROWSER_COMPATIBILITY);
		currentMethod = post;
		
		// enable automatic retrying
		post.getParams().setParameter(HttpMethodParams.RETRY_HANDLER, 
	    		new DefaultHttpMethodRetryHandler(3, false));
		
		NameValuePair[] data = {
			new NameValuePair("wpName", username),
		    new NameValuePair("wpPassword", password),
		    new NameValuePair("wpRemember", "1"),
		    // ugh... this changed between 1.4beta5 and 1.4.7
		    // for 1.4beta5
		    new NameValuePair("wpLoginAttempt", "1"),
		    // for 1.4.7
		    new NameValuePair("wpLoginattempt", "Log in")
		};
		
		post.setRequestBody(data);
		
		int status = 1;
		try {
			status = client.executeMethod(post);
		} catch (HttpException e) {
			// uhoh... the HttpClient object is probably in a bad state.
			// deal with this!!! I can't just dispose here... or can I?
			e.printStackTrace();
			disposeClient();
			sleep();
		} catch (IOException e) {
			// who cares
			e.printStackTrace();
			disposeClient();
			sleep();
		} finally {
			post.releaseConnection();
			currentMethod = null;
		}
		return (status == 302);
	}

	public String getPageText(String title) throws HttpException, IOException, JDOMException {
		renewClient();
		
		String URL = baseURL + "?title=Special:Export/" + title;
		GetMethod get = new GetMethod(URL);
		currentMethod = get;
		
		// enable automatic retrying
		get.getParams().setParameter(HttpMethodParams.RETRY_HANDLER, 
	    		new DefaultHttpMethodRetryHandler(3, false));
		
		client.executeMethod(get);

		InputStream is = get.getResponseBodyAsStream();
		
		SAXBuilder builder = new SAXBuilder();
		Document doc = builder.build(is);
		Element page = doc.getRootElement().getChild("page");
		String text = "";
		if(page != null)
			text = page.getChild("revision").getChild("text").getText();

		// finally, make sure we release the connection!
		get.releaseConnection();
		currentMethod = null;
		
		return text;
	}
	
	private String[][] mapToArray(Map map) {
		String[][] array = new String[map.size()][2];
		Iterator iter = map.keySet().iterator();
		String title;
		int i = 0;
		while(iter.hasNext()) {
			title = (String)iter.next();
			array[i][0] = title;
			array[i][1] = (String)map.get(title);
			i++;
		}
		return array;
	}
	
	private Map arrayToMap(String[][] array) {
		Map map = new HashMap();
		for(int i = 0; i < array.length; i++) {
			map.put(array[i][0], array[i][1]);
		}
		return map;
	}
	
	public String[][] getPageTexts(String[] titles) {
		return mapToArray(getPageTexts(new HashSet(Arrays.asList(titles))));
	}
	
	public Map getPageTexts(Set titles) {
		renewClient();
		
		Map pageTexts = new HashMap();
		
		String titlesString = "";
		
		Iterator iter = titles.iterator();
		while(iter.hasNext()) {
			titlesString = titlesString + (String)iter.next() + "\r\n";
		}
		
		String URL = baseURL + "?title=Special:Export";
		PostMethod post = new PostMethod(URL);
		currentMethod = post;
		post.getParams().setParameter(HttpMethodParams.RETRY_HANDLER, 
	    		new DefaultHttpMethodRetryHandler(3, false));
		post.getParams().setIntParameter(HttpMethodParams.SO_TIMEOUT, 5000);
		
		NameValuePair[] data = {
			new NameValuePair("action", "submit"),
			new NameValuePair("curonly", "true"),
			new NameValuePair("pages", titlesString),			
		};
		
		post.setRequestBody(data);
		
		InputStream is = null;
		int status = 1;
		try {
			status = client.executeMethod(post);
			is = post.getResponseBodyAsStream();
		} catch (HttpException e) {
			// uhoh... the HttpClient object is probably in a bad state.
			e.printStackTrace();
			disposeClient();
			post.releaseConnection();
			return pageTexts;
		} catch (IOException e) {
			// who cares		
			e.printStackTrace();
			disposeClient();
			post.releaseConnection();
			return pageTexts;
		}
		
		SAXBuilder builder = new SAXBuilder();
		try {
			Document doc = builder.build(is);
			Element page;
			String title;
			String text;
			List pages = doc.getRootElement().getChildren("page");
			iter = pages.iterator();
			while(iter.hasNext()) {
				page = (Element)iter.next();
				text = page.getChild("revision").getChild("text").getText();
				title = page.getChild("title").getText();
				pageTexts.put(title, text);
			}
		} catch (IOException e) {
			e.printStackTrace();
			disposeClient();
		} catch (JDOMException e) {
			e.printStackTrace();
		}
		
		post.releaseConnection();
		return pageTexts;
	}
	
	public String[][] filterSetPageTexts(String[][] edits) {
		return mapToArray(filterSetPageTexts(arrayToMap(edits)));
	}
	
	public Map filterSetPageTexts(Map edits) {
		Map pageTexts = getPageTexts(edits.keySet());
		Iterator iter = pageTexts.keySet().iterator();
		String title;
		while(iter.hasNext()) {
			title = (String)iter.next();
			if( ((String)edits.get(title)).equals((String)pageTexts.get(title)) ) {
				edits.remove(title);
			}
		}
		return edits;
	}
	
	private Map setPageTextsOnce(Map edits) {
		Map failedEdits = new HashMap();
		Iterator iter = edits.keySet().iterator();
		String title, text;
		while(iter.hasNext()) {
			title = (String)iter.next();
			text = (String)edits.get(title);
			if(!setPageText(title, text)) {
				failedEdits.put(title, text);
			}
		}
		return failedEdits;
	}
	
	private Map setFilteredPageTexts(Map edits) {
		return setPageTextsOnce(filterSetPageTexts(edits));
	}
	
	public Map setPageTexts(Map edits) {
		return setFilteredPageTexts(setFilteredPageTexts(edits));
	}
	
	public String[][] setPageTexts(String[][] edits) {
		return mapToArray(setPageTexts(arrayToMap(edits)));
	}
	
	public boolean setPageText(String title, String text) {
		return setPageText(title, text, "");		
	}
		
	public boolean setPageText(String title, String text, String summary) {
		renewClient();
		
		String URL = baseURL + "?title=" + title + "&action=edit";
		
		// first, load the edit page, to get an 'edittime' and 'edittoken'
		String editTime = "";
		String editToken = "";
		String mungedWikiSource = "";
		GetMethod get = new GetMethod(URL);
		currentMethod = get;
		// enable automatic retrying
		get.getParams().setParameter(HttpMethodParams.RETRY_HANDLER, 
	    		new DefaultHttpMethodRetryHandler(3, false));
		
		String response = "";
		
		boolean failure = false;
		try {
			client.executeMethod(get);
			response = get.getResponseBodyAsString();			
		} catch (HttpException e) {
			// uhoh... the HttpClient object is probably in a bad state.
			e.printStackTrace();
			failure = true;
		} catch (IOException e) {
			// who cares
			e.printStackTrace();
			failure = true;
		} finally {
			get.releaseConnection();
			currentMethod = null;
			if(failure) {
				disposeClient();
				return false;
			}
		}
		
		// now sift through the response looking for the editime and edittoken
		java.util.regex.Pattern regex1 = java.util.regex.Pattern.compile(
				"value=\"([0-9]+)\" name=\"wpEdittime\"");
		java.util.regex.Matcher matcher1 = regex1.matcher(response);
		if(matcher1.find()) editTime = matcher1.group(1);

		java.util.regex.Pattern regex2 = java.util.regex.Pattern.compile(
			"value=\"([0-9a-z]+)\" name=\"wpEditToken\"");
		java.util.regex.Matcher matcher2 = regex2.matcher(response);
		if(matcher2.find()) editToken = matcher2.group(1);		
		
		// while we're at it, let's get the (html-munged) wiki source as well.
		java.util.regex.Pattern regex3 = java.util.regex.Pattern.compile(
			"<textarea .*? name=\"wpTextbox1\" .*?>(.*?)</textarea>", java.util.regex.Pattern.DOTALL);
		java.util.regex.Matcher matcher3 = regex3.matcher(response);
		if(matcher3.find()) mungedWikiSource = matcher3.group(1);			
		
		String wikiSource = StringEscapeUtils.unescapeHtml(mungedWikiSource).trim();
		
		if(wikiSource.equals(text.trim())) {
			// don't need to do anything
			return true;
		}
		
		PostMethod post = new PostMethod(URL);
		currentMethod = post;
		post.getParams().setParameter(HttpMethodParams.RETRY_HANDLER, 
	    		new DefaultHttpMethodRetryHandler(3, false));
		post.getParams().setIntParameter(HttpMethodParams.SO_TIMEOUT, 5000);
		
		NameValuePair[] data = {
			new NameValuePair("wpTextbox1", text),
			new NameValuePair("wpEdittime", editTime),
			new NameValuePair("wpEditToken", editToken),			
		    new NameValuePair("wpSummary", summary)
		};
			
		post.setRequestBody(data);
		
		int status = 1;
		try {
			System.out.println("Starting post, name: " + title);	
			status = client.executeMethod(post);
			System.out.println("Finished setPageText post, status: " + status);			
			System.out.println(post.getResponseBodyAsString());
		} catch (HttpException e) {
			// uhoh... the HttpClient object is probably in a bad state.
			e.printStackTrace();
			disposeClient();
		} catch (IOException e) {
			// who cares		
			e.printStackTrace();
			disposeClient();
		} finally {
			// debug!
			post.releaseConnection();
			currentMethod = null;
		}
	
		return (status == 302);
	}
	
	public boolean uploadFile(String filename, String description) throws IOException {
		return uploadFile(new File(filename), description);
	}
	
	public boolean uploadFile(File f, String description) throws IOException {
		renewClient();
		
		String URL = baseURL + "?title=Special:Upload";
		
		PostMethod upload = new PostMethod(URL);
		currentMethod = upload;
		Part[] parts = {
				new StringPart("wpUploadDescription", description),
				new StringPart("wpUploadAffirm", "1"),
				new StringPart("wpIgnoreWarning", "1"),
				new StringPart("wpUpload", "Upload file"),
				new FilePart("wpUploadFile", f.getName(), f)
		};
		upload.setRequestEntity( new MultipartRequestEntity(parts, upload.getParams()) );

		int status = 1;
		try {
			status = client.executeMethod(upload);
		} catch (HttpException e) {
			e.printStackTrace();
			disposeClient();
		} catch (IOException e) {
			e.printStackTrace();
			disposeClient();
		} finally {
			upload.releaseConnection();
			currentMethod = null;
		}
	
		return (status == 302);
	}	
	
	// deleteFile doesn't work! The mediawiki software says that something is wrong with the login session.
	public boolean deleteFile(String filename, String reason) throws HttpException, IOException {
		String URL1 = baseURL + "?title=Image:" + filename + "&action=delete";
				
		// first, load the edit page, to get an 'edittime' and 'edittoken'
		String editTime = "";
		String editToken = "";
		GetMethod get = new GetMethod(URL1);
		currentMethod = get;
		// enable automatic retrying
		get.getParams().setParameter(HttpMethodParams.RETRY_HANDLER, 
	    		new DefaultHttpMethodRetryHandler(3, false));
		client.executeMethod(get);
		
		// now sift through the response looking for the editime and edittoken
		String response = get.getResponseBodyAsString();
		get.releaseConnection();
		currentMethod = null;
		java.util.regex.Pattern regex1 = java.util.regex.Pattern.compile(
				"value=\"([0-9]+)\" name=\"wpEdittime\"");
		java.util.regex.Matcher matcher1 = regex1.matcher(response);
		if(matcher1.find()) editTime = matcher1.group(1);

		java.util.regex.Pattern regex2 = java.util.regex.Pattern.compile(
			"value=\"([0-9a-z]+)\" name=\"wpEditToken\"");
		java.util.regex.Matcher matcher2 = regex2.matcher(response);
		if(matcher2.find()) editToken = matcher2.group(1);		
		
		String URL2 = baseURL + "?title=Image:" + filename + "&action=delete" + "&image=" + filename;		
		PostMethod post = new PostMethod(URL2);
		currentMethod = post;
		// post.getParams().setCookiePolicy(CookiePolicy.BROWSER_COMPATIBILITY);
		
		NameValuePair[] data = {
			new NameValuePair("wpConfirm", "1"),
			// new NameValuePair("wpEdittime", editTime),			
		    new NameValuePair("wpReason", reason),
		    new NameValuePair("wpConfirmB", "Confirm"),
			new NameValuePair("wpEditToken", editToken)
		};
			
		post.setRequestBody(data);
		
		int status = 1;
		try {
			status = client.executeMethod(post);
		} catch (HttpException e) {
			e.printStackTrace();
		} catch (IOException e) {
			e.printStackTrace();
		} finally {
			post.releaseConnection();
			currentMethod = null;
		}
		
		return (status == 302);
	}
		
	public static class Test {
		public static void main(String[] args) throws HttpException, IOException, JDOMException {
			MediawikiConnection conn1 = new MediawikiConnection(
					"http://katlas.math.toronto.edu/w/index.php", "TestRobot", "******");
//			MediawikiConnection conn2 = new MediawikiConnection(
//					"http://katlas.math.toronto.edu/w/index.php", "TestRobot", "******");
//			MediawikiConnection conn3 = new MediawikiConnection(
//					"http://katlas.math.toronto.edu/w/index.php", "TestRobot", "******");
//			MediawikiConnection conn2 = new MediawikiConnection(
//					"http://math.berkeley.edu/~scott/w/index.php", "TestRobot", "******");
//			MediawikiConnection conn3 = new MediawikiConnection(
//					"http://katlas.math.toronto.edu/w/index.php", "ScottSysopRobot", "******");
//			System.out.println(conn.getPageText("Main_Page"));
//			System.out.println(conn.setPageText("foo", "crel3", "auto!"));
			System.out.println(conn1.setPageText("foo","<math>1+2</math> "+Math.random()));
			System.out.println(conn1.setPageText("foo","<math>1+2</math> "));
			System.out.println(conn1.setPageText("foo","<math>1+2</math> "));
//			System.out.println(conn.deleteFile("K7a4_m.gif","redundant, see [[Image:K7a4.gif]]"));
		}
	}
}

/*
  • /