/* * 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.wakachan; import java.io.ByteArrayOutputStream; import java.io.InputStream; import java.text.DateFormat; import java.text.SimpleDateFormat; import java.util.ArrayList; import java.util.List; import java.util.Locale; import java.util.TimeZone; import cz.msebera.android.httpclient.NameValuePair; import cz.msebera.android.httpclient.client.entity.UrlEncodedFormEntity; import cz.msebera.android.httpclient.message.BasicNameValuePair; import android.content.SharedPreferences; import android.content.res.Resources; import android.graphics.drawable.Drawable; import android.support.v4.content.res.ResourcesCompat; import nya.miku.wishmaster.R; import nya.miku.wishmaster.api.AbstractWakabaModule; 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.SendPostModel; import nya.miku.wishmaster.api.models.SimpleBoardModel; import nya.miku.wishmaster.api.models.UrlPageModel; import nya.miku.wishmaster.api.util.ChanModels; import nya.miku.wishmaster.api.util.RegexUtils; import nya.miku.wishmaster.api.util.WakabaReader; import nya.miku.wishmaster.common.IOUtils; 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; public class WakachanModule extends AbstractWakabaModule { private static final String CHAN_NAME = "wakachan.org"; private static final String[] DOMAINS = new String[] { "www.wakachan.org", "wakachan.org" }; private static final SimpleBoardModel[] BOARDS = new SimpleBoardModel[] { ChanModels.obtainSimpleBoardModel(CHAN_NAME, "dress", "Fancy Clothes", "Experiment", false), ChanModels.obtainSimpleBoardModel(CHAN_NAME, "unyl", "Russia!", "DQN", true), ChanModels.obtainSimpleBoardModel(CHAN_NAME, "anime", "Anime", "General Anime", false), ChanModels.obtainSimpleBoardModel(CHAN_NAME, "mai", "Maid & Uniform", "Special Interest", false), ChanModels.obtainSimpleBoardModel(CHAN_NAME, "os", "Net Characters", "Special Interest", false), ChanModels.obtainSimpleBoardModel(CHAN_NAME, "elf", "Pointy Ears", "Special Interest", false), ChanModels.obtainSimpleBoardModel(CHAN_NAME, "yuu", "Scenic Route", "Special Interest", false), ChanModels.obtainSimpleBoardModel(CHAN_NAME, "miz", "Swimsuits", "Special Interest", false), ChanModels.obtainSimpleBoardModel(CHAN_NAME, "azu", "Azumanga", "Artist/Series", false), ChanModels.obtainSimpleBoardModel(CHAN_NAME, "fate", "Type-Moon", "Artist/Series", false), ChanModels.obtainSimpleBoardModel(CHAN_NAME, "ff", "Females", "Adult", true), ChanModels.obtainSimpleBoardModel(CHAN_NAME, "mf", "Male/Female", "Adult", true), ChanModels.obtainSimpleBoardModel(CHAN_NAME, "happy", "Happy Sex", "Adult", true) }; private static final DateFormat DATE_FORMAT, DATE_FORMAT_UNYL; static { DATE_FORMAT = new SimpleDateFormat("yy/MM/dd(EEE)HH:mm", Locale.US); DATE_FORMAT.setTimeZone(TimeZone.getTimeZone("US/Pacific")); DATE_FORMAT_UNYL = new SimpleDateFormat("yy/MM/dd(EEE)HH:mm", Locale.US); DATE_FORMAT_UNYL.setTimeZone(TimeZone.getTimeZone("GMT+4")); } public WakachanModule(SharedPreferences preferences, Resources resources) { super(preferences, resources); } @Override public String getChanName() { return CHAN_NAME; } @Override public String getDisplayingName() { return "Wakachan"; } @Override public Drawable getChanFavicon() { return ResourcesCompat.getDrawable(resources, R.drawable.favicon_cirno, null); } @Override protected String getUsingDomain() { return DOMAINS[0]; } @Override protected String[] getAllDomains() { return DOMAINS; } @Override protected boolean canHttps() { return true; } @Override protected boolean canCloudflare() { return true; } @Override protected WakabaReader getWakabaReader(InputStream stream, UrlPageModel urlModel) { return new WakabaReader(stream, urlModel.boardName.equals("unyl") ? DATE_FORMAT_UNYL : DATE_FORMAT, canCloudflare()) { @Override protected void parseOmittedString(String omitted) { if (omitted.indexOf('>') != -1) omitted = omitted.substring(omitted.indexOf('>')); super.parseOmittedString(omitted); } }; } @Override protected SimpleBoardModel[] getBoardsList() { return BOARDS; } @Override public BoardModel getBoard(String shortName, ProgressListener listener, CancellableTask task) throws Exception { BoardModel model = super.getBoard(shortName, listener, task); model.timeZoneId = shortName.equals("unyl") ? "GMT+4" : "US/Pacific"; model.readonlyBoard = false; model.requiredFileForNewThread = false; model.allowDeletePosts = true; model.allowDeleteFiles = true; model.allowReport = BoardModel.REPORT_SIMPLE; model.allowNames = true; model.allowSubjects = true; model.allowSage = false; model.allowEmails = false; model.allowCustomMark = false; model.allowRandomHash = true; model.allowIcons = false; model.attachmentsMaxCount = 1; model.attachmentsFormatFilters = null; model.markType = BoardModel.MARK_WAKABAMARK; return model; } @Override public CaptchaModel getNewCaptcha(String boardName, String threadNumber, ProgressListener listener, CancellableTask task) throws Exception { String captchaUrl = getUsingUrl() + boardName + "/captcha.pl" + "?key=" + (threadNumber == null ? "mainpage" : ("res" + threadNumber)); return downloadCaptcha(captchaUrl, listener, task); } @Override public String sendPost(SendPostModel model, ProgressListener listener, CancellableTask task) throws Exception { String url = getUsingUrl() + model.boardName + "/wakaba.pl"; ExtendedMultipartBuilder postEntityBuilder = ExtendedMultipartBuilder.create().setDelegates(listener, task). addString("task", "post"); if (model.threadNumber != null) postEntityBuilder.addString("parent", model.threadNumber); postEntityBuilder.addString("password", model.password); if (model.attachments != null && model.attachments.length > 0) postEntityBuilder.addFile("file", model.attachments[0], model.randomHash); postEntityBuilder. addString("field1", model.name). addString("field3", model.subject). addString("field4", model.comment). addString("captcha", model.captchaAnswer); HttpRequestModel request = HttpRequestModel.builder().setPOST(postEntityBuilder.build()).setNoRedirect(true).build(); HttpResponseModel response = null; try { response = HttpStreamer.getInstance().getFromUrl(url, request, httpClient, null, task); if (response.statusCode == 303) { return null; } else if (response.statusCode == 200) { ByteArrayOutputStream output = new ByteArrayOutputStream(1024); IOUtils.copyStream(response.stream, output); String htmlResponse = output.toString("UTF-8"); if (!htmlResponse.contains("<blockquote")) { int start = htmlResponse.indexOf("<h1 style=\"text-align: center\">"); if (start != -1) { int end = htmlResponse.indexOf("<", start + 31); if (end != -1) { throw new Exception(htmlResponse.substring(start + 31, end).trim()); } } start = htmlResponse.indexOf("<h1>"); if (start != -1) { int end = htmlResponse.indexOf("</h1>", start + 4); if (end != -1) { throw new Exception(htmlResponse.substring(start + 4, end).trim()); } } } } else throw new Exception(response.statusCode + " - " + response.statusReason); } finally { if (response != null) response.release(); } return null; } @Override public String deletePost(DeletePostModel model, ProgressListener listener, CancellableTask task) throws Exception { String url = getUsingUrl() + model.boardName + "/wakaba.pl"; List<NameValuePair> pairs = new ArrayList<NameValuePair>(); pairs.add(new BasicNameValuePair("delete", model.postNumber)); pairs.add(new BasicNameValuePair("task", "delete")); if (model.onlyFiles) pairs.add(new BasicNameValuePair("fileonly", "on")); pairs.add(new BasicNameValuePair("password", model.password)); HttpRequestModel request = HttpRequestModel.builder().setPOST(new UrlEncodedFormEntity(pairs, "UTF-8")).setNoRedirect(true).build(); HttpResponseModel response = null; try { response = HttpStreamer.getInstance().getFromUrl(url, request, httpClient, null, task); if (response.statusCode == 200) { ByteArrayOutputStream output = new ByteArrayOutputStream(1024); IOUtils.copyStream(response.stream, output); String htmlResponse = output.toString("UTF-8"); if (!htmlResponse.contains("<blockquote")) { int start = htmlResponse.indexOf("<h1 style=\"text-align: center\">"); if (start != -1) { int end = htmlResponse.indexOf("<", start + 31); if (end != -1) { throw new Exception(htmlResponse.substring(start + 31, end).trim()); } } } } } finally { if (response != null) response.release(); } return null; } @Override public String reportPost(DeletePostModel model, ProgressListener listener, CancellableTask task) throws Exception { String url = getUsingUrl() + "banned/report.pl?url=/" + model.boardName + "/res/" + model.threadNumber + "%23" + model.postNumber; String response = RegexUtils.removeHtmlTags(HttpStreamer.getInstance().getStringFromUrl( url, HttpRequestModel.DEFAULT_GET, getHttpClient(), listener, task, false)).trim(); if (response.startsWith("Post reported.")) return null; throw new Exception(response); } }