/* * Overchan Android (Meta Imageboard Client) * Copyright (C) 2014-2016 miku-nyan <https://github.com/miku-nyan> * * This program 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. * * This program 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 this program. If not, see <http://www.gnu.org/licenses/>. */ package nya.miku.wishmaster.chans.infinity; import java.io.ByteArrayOutputStream; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.regex.Matcher; import java.util.regex.Pattern; import cz.msebera.android.httpclient.Header; import cz.msebera.android.httpclient.HttpHeaders; import cz.msebera.android.httpclient.NameValuePair; import cz.msebera.android.httpclient.client.entity.UrlEncodedFormEntity; import cz.msebera.android.httpclient.message.BasicHeader; import cz.msebera.android.httpclient.message.BasicNameValuePair; import cz.msebera.android.httpclient.util.TextUtils; import android.content.Context; import android.content.SharedPreferences; import android.content.res.Resources; import android.graphics.BitmapFactory; import android.graphics.drawable.Drawable; import android.preference.CheckBoxPreference; import android.preference.PreferenceGroup; import android.support.v4.content.res.ResourcesCompat; import nya.miku.wishmaster.R; import nya.miku.wishmaster.api.AbstractVichanModule; import nya.miku.wishmaster.api.interfaces.CancellableTask; import nya.miku.wishmaster.api.interfaces.ProgressListener; import nya.miku.wishmaster.api.models.BoardModel; import nya.miku.wishmaster.api.models.CaptchaModel; import nya.miku.wishmaster.api.models.DeletePostModel; import nya.miku.wishmaster.api.models.PostModel; import nya.miku.wishmaster.api.models.SendPostModel; import nya.miku.wishmaster.api.models.SimpleBoardModel; import nya.miku.wishmaster.api.models.ThreadModel; import nya.miku.wishmaster.api.models.UrlPageModel; import nya.miku.wishmaster.api.util.FastHtmlTagParser; import nya.miku.wishmaster.api.util.LazyPreferences; import nya.miku.wishmaster.common.IOUtils; import nya.miku.wishmaster.common.Logger; import nya.miku.wishmaster.http.ExtendedMultipartBuilder; import nya.miku.wishmaster.http.streamer.HttpRequestModel; import nya.miku.wishmaster.http.streamer.HttpResponseModel; import nya.miku.wishmaster.http.streamer.HttpStreamer; import nya.miku.wishmaster.http.streamer.HttpWrongStatusCodeException; import nya.miku.wishmaster.lib.base64.Base64; import nya.miku.wishmaster.lib.org_json.JSONException; import nya.miku.wishmaster.lib.org_json.JSONObject; public class InfinityModule extends AbstractVichanModule { private static final String TAG = "InfinityModule"; private static final String CHAN_NAME = "8chan"; private static final String DEFAULT_DOMAIN = "8ch.net"; private static final String ONION_DOMAIN = "oxwugzccvk3dk6tj.onion"; private static final String[] DOMAINS = new String[] { DEFAULT_DOMAIN, ONION_DOMAIN, "8chan.co" }; private static final String[] ATTACHMENT_FORMATS = new String[] { "jpg", "jpeg", "gif", "png", "webm", "mp4", "swf" }; private static final FastHtmlTagParser.TagReplaceHandler QUOTE_REPLACER = new FastHtmlTagParser.TagReplaceHandler() { @Override public FastHtmlTagParser.TagsPair replace(FastHtmlTagParser.TagsPair source) { if (source.openTag.equalsIgnoreCase("<p class=\"body-line ltr quote\">")) return new FastHtmlTagParser.TagsPair("<blockquote class=\"unkfunc\">", "</blockquote>"); return null; } }; private static final Pattern CAPTCHA_BASE64 = Pattern.compile("data:image/png;base64,([^\"]*)\""); private static final Pattern CAPTCHA_COOKIE = Pattern.compile("<input[^>]*name='captcha_cookie'[^>]*value='([^']*)'"); private static final Pattern CAPTCHA_ID = Pattern.compile("CAPTCHA ID: (.*?)<"); private static final Pattern ERROR_PATTERN = Pattern.compile("<h2 [^>]*>(.*?)</h2>"); private static final Pattern BAN_REASON_PATTERN = Pattern.compile("<p class=\"reason\">(.*?)</p>"); protected static final String PREF_KEY_USE_ONION = "PREF_KEY_USE_ONION"; private Map<String, BoardModel> boardsMap = new HashMap<>(); private boolean needTorCaptcha = false; private String torCaptchaCookie = null; private boolean needNewthreadCaptcha = false; private String newThreadCaptchaId = null; public InfinityModule(SharedPreferences preferences, Resources resources) { super(preferences, resources); } @Override public String getChanName() { return CHAN_NAME; } @Override public String getDisplayingName() { return "\u221Echan"; } @Override public Drawable getChanFavicon() { return ResourcesCompat.getDrawable(resources, R.drawable.favicon_8chan, null); } @Override protected String getCloudflareCookieDomain() { return DEFAULT_DOMAIN; } @Override public void addPreferencesOnScreen(PreferenceGroup preferenceGroup) { Context context = preferenceGroup.getContext(); addPasswordPreference(preferenceGroup); CheckBoxPreference httpsPref = addHttpsPreference(preferenceGroup, true); CheckBoxPreference onionPref = new LazyPreferences.CheckBoxPreference(context); onionPref.setTitle(R.string.pref_use_onion); onionPref.setSummary(R.string.pref_use_onion_summary); onionPref.setKey(getSharedKey(PREF_KEY_USE_ONION)); onionPref.setDefaultValue(false); onionPref.setDisableDependentsState(true); preferenceGroup.addPreference(onionPref); httpsPref.setDependency(getSharedKey(PREF_KEY_USE_ONION)); addProxyPreferences(preferenceGroup); } @Override protected boolean canCloudflare() { return true; } @Override protected String getUsingDomain() { return preferences.getBoolean(getSharedKey(PREF_KEY_USE_ONION), false) ? ONION_DOMAIN : DEFAULT_DOMAIN; } @Override protected String[] getAllDomains() { return DOMAINS; } @Override protected boolean useHttps() { return !preferences.getBoolean(getSharedKey(PREF_KEY_USE_ONION), false) && useHttps(true); } @Override public SimpleBoardModel[] getBoardsList(ProgressListener listener, CancellableTask task, SimpleBoardModel[] oldBoardsList) throws Exception { String url = getUsingUrl() + "boards.json"; HttpResponseModel responseModel = null; InfinityBoardsListReader in = null; HttpRequestModel rqModel = HttpRequestModel.builder().setGET().setCheckIfModified(oldBoardsList != null).build(); try { responseModel = HttpStreamer.getInstance().getFromUrl(url, rqModel, httpClient, listener, task); if (responseModel.statusCode == 200) { in = new InfinityBoardsListReader(responseModel.stream); if (task != null && task.isCancelled()) throw new Exception("interrupted"); return in.readBoardsList(); } else { if (responseModel.notModified()) return null; byte[] html = null; try { ByteArrayOutputStream byteStream = new ByteArrayOutputStream(1024); IOUtils.copyStream(responseModel.stream, byteStream); html = byteStream.toByteArray(); } catch (Exception e) {} throw new HttpWrongStatusCodeException(responseModel.statusCode, responseModel.statusCode + " - " + responseModel.statusReason, html); } } catch (HttpWrongStatusCodeException e) { checkCloudflareError(e, url); throw e; } catch (Exception e) { if (responseModel != null) HttpStreamer.getInstance().removeFromModifiedMap(url); throw e; } finally { IOUtils.closeQuietly(in); if (responseModel != null) responseModel.release(); } } @Override public BoardModel getBoard(String shortName, ProgressListener listener, CancellableTask task) throws Exception { BoardModel fromMap = boardsMap.get(shortName); if (fromMap != null) return fromMap; String url = getUsingUrl() + "settings.php?board=" + shortName; JSONObject json; try { json = downloadJSONObject(url, false, listener, task); } catch (Exception e) { json = new JSONObject(); } BoardModel model = new BoardModel(); model.chan = getChanName(); model.boardName = shortName; model.boardDescription = json.optString("title", shortName); model.uniqueAttachmentNames = true; model.timeZoneId = "US/Eastern"; model.defaultUserName = json.optString("anonymous", "Anonymous"); model.bumpLimit = json.optInt("reply_limit", 500); model.readonlyBoard = false; model.requiredFileForNewThread = false; model.allowDeletePosts = json.optBoolean("allow_delete", false); model.allowDeleteFiles = model.allowDeletePosts; model.allowNames = true; model.allowSubjects = true; model.allowSage = true; model.allowEmails = true; model.ignoreEmailIfSage = true; model.allowCustomMark = true; model.customMarkDescription = "Spoiler"; model.allowRandomHash = true; model.allowIcons = false; model.attachmentsMaxCount = json.optBoolean("disable_images", false) ? 0 : 5; model.attachmentsFormatFilters = ATTACHMENT_FORMATS; model.markType = BoardModel.MARK_NOMARK; model.firstPage = 1; model.lastPage = json.optInt("max_pages", BoardModel.LAST_PAGE_UNDEFINED); model.searchAllowed = false; model.catalogAllowed = true; boardsMap.put(shortName, model); return model; } @Override protected PostModel mapPostModel(JSONObject object, String boardName) { PostModel model = super.mapPostModel(object, boardName); try { model.comment = FastHtmlTagParser.getPTagParser().replace(model.comment, QUOTE_REPLACER); } catch (Exception e) {} return model; } @Override public ThreadModel[] getThreadsList(String boardName, int page, ProgressListener listener, CancellableTask task, ThreadModel[] oldList) throws Exception { try { return super.getThreadsList(boardName, page, listener, task, oldList); } catch (JSONException e) { if (page >= 3) throw new Exception("Back pages are disabled. Use the catalog to find threads on pages greater than 3.", e); throw e; } } @Override public CaptchaModel getNewCaptcha(String boardName, String threadNumber, ProgressListener listener, CancellableTask task) throws Exception { if (needTorCaptcha) { String url = getUsingUrl() + "dnsbls_bypass.php"; String response = HttpStreamer.getInstance().getStringFromUrl(url, HttpRequestModel.DEFAULT_GET, httpClient, listener, task, false); Matcher base64Matcher = CAPTCHA_BASE64.matcher(response); Matcher cookieMatcher = CAPTCHA_COOKIE.matcher(response); if (base64Matcher.find() && cookieMatcher.find()) { byte[] bitmap = Base64.decode(base64Matcher.group(1), Base64.DEFAULT); torCaptchaCookie = cookieMatcher.group(1); CaptchaModel captcha = new CaptchaModel(); captcha.type = CaptchaModel.TYPE_NORMAL; captcha.bitmap = BitmapFactory.decodeByteArray(bitmap, 0, bitmap.length); return captcha; } } if (needNewthreadCaptcha) { String url = getUsingUrl() + "8chan-captcha/entrypoint.php?mode=get&extra=abcdefghijklmnopqrstuvwxyz&nojs=true"; HttpRequestModel request = HttpRequestModel.builder().setGET(). setCustomHeaders(new Header[] { new BasicHeader(HttpHeaders.CACHE_CONTROL, "max-age=0") }).build(); String response = HttpStreamer.getInstance().getStringFromUrl(url, request, httpClient, listener, task, false); Matcher base64Matcher = CAPTCHA_BASE64.matcher(response); Matcher captchaIdMatcher = CAPTCHA_ID.matcher(response); if (base64Matcher.find() && captchaIdMatcher.find()) { byte[] bitmap = Base64.decode(base64Matcher.group(1), Base64.DEFAULT); newThreadCaptchaId = captchaIdMatcher.group(1); CaptchaModel captcha = new CaptchaModel(); captcha.type = CaptchaModel.TYPE_NORMAL; captcha.bitmap = BitmapFactory.decodeByteArray(bitmap, 0, bitmap.length); return captcha; } } return null; } private void checkCaptcha(String answer, CancellableTask task) throws Exception { try { if (torCaptchaCookie == null) throw new Exception("Invalid captcha"); String url = getUsingUrl() + "dnsbls_bypass.php"; List<NameValuePair> pairs = new ArrayList<NameValuePair>(); pairs.add(new BasicNameValuePair("captcha_text", answer)); pairs.add(new BasicNameValuePair("captcha_cookie", torCaptchaCookie)); HttpRequestModel rqModel = HttpRequestModel.builder().setPOST(new UrlEncodedFormEntity(pairs, "UTF-8")).setTimeout(30000).build(); String response = HttpStreamer.getInstance().getStringFromUrl(url, rqModel, httpClient, null, task, true); if (response.contains("Error") && !response.contains("Success")) throw new HttpWrongStatusCodeException(400, "400"); needTorCaptcha = false; } catch (HttpWrongStatusCodeException e) { if (task != null && task.isCancelled()) throw new InterruptedException("interrupted"); if (e.getStatusCode() == 400) throw new Exception("You failed the CAPTCHA"); throw e; } } @Override public String sendPost(SendPostModel model, ProgressListener listener, CancellableTask task) throws Exception { if (needTorCaptcha) checkCaptcha(model.captchaAnswer, task); if (task != null && task.isCancelled()) throw new InterruptedException("interrupted"); String url = getUsingUrl() + "post.php"; ExtendedMultipartBuilder postEntityBuilder = ExtendedMultipartBuilder.create().setDelegates(listener, task). addString("name", model.name). addString("email", model.sage ? "sage" : model.email). addString("subject", model.subject). addString("body", model.comment). addString("post", model.threadNumber == null ? "New Topic" : "New Reply"). addString("board", model.boardName); if (model.threadNumber != null) postEntityBuilder.addString("thread", model.threadNumber); if (model.custommark) postEntityBuilder.addString("spoiler", "on"); postEntityBuilder.addString("password", TextUtils.isEmpty(model.password) ? getDefaultPassword() : model.password); if (model.attachments != null) { String[] images = new String[] { "file", "file2", "file3", "file4", "file5" }; for (int i=0; i<model.attachments.length; ++i) { postEntityBuilder.addFile(images[i], model.attachments[i], model.randomHash); } } if (needNewthreadCaptcha) { postEntityBuilder.addString("captcha_text", model.captchaAnswer).addString("captcha_cookie", newThreadCaptchaId); needNewthreadCaptcha = false; } UrlPageModel refererPage = new UrlPageModel(); refererPage.chanName = getChanName(); refererPage.boardName = model.boardName; if (model.threadNumber == null) { refererPage.type = UrlPageModel.TYPE_BOARDPAGE; refererPage.boardPage = UrlPageModel.DEFAULT_FIRST_PAGE; } else { refererPage.type = UrlPageModel.TYPE_THREADPAGE; refererPage.threadNumber = model.threadNumber; } Header[] customHeaders = new Header[] { new BasicHeader(HttpHeaders.REFERER, buildUrl(refererPage)) }; HttpRequestModel request = HttpRequestModel.builder().setPOST(postEntityBuilder.build()).setCustomHeaders(customHeaders).setNoRedirect(true).build(); HttpResponseModel response = null; try { response = HttpStreamer.getInstance().getFromUrl(url, request, httpClient, listener, task); if (response.statusCode == 200) { Logger.d(TAG, "200 OK"); ByteArrayOutputStream output = new ByteArrayOutputStream(1024); IOUtils.copyStream(response.stream, output); String htmlResponse = output.toString("UTF-8"); if (htmlResponse.contains("<div class=\"ban\">")) { String error = "You are banned! ;_;"; Matcher banReasonMatcher = BAN_REASON_PATTERN.matcher(htmlResponse); if (banReasonMatcher.find()) { error += "\nReason: " + banReasonMatcher.group(1); } throw new Exception(error); } return null; } else if (response.statusCode == 303) { for (Header header : response.headers) { if (header != null && HttpHeaders.LOCATION.equalsIgnoreCase(header.getName())) { return fixRelativeUrl(header.getValue()); } } } else if (response.statusCode == 400) { ByteArrayOutputStream output = new ByteArrayOutputStream(1024); IOUtils.copyStream(response.stream, output); String htmlResponse = output.toString("UTF-8"); if (htmlResponse.contains("dnsbls_bypass.php")) { needTorCaptcha = true; throw new Exception("Please complete your CAPTCHA. (Bypass DNSBL)"); } else if (htmlResponse.contains("captcha_text") || htmlResponse.contains("entrypoint.php")) { needNewthreadCaptcha = true; throw new Exception(htmlResponse.contains("entrypoint.php") ? "You seem to have mistyped the verification, or your CAPTCHA expired. Please fill it out again." : "Please complete your CAPTCHA."); } else if (htmlResponse.contains("<h1>Error</h1>")) { Matcher errorMatcher = ERROR_PATTERN.matcher(htmlResponse); if (errorMatcher.find()) { String error = errorMatcher.group(1); if (error.contains("To post on 8chan over Tor, you must use the hidden service for security reasons.")) throw new Exception("To post on 8chan over Tor, you must use the onion domain."); throw new Exception(error); } } } throw new Exception(response.statusCode + " - " + response.statusReason); } finally { if (response != null) response.release(); } } @Override public String deletePost(DeletePostModel model, ProgressListener listener, CancellableTask task) throws Exception { String url = getUsingUrl() + "post.php"; List<NameValuePair> pairs = new ArrayList<NameValuePair>(); pairs.add(new BasicNameValuePair("board", model.boardName)); pairs.add(new BasicNameValuePair("delete_" + model.postNumber, "on")); if (model.onlyFiles) pairs.add(new BasicNameValuePair("file", "on")); pairs.add(new BasicNameValuePair("password", model.password)); pairs.add(new BasicNameValuePair("delete", "Delete")); pairs.add(new BasicNameValuePair("reason", "")); UrlPageModel refererPage = new UrlPageModel(); refererPage.type = UrlPageModel.TYPE_THREADPAGE; refererPage.chanName = getChanName(); refererPage.boardName = model.boardName; refererPage.threadNumber = model.threadNumber; Header[] customHeaders = new Header[] { new BasicHeader(HttpHeaders.REFERER, buildUrl(refererPage)) }; HttpRequestModel rqModel = HttpRequestModel.builder(). setPOST(new UrlEncodedFormEntity(pairs, "UTF-8")).setCustomHeaders(customHeaders).setNoRedirect(true).build(); HttpResponseModel response = null; try { response = HttpStreamer.getInstance().getFromUrl(url, rqModel, httpClient, listener, task); if (response.statusCode == 200 || response.statusCode == 303) { Logger.d(TAG, response.statusCode + " - " + response.statusReason); return null; } else if (response.statusCode == 400) { ByteArrayOutputStream output = new ByteArrayOutputStream(1024); IOUtils.copyStream(response.stream, output); String htmlResponse = output.toString("UTF-8"); if (htmlResponse.contains("dnsbls_bypass.php")) { needTorCaptcha = true; throw new Exception("Please complete your CAPTCHA.\n(try to post anything)"); } else if (htmlResponse.contains("<h1>Error</h1>")) { Matcher errorMatcher = ERROR_PATTERN.matcher(htmlResponse); if (errorMatcher.find()) { String error = errorMatcher.group(1); if (error.contains("To post on 8chan over Tor, you must use the hidden service for security reasons.")) throw new Exception(resources.getString(R.string.infinity_tor_message)); //? Tor users cannot into deleting throw new Exception(error); } } } throw new Exception(response.statusCode + " - " + response.statusReason); } finally { if (response != null) response.release(); } } }