"allDeclaredMethods":true,
"allDeclaredConstructors":true}
,
+{
+ "name":"org.asamk.signal.commands.RegisterCommand$RegistrationParams",
+ "allDeclaredFields":true,
+ "queryAllDeclaredMethods":true,
+ "queryAllDeclaredConstructors":true,
+ "methods":[{"name":"<init>","parameterTypes":["java.lang.Boolean","java.lang.String"] }]}
+,
+{
+ "name":"org.asamk.signal.commands.VerifyCommand$VerifyParams",
+ "allDeclaredFields":true,
+ "queryAllDeclaredMethods":true,
+ "queryAllDeclaredConstructors":true,
+ "methods":[{"name":"<init>","parameterTypes":["java.lang.String","java.lang.String"] }]}
+,
{
"name":"org.asamk.signal.json.JsonAttachment",
"allDeclaredFields":true,
{"name":"type_"}
]}
,
+{
+ "name":"org.whispersystems.signalservice.internal.push.SignalServiceProtos$SyncMessage$Groups",
+ "fields":[
+ {"name":"bitField0_"},
+ {"name":"blob_"}
+ ]}
+,
{
"name":"org.whispersystems.signalservice.internal.push.SignalServiceProtos$SyncMessage$Keys",
"fields":[
}
SLF4JBridgeHandler.removeHandlersForRootLogger();
SLF4JBridgeHandler.install();
- // java.util.logging.Logger.getLogger("").setLevel(java.util.logging.Level.FINEST);
+ java.util.logging.Logger.getLogger("").setLevel(java.util.logging.Level.FINEST);
}
private static int getStatusForError(final CommandException e) {
--- /dev/null
+package org.asamk.signal.commands;
+
+import org.asamk.signal.JsonWriter;
+import org.asamk.signal.commands.exceptions.CommandException;
+import org.asamk.signal.manager.RegistrationManager;
+
+public interface JsonRpcRegistrationCommand<T> extends JsonRpcCommand<T> {
+
+ void handleCommand(T request, RegistrationManager m, JsonWriter jsonWriter) throws CommandException;
+}
package org.asamk.signal.commands;
+import com.fasterxml.jackson.core.type.TypeReference;
+
import net.sourceforge.argparse4j.impl.Arguments;
import net.sourceforge.argparse4j.inf.Namespace;
import net.sourceforge.argparse4j.inf.Subparser;
+import org.asamk.signal.JsonWriter;
+import org.asamk.signal.OutputType;
import org.asamk.signal.commands.exceptions.CommandException;
import org.asamk.signal.commands.exceptions.IOErrorException;
import org.asamk.signal.commands.exceptions.UserErrorException;
import org.asamk.signal.manager.api.CaptchaRequiredException;
import java.io.IOException;
+import java.util.List;
-public class RegisterCommand implements RegistrationCommand {
+public class RegisterCommand implements RegistrationCommand, JsonRpcRegistrationCommand<RegisterCommand.RegistrationParams> {
@Override
public String getName() {
@Override
public void handleCommand(final Namespace ns, final RegistrationManager m) throws CommandException {
final boolean voiceVerification = Boolean.TRUE.equals(ns.getBoolean("voice"));
- final var captchaString = ns.getString("captcha");
- final var captcha = captchaString == null ? null : captchaString.replace("signalcaptcha://", "");
+ final var captcha = ns.getString("captcha");
+
+ register(m, voiceVerification, captcha);
+ }
+ @Override
+ public TypeReference<RegistrationParams> getRequestType() {
+ return new TypeReference<>() {};
+ }
+
+ @Override
+ public List<OutputType> getSupportedOutputTypes() {
+ return List.of(OutputType.PLAIN_TEXT, OutputType.JSON);
+ }
+
+ @Override
+ public void handleCommand(
+ final RegistrationParams request, final RegistrationManager m, final JsonWriter jsonWriter
+ ) throws CommandException {
+ register(m, Boolean.TRUE.equals(request.voice()), request.captcha());
+ }
+
+ private void register(
+ final RegistrationManager m, final boolean voiceVerification, final String captcha
+ ) throws UserErrorException, IOErrorException {
try {
m.register(voiceVerification, captcha);
} catch (CaptchaRequiredException e) {
throw new IOErrorException("Request verify error: " + e.getMessage(), e);
}
}
+
+ record RegistrationParams(Boolean voice, String captcha) {}
}
package org.asamk.signal.commands;
+import com.fasterxml.jackson.core.type.TypeReference;
+
import net.sourceforge.argparse4j.inf.Namespace;
import net.sourceforge.argparse4j.inf.Subparser;
+import org.asamk.signal.JsonWriter;
+import org.asamk.signal.OutputType;
import org.asamk.signal.commands.exceptions.CommandException;
import org.asamk.signal.commands.exceptions.IOErrorException;
import org.asamk.signal.commands.exceptions.UserErrorException;
import org.asamk.signal.manager.RegistrationManager;
import org.asamk.signal.manager.api.IncorrectPinException;
import org.asamk.signal.manager.api.PinLockedException;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
import java.io.IOException;
+import java.util.List;
+
+public class VerifyCommand implements RegistrationCommand, JsonRpcRegistrationCommand<VerifyCommand.VerifyParams> {
-public class VerifyCommand implements RegistrationCommand {
+ private final static Logger logger = LoggerFactory.getLogger(VerifyCommand.class);
@Override
public String getName() {
@Override
public void attachToSubparser(final Subparser subparser) {
subparser.help("Verify the number using the code received via SMS or voice.");
- subparser.addArgument("verificationCode").help("The verification code you received via sms or voice call.");
+ subparser.addArgument("verification-code").help("The verification code you received via sms or voice call.");
subparser.addArgument("-p", "--pin").help("The registration lock PIN, that was set by the user (Optional)");
}
@Override
public void handleCommand(final Namespace ns, final RegistrationManager m) throws CommandException {
- var verificationCode = ns.getString("verificationCode");
+ var verificationCode = ns.getString("verification-code");
var pin = ns.getString("pin");
+ verify(m, verificationCode, pin);
+ }
+
+ @Override
+ public TypeReference<VerifyParams> getRequestType() {
+ return new TypeReference<>() {};
+ }
+
+ @Override
+ public List<OutputType> getSupportedOutputTypes() {
+ return List.of(OutputType.PLAIN_TEXT, OutputType.JSON);
+ }
+
+ @Override
+ public void handleCommand(
+ final VerifyParams request, final RegistrationManager m, final JsonWriter jsonWriter
+ ) throws CommandException {
+ verify(m, request.verificationCode(), request.pin());
+ }
+
+ private void verify(
+ final RegistrationManager m, final String verificationCode, final String pin
+ ) throws UserErrorException, IOErrorException {
try {
m.verifyAccount(verificationCode, pin);
} catch (PinLockedException e) {
throw new IOErrorException("Verify error: " + e.getMessage(), e);
}
}
+
+ record VerifyParams(String verificationCode, String pin) {}
}
import org.asamk.signal.commands.Command;
import org.asamk.signal.commands.Commands;
import org.asamk.signal.commands.JsonRpcMultiCommand;
+import org.asamk.signal.commands.JsonRpcRegistrationCommand;
import org.asamk.signal.commands.JsonRpcSingleCommand;
import org.asamk.signal.commands.exceptions.CommandException;
import org.asamk.signal.commands.exceptions.IOErrorException;
import org.asamk.signal.commands.exceptions.UserErrorException;
import org.asamk.signal.manager.Manager;
import org.asamk.signal.manager.MultiAccountManager;
+import org.asamk.signal.manager.RegistrationManager;
import org.asamk.signal.util.Util;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
final ObjectMapper objectMapper, final String method, ContainerNode<?> params
) throws JsonRpcException {
var command = getCommand(method);
- // TODO implement register, verify, link
+ // TODO implement link
if (c != null) {
if (command instanceof JsonRpcMultiCommand<?> jsonRpcCommand) {
return runCommand(objectMapper, params, new MultiCommandRunnerImpl<>(c, jsonRpcCommand));
}
+ if (command instanceof JsonRpcRegistrationCommand<?> jsonRpcCommand) {
+ try (var manager = getRegistrationManagerFromParams(params)) {
+ if (manager != null) {
+ return runCommand(objectMapper,
+ params,
+ new RegistrationCommandRunnerImpl<>(manager, c, jsonRpcCommand));
+ } else {
+ throw new JsonRpcException(new JsonRpcResponse.Error(JsonRpcResponse.Error.INVALID_PARAMS,
+ "Method requires valid account parameter",
+ null));
+ }
+ } catch (IOException e) {
+ logger.warn("Failed to close registration manager", e);
+ }
+ }
}
if (command instanceof JsonRpcSingleCommand<?> jsonRpcCommand) {
if (m != null) {
}
private Manager getManagerFromParams(final ContainerNode<?> params) {
- if (params.has("account")) {
+ if (params != null && params.has("account")) {
final var manager = c.getManager(params.get("account").asText());
((ObjectNode) params).remove("account");
return manager;
return null;
}
+ private RegistrationManager getRegistrationManagerFromParams(final ContainerNode<?> params) {
+ if (params != null && params.has("account")) {
+ try {
+ final var registrationManager = c.getNewRegistrationManager(params.get("account").asText());
+ ((ObjectNode) params).remove("account");
+ return registrationManager;
+ } catch (IOException | IllegalStateException e) {
+ logger.warn("Failed to load registration manager", e);
+ return null;
+ }
+ }
+ return null;
+ }
+
private Command getCommand(final String method) {
if ("subscribeReceive".equals(method)) {
return new SubscribeReceiveCommand();
}
}
+ private record RegistrationCommandRunnerImpl<T>(
+ RegistrationManager m, MultiAccountManager c, JsonRpcRegistrationCommand<T> command
+ ) implements CommandRunner<T> {
+
+ @Override
+ public void handleCommand(final T request, final JsonWriter jsonWriter) throws CommandException {
+ command.handleCommand(request, m, jsonWriter);
+ }
+
+ @Override
+ public TypeReference<T> getRequestType() {
+ return command.getRequestType();
+ }
+ }
+
private record MultiCommandRunnerImpl<T>(
MultiAccountManager c, JsonRpcMultiCommand<T> command
) implements CommandRunner<T> {