ctbrec/common/src/main/java/ctbrec/sites/xlovecam/XloveCamHttpClient.java

202 lines
7.9 KiB
Java

package ctbrec.sites.xlovecam;
import static ctbrec.io.HttpConstants.*;
import static java.nio.charset.StandardCharsets.*;
import java.io.IOException;
import java.util.Arrays;
import java.util.Base64;
import java.util.Locale;
import java.util.Random;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import org.json.JSONObject;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import ctbrec.Config;
import ctbrec.io.HttpClient;
import ctbrec.io.HttpException;
import okhttp3.FormBody;
import okhttp3.Request;
import okhttp3.RequestBody;
import okhttp3.Response;
public class XloveCamHttpClient extends HttpClient {
private static final Logger LOG = LoggerFactory.getLogger(XloveCamHttpClient.class);
private static final Pattern CSRF_PATTERN = Pattern.compile("CSRFToken\\s*=\\s*\"(.*?)\";");
private final Random rng = new Random();
public XloveCamHttpClient() {
super("xlovecam");
}
@Override
public boolean login() throws IOException {
String username = Config.getInstance().getSettings().xlovecamUsername;
String csrfToken = getCsrfToken();
JSONObject config = getConfig();
String token = config.getString("token");
byte[] passwordKey = getPasswordKey(config);
byte[] encryptedPassword = encryptPassword(Config.getInstance().getSettings().xlovecamPassword, passwordKey);
String base64EncryptedPassword = Base64.getEncoder().encodeToString(encryptedPassword);
LOG.debug("csrf:{} token:{} key:{}", csrfToken, token, Arrays.toString(passwordKey));
LOG.debug("encrypted password: {}", base64EncryptedPassword);
long time = System.currentTimeMillis() / 1000;
long rnd = rng.nextInt(100_000_000);
String rndTokenString = token + ':' + Long.toString(rnd) + ":\u20ac\u2716\u21aa:" + time + ':' + username;
long rndToken = fnv32a(rndTokenString.getBytes(UTF_8));
RequestBody body = new FormBody.Builder() // @formatter:off
.add("pseudo", username)
.add("passwd", "")
.add("keep", "1")
.add("isLayer", "1")
.add("token", token)
.add("grecaptchaLoaded", "0")
.add("extra[screen][width]", "1920")
.add("extra[screen][height]", "1080")
.add("extra[tmz]", "0")
.add("extra[blng]", "en")
.add("extra[clientTimezone]", "GMT")
.add("extra[flash][installed]", "false")
.add("extra[flash][raw]", "")
.add("extra[flash][major]", "-1")
.add("extra[flash][minor]", "-1")
.add("extra[flash][revision]", "-1")
.add("extra[flash][revisionStr]", "")
.add("extra[rnd][rnd]", Long.toString(rnd))
.add("extra[rnd][time]", Long.toString(time))
.add("extra[rnd][token]", Integer.toString((int)rndToken))
.add("csrf", csrfToken)
.add("passwdEncoded", "b64:" + base64EncryptedPassword)
.add("g-recaptcha-response", "")
.build(); // @formatter:on
String url = XloveCam.baseUrl + "/en/ajax/client/login2";
Request req = new Request.Builder()
.url(url)
.method("POST", body)
.header(ACCEPT, MIMETYPE_APPLICATION_JSON)
.header(ACCEPT_LANGUAGE, Locale.ENGLISH.getLanguage())
.header(REFERER, XloveCam.baseUrl + "/en/login")
.header(ORIGIN, XloveCam.baseUrl)
.header(USER_AGENT, Config.getInstance().getSettings().httpUserAgent)
.header(CONTENT_TYPE, FORM_ENCODED)
.header(X_REQUESTED_WITH, XML_HTTP_REQUEST)
.build();
try (Response resp = execute(req)) {
if (resp.isSuccessful()) {
String msg = resp.body().string();
JSONObject json = new JSONObject(msg);
LOG.debug(json.toString(2));
return json.optBoolean("success");
} else {
throw new HttpException(resp.code(), resp.message());
}
}
}
private long fnv32a(byte[] a) {
long b = 2166136261l;
for (int d = 0; d < a.length; d++) {
b ^= (a[d] & 0xFF);
b += (b << 1) + (b << 4) + (b << 7) + (b << 8) + (b << 24) & 4294967295l;
}
return b >>> 0;
}
private byte[] encryptPassword(String xlovecamPassword, byte[] passwordKey) {
byte[] password = xlovecamPassword.getBytes(UTF_8);
byte d = (byte) password.length;
byte[] c = new byte[password.length + rng.nextInt(20)];
System.arraycopy(password, 0, c, 0, password.length);
for (int i = password.length; i < c.length; i++) {
c[i] = (byte)rng.nextInt(255);
}
byte[] b = new byte[c.length + 1];
b[0] = (byte)(d ^ (passwordKey[0] & 0xFF));
d = 1;
for (int e = 0; e < c.length; e++) {
b[e+1] = (byte)(c[e] ^ passwordKey[d]);
d++;
if(d >= passwordKey.length) {
d = 0;
}
}
return b;
}
private byte[] getPasswordKey(JSONObject config) {
String passwordKeyString = config.getString("passwordKey");
LOG.debug(passwordKeyString);
String[] numbers = passwordKeyString.split(",");
byte[] passwordKey = new byte[numbers.length];
for (int i = 0; i < numbers.length; i++) {
passwordKey[i] = (byte)(Integer.parseInt(numbers[i]) & 0xFF);
}
return passwordKey;
}
private String getCsrfToken() throws IOException {
Request req = new Request.Builder()
.url(XloveCam.baseUrl)
.header(ACCEPT_LANGUAGE, Locale.ENGLISH.getLanguage())
.header(USER_AGENT, Config.getInstance().getSettings().httpUserAgent)
.build();
try (Response resp = execute(req)) {
if (resp.isSuccessful()) {
String body = resp.body().string();
Matcher m = CSRF_PATTERN.matcher(body);
if (m.find()) {
return m.group(1);
} else {
throw new IOException("CSRF token not found in landing page");
}
} else {
throw new HttpException(resp.code(), resp.message());
}
}
}
private JSONObject getConfig() throws IOException {
String url = XloveCam.baseUrl + "/en/popup/login";
LOG.debug("Calling {}", url);
RequestBody body = new FormBody.Builder()
.add("referrer", "https://www.xlovecam.com/en/")
.add("referrer_is_layer", "0")
.add("referrer_model_id", "")
.add("referrer_model_name", "")
.build();
Request req = new Request.Builder()
.url(url)
.method("POST", body)
.header(ACCEPT, MIMETYPE_APPLICATION_JSON)
.header(ACCEPT_LANGUAGE, Locale.ENGLISH.getLanguage())
.header(REFERER, XloveCam.baseUrl)
.header(ORIGIN, XloveCam.baseUrl)
.header(USER_AGENT, Config.getInstance().getSettings().httpUserAgent)
.header(CONTENT_TYPE, FORM_ENCODED)
.header(X_REQUESTED_WITH, XML_HTTP_REQUEST)
.build();
try (Response resp = execute(req)) {
if (resp.isSuccessful()) {
String msg = resp.body().string();
JSONObject json = new JSONObject(msg);
if (json.has("config")) {
return json.getJSONObject("config");
} else {
throw new IOException("JSON doesn't contain \"config\":\n" + json.toString(2));
}
} else {
throw new HttpException(resp.code(), resp.message());
}
}
}
}