parent
d486563099
commit
7cf3a989bf
@ -14,6 +14,7 @@
|
|||||||
behavior, the `--notify-self` parameter can be added
|
behavior, the `--notify-self` parameter can be added
|
||||||
- New `--unrestricted-unidentified-sender` parameter for `updateAccount command`
|
- New `--unrestricted-unidentified-sender` parameter for `updateAccount command`
|
||||||
- New `--bus-name` parameter for `daemon` command to use another D-Bus bus name
|
- New `--bus-name` parameter for `daemon` command to use another D-Bus bus name
|
||||||
|
- New `getAvatar` and `getSticker` commands to get avatar and sticker images
|
||||||
|
|
||||||
### Improved
|
### Improved
|
||||||
|
|
||||||
|
@ -68,6 +68,20 @@ pub enum CliCommands {
|
|||||||
#[arg(short = 'g', long = "group-id")]
|
#[arg(short = 'g', long = "group-id")]
|
||||||
group_id: Option<String>,
|
group_id: Option<String>,
|
||||||
},
|
},
|
||||||
|
GetAvatar {
|
||||||
|
#[arg(long)]
|
||||||
|
contact: Option<String>,
|
||||||
|
#[arg(long)]
|
||||||
|
profile: Option<String>,
|
||||||
|
#[arg(short = 'g', long = "group-id")]
|
||||||
|
group_id: Option<String>,
|
||||||
|
},
|
||||||
|
GetSticker {
|
||||||
|
#[arg(long = "pack-id")]
|
||||||
|
pack_id: String,
|
||||||
|
#[arg(long = "sticker-id")]
|
||||||
|
sticker_id: u32,
|
||||||
|
},
|
||||||
GetUserStatus {
|
GetUserStatus {
|
||||||
recipient: Vec<String>,
|
recipient: Vec<String>,
|
||||||
},
|
},
|
||||||
|
@ -45,7 +45,24 @@ pub trait Rpc {
|
|||||||
account: Option<String>,
|
account: Option<String>,
|
||||||
id: String,
|
id: String,
|
||||||
recipient: Option<String>,
|
recipient: Option<String>,
|
||||||
group_id: Option<String>,
|
#[allow(non_snake_case)] groupId: Option<String>,
|
||||||
|
) -> Result<Value, ErrorObjectOwned>;
|
||||||
|
|
||||||
|
#[method(name = "getAvatar", param_kind = map)]
|
||||||
|
fn get_avatar(
|
||||||
|
&self,
|
||||||
|
account: Option<String>,
|
||||||
|
contact: Option<String>,
|
||||||
|
profile: Option<String>,
|
||||||
|
#[allow(non_snake_case)] groupId: Option<String>,
|
||||||
|
) -> Result<Value, ErrorObjectOwned>;
|
||||||
|
|
||||||
|
#[method(name = "getSticker", param_kind = map)]
|
||||||
|
fn get_sticker(
|
||||||
|
&self,
|
||||||
|
account: Option<String>,
|
||||||
|
#[allow(non_snake_case)] packId: String,
|
||||||
|
#[allow(non_snake_case)] stickerId: u32,
|
||||||
) -> Result<Value, ErrorObjectOwned>;
|
) -> Result<Value, ErrorObjectOwned>;
|
||||||
|
|
||||||
#[method(name = "getUserStatus", param_kind = map)]
|
#[method(name = "getUserStatus", param_kind = map)]
|
||||||
|
@ -407,6 +407,19 @@ async fn handle_command(
|
|||||||
.get_attachment(cli.account, id, recipient, group_id)
|
.get_attachment(cli.account, id, recipient, group_id)
|
||||||
.await
|
.await
|
||||||
}
|
}
|
||||||
|
CliCommands::GetAvatar {
|
||||||
|
contact,
|
||||||
|
profile,
|
||||||
|
group_id,
|
||||||
|
} => {
|
||||||
|
client
|
||||||
|
.get_avatar(cli.account, contact, profile, group_id)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
CliCommands::GetSticker {
|
||||||
|
pack_id,
|
||||||
|
sticker_id,
|
||||||
|
} => client.get_sticker(cli.account, pack_id, sticker_id).await,
|
||||||
CliCommands::StartChangeNumber {
|
CliCommands::StartChangeNumber {
|
||||||
number,
|
number,
|
||||||
voice,
|
voice,
|
||||||
|
@ -34,6 +34,7 @@ import org.asamk.signal.manager.api.RecipientIdentifier;
|
|||||||
import org.asamk.signal.manager.api.SendGroupMessageResults;
|
import org.asamk.signal.manager.api.SendGroupMessageResults;
|
||||||
import org.asamk.signal.manager.api.SendMessageResults;
|
import org.asamk.signal.manager.api.SendMessageResults;
|
||||||
import org.asamk.signal.manager.api.StickerPack;
|
import org.asamk.signal.manager.api.StickerPack;
|
||||||
|
import org.asamk.signal.manager.api.StickerPackId;
|
||||||
import org.asamk.signal.manager.api.StickerPackInvalidException;
|
import org.asamk.signal.manager.api.StickerPackInvalidException;
|
||||||
import org.asamk.signal.manager.api.StickerPackUrl;
|
import org.asamk.signal.manager.api.StickerPackUrl;
|
||||||
import org.asamk.signal.manager.api.TypingAction;
|
import org.asamk.signal.manager.api.TypingAction;
|
||||||
@ -307,6 +308,14 @@ public interface Manager extends Closeable {
|
|||||||
|
|
||||||
InputStream retrieveAttachment(final String id) throws IOException;
|
InputStream retrieveAttachment(final String id) throws IOException;
|
||||||
|
|
||||||
|
InputStream retrieveContactAvatar(final RecipientIdentifier.Single recipient) throws IOException, UnregisteredRecipientException;
|
||||||
|
|
||||||
|
InputStream retrieveProfileAvatar(final RecipientIdentifier.Single recipient) throws IOException, UnregisteredRecipientException;
|
||||||
|
|
||||||
|
InputStream retrieveGroupAvatar(final GroupId groupId) throws IOException;
|
||||||
|
|
||||||
|
InputStream retrieveSticker(final StickerPackId stickerPackId, final int stickerId) throws IOException;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
void close();
|
void close();
|
||||||
|
|
||||||
|
@ -1,7 +1,8 @@
|
|||||||
package org.asamk.signal.manager.api;
|
package org.asamk.signal.manager.api;
|
||||||
|
|
||||||
|
import org.whispersystems.signalservice.internal.util.Hex;
|
||||||
|
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.Base64;
|
|
||||||
|
|
||||||
public class StickerPackId {
|
public class StickerPackId {
|
||||||
|
|
||||||
@ -36,6 +37,6 @@ public class StickerPackId {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String toString() {
|
public String toString() {
|
||||||
return "StickerPackId{" + Base64.getUrlEncoder().encodeToString(id) + '}';
|
return "StickerPackId{" + Hex.toStringCondensed(id) + '}';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -104,6 +104,7 @@ import org.whispersystems.signalservice.internal.util.Util;
|
|||||||
|
|
||||||
import java.io.ByteArrayInputStream;
|
import java.io.ByteArrayInputStream;
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.nio.charset.StandardCharsets;
|
import java.nio.charset.StandardCharsets;
|
||||||
@ -1337,6 +1338,58 @@ public class ManagerImpl implements Manager {
|
|||||||
return context.getAttachmentHelper().retrieveAttachment(id).getStream();
|
return context.getAttachmentHelper().retrieveAttachment(id).getStream();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public InputStream retrieveContactAvatar(final RecipientIdentifier.Single recipient) throws IOException, UnregisteredRecipientException {
|
||||||
|
final var recipientId = context.getRecipientHelper().resolveRecipient(recipient);
|
||||||
|
final var address = account.getRecipientStore().resolveRecipientAddress(recipientId);
|
||||||
|
final var streamDetails = context.getAvatarStore().retrieveContactAvatar(address);
|
||||||
|
if (streamDetails == null) {
|
||||||
|
throw new FileNotFoundException();
|
||||||
|
}
|
||||||
|
return streamDetails.getStream();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public InputStream retrieveProfileAvatar(final RecipientIdentifier.Single recipient) throws IOException, UnregisteredRecipientException {
|
||||||
|
final var recipientId = context.getRecipientHelper().resolveRecipient(recipient);
|
||||||
|
context.getProfileHelper().getRecipientProfile(recipientId);
|
||||||
|
final var address = account.getRecipientStore().resolveRecipientAddress(recipientId);
|
||||||
|
final var streamDetails = context.getAvatarStore().retrieveProfileAvatar(address);
|
||||||
|
if (streamDetails == null) {
|
||||||
|
throw new FileNotFoundException();
|
||||||
|
}
|
||||||
|
return streamDetails.getStream();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public InputStream retrieveGroupAvatar(final GroupId groupId) throws IOException {
|
||||||
|
final var streamDetails = context.getAvatarStore().retrieveGroupAvatar(groupId);
|
||||||
|
context.getGroupHelper().getGroup(groupId);
|
||||||
|
if (streamDetails == null) {
|
||||||
|
throw new FileNotFoundException();
|
||||||
|
}
|
||||||
|
return streamDetails.getStream();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public InputStream retrieveSticker(final StickerPackId stickerPackId, final int stickerId) throws IOException {
|
||||||
|
var streamDetails = context.getStickerPackStore().retrieveSticker(stickerPackId, stickerId);
|
||||||
|
if (streamDetails == null) {
|
||||||
|
final var pack = account.getStickerStore().getStickerPack(stickerPackId);
|
||||||
|
if (pack != null) {
|
||||||
|
try {
|
||||||
|
context.getStickerHelper().retrieveStickerPack(stickerPackId, pack.packKey());
|
||||||
|
} catch (InvalidMessageException e) {
|
||||||
|
logger.warn("Failed to download sticker pack");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (streamDetails == null) {
|
||||||
|
throw new FileNotFoundException();
|
||||||
|
}
|
||||||
|
return streamDetails.getStream();
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void close() {
|
public void close() {
|
||||||
Thread thread;
|
Thread thread;
|
||||||
|
@ -722,6 +722,31 @@ Referred to generally as recipient.
|
|||||||
*-g* [GROUP], *--group-id* [GROUP]::
|
*-g* [GROUP], *--group-id* [GROUP]::
|
||||||
Alternatively, specify the group IDs for which to get the attachment.
|
Alternatively, specify the group IDs for which to get the attachment.
|
||||||
|
|
||||||
|
=== getAvatar
|
||||||
|
|
||||||
|
Gets the raw data for a specified contact, contact's profile or group avatar.
|
||||||
|
The attachment data is returned as a Base64 String.
|
||||||
|
|
||||||
|
*--contact* [RECIPIENT]::
|
||||||
|
Specify the number of a recipient.
|
||||||
|
|
||||||
|
*--profile* [RECIPIENT]::
|
||||||
|
Specify the number of a recipient.
|
||||||
|
|
||||||
|
*-g* [GROUP], *--group-id* [GROUP]::
|
||||||
|
Alternatively, specify the group ID for which to get the avatar.
|
||||||
|
|
||||||
|
=== getSticker
|
||||||
|
|
||||||
|
Gets the raw data for a specified sticker.
|
||||||
|
The attachment data is returned as a Base64 String.
|
||||||
|
|
||||||
|
*--pack-id* [PACK_ID]::
|
||||||
|
Specify the id of a sticker pack (hex encoded).
|
||||||
|
|
||||||
|
*--sticker-id* [STICKER_ID]::
|
||||||
|
Specify the index of a sticker in the sticker pack.
|
||||||
|
|
||||||
=== daemon
|
=== daemon
|
||||||
|
|
||||||
signal-cli can run in daemon mode and provides JSON-RPC or an experimental dbus interface.
|
signal-cli can run in daemon mode and provides JSON-RPC or an experimental dbus interface.
|
||||||
|
@ -17,6 +17,8 @@ public class Commands {
|
|||||||
addCommand(new FinishChangeNumberCommand());
|
addCommand(new FinishChangeNumberCommand());
|
||||||
addCommand(new FinishLinkCommand());
|
addCommand(new FinishLinkCommand());
|
||||||
addCommand(new GetAttachmentCommand());
|
addCommand(new GetAttachmentCommand());
|
||||||
|
addCommand(new GetAvatarCommand());
|
||||||
|
addCommand(new GetStickerCommand());
|
||||||
addCommand(new GetUserStatusCommand());
|
addCommand(new GetUserStatusCommand());
|
||||||
addCommand(new AddStickerPackCommand());
|
addCommand(new AddStickerPackCommand());
|
||||||
addCommand(new JoinGroupCommand());
|
addCommand(new JoinGroupCommand());
|
||||||
|
@ -26,6 +26,7 @@ public class GetAttachmentCommand implements JsonRpcLocalCommand {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void attachToSubparser(final Subparser subparser) {
|
public void attachToSubparser(final Subparser subparser) {
|
||||||
|
subparser.help("Retrieve an already downloaded attachment base64 encoded.");
|
||||||
subparser.addArgument("--id").required(true).help("The ID of the attachment file.");
|
subparser.addArgument("--id").required(true).help("The ID of the attachment file.");
|
||||||
var mut = subparser.addMutuallyExclusiveGroup().required(true);
|
var mut = subparser.addMutuallyExclusiveGroup().required(true);
|
||||||
mut.addArgument("--recipient").help("Sender of the attachment");
|
mut.addArgument("--recipient").help("Sender of the attachment");
|
||||||
|
@ -0,0 +1,76 @@
|
|||||||
|
package org.asamk.signal.commands;
|
||||||
|
|
||||||
|
import net.sourceforge.argparse4j.inf.Namespace;
|
||||||
|
import net.sourceforge.argparse4j.inf.Subparser;
|
||||||
|
|
||||||
|
import org.asamk.signal.commands.exceptions.CommandException;
|
||||||
|
import org.asamk.signal.commands.exceptions.UnexpectedErrorException;
|
||||||
|
import org.asamk.signal.commands.exceptions.UserErrorException;
|
||||||
|
import org.asamk.signal.json.JsonAttachmentData;
|
||||||
|
import org.asamk.signal.manager.Manager;
|
||||||
|
import org.asamk.signal.manager.api.UnregisteredRecipientException;
|
||||||
|
import org.asamk.signal.output.JsonWriter;
|
||||||
|
import org.asamk.signal.output.OutputWriter;
|
||||||
|
import org.asamk.signal.output.PlainTextWriter;
|
||||||
|
import org.asamk.signal.util.CommandUtil;
|
||||||
|
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.util.Base64;
|
||||||
|
|
||||||
|
public class GetAvatarCommand implements JsonRpcLocalCommand {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getName() {
|
||||||
|
return "getAvatar";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void attachToSubparser(final Subparser subparser) {
|
||||||
|
subparser.help("Retrieve the avatar of a contact, contact's profile or group base64 encoded.");
|
||||||
|
var mut = subparser.addMutuallyExclusiveGroup().required(true);
|
||||||
|
mut.addArgument("-c", "--contact").help("Get a contact avatar");
|
||||||
|
mut.addArgument("-p", "--profile").help("Get a profile avatar");
|
||||||
|
mut.addArgument("-g", "--group-id").help("Get a group avatar");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void handleCommand(
|
||||||
|
final Namespace ns, final Manager m, final OutputWriter outputWriter
|
||||||
|
) throws CommandException {
|
||||||
|
final var contactRecipient = ns.getString("contact");
|
||||||
|
final var profileRecipient = ns.getString("profile");
|
||||||
|
final var groupId = ns.getString("groupId");
|
||||||
|
|
||||||
|
final InputStream data;
|
||||||
|
try {
|
||||||
|
if (contactRecipient != null) {
|
||||||
|
data = m.retrieveContactAvatar(CommandUtil.getSingleRecipientIdentifier(contactRecipient,
|
||||||
|
m.getSelfNumber()));
|
||||||
|
} else if (profileRecipient != null) {
|
||||||
|
data = m.retrieveProfileAvatar(CommandUtil.getSingleRecipientIdentifier(profileRecipient,
|
||||||
|
m.getSelfNumber()));
|
||||||
|
} else {
|
||||||
|
data = m.retrieveGroupAvatar(CommandUtil.getGroupId(groupId));
|
||||||
|
}
|
||||||
|
} catch (FileNotFoundException ex) {
|
||||||
|
throw new UserErrorException("Could not find avatar", ex);
|
||||||
|
} catch (IOException ex) {
|
||||||
|
throw new UnexpectedErrorException("An error occurred reading avatar", ex);
|
||||||
|
} catch (UnregisteredRecipientException e) {
|
||||||
|
throw new UserErrorException("The user " + e.getSender().getIdentifier() + " is not registered.");
|
||||||
|
}
|
||||||
|
|
||||||
|
try (data) {
|
||||||
|
final var bytes = data.readAllBytes();
|
||||||
|
final var base64 = Base64.getEncoder().encodeToString(bytes);
|
||||||
|
switch (outputWriter) {
|
||||||
|
case PlainTextWriter writer -> writer.println(base64);
|
||||||
|
case JsonWriter writer -> writer.write(new JsonAttachmentData(base64));
|
||||||
|
}
|
||||||
|
} catch (IOException ex) {
|
||||||
|
throw new UnexpectedErrorException("An error occurred reading avatar", ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,60 @@
|
|||||||
|
package org.asamk.signal.commands;
|
||||||
|
|
||||||
|
import net.sourceforge.argparse4j.inf.Namespace;
|
||||||
|
import net.sourceforge.argparse4j.inf.Subparser;
|
||||||
|
|
||||||
|
import org.asamk.signal.commands.exceptions.CommandException;
|
||||||
|
import org.asamk.signal.commands.exceptions.UnexpectedErrorException;
|
||||||
|
import org.asamk.signal.commands.exceptions.UserErrorException;
|
||||||
|
import org.asamk.signal.json.JsonAttachmentData;
|
||||||
|
import org.asamk.signal.manager.Manager;
|
||||||
|
import org.asamk.signal.manager.api.StickerPackId;
|
||||||
|
import org.asamk.signal.output.JsonWriter;
|
||||||
|
import org.asamk.signal.output.OutputWriter;
|
||||||
|
import org.asamk.signal.output.PlainTextWriter;
|
||||||
|
import org.asamk.signal.util.Hex;
|
||||||
|
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.util.Base64;
|
||||||
|
|
||||||
|
public class GetStickerCommand implements JsonRpcLocalCommand {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getName() {
|
||||||
|
return "getSticker";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void attachToSubparser(final Subparser subparser) {
|
||||||
|
subparser.help("Retrieve the sticker of a sticker pack base64 encoded.");
|
||||||
|
subparser.addArgument("--pack-id").required(true).help("The ID of the sticker pack.");
|
||||||
|
subparser.addArgument("--sticker-id").type(int.class).required(true).help("The ID of the sticker.");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void handleCommand(
|
||||||
|
final Namespace ns, final Manager m, final OutputWriter outputWriter
|
||||||
|
) throws CommandException {
|
||||||
|
|
||||||
|
final var packId = StickerPackId.deserialize(Hex.toByteArray(ns.getString("pack-id")));
|
||||||
|
final var stickerId = ns.getInt("sticker-id");
|
||||||
|
|
||||||
|
try (InputStream data = m.retrieveSticker(packId, stickerId)) {
|
||||||
|
final var bytes = data.readAllBytes();
|
||||||
|
final var base64 = Base64.getEncoder().encodeToString(bytes);
|
||||||
|
switch (outputWriter) {
|
||||||
|
case PlainTextWriter writer -> writer.println(base64);
|
||||||
|
case JsonWriter writer -> writer.write(new JsonAttachmentData(base64));
|
||||||
|
}
|
||||||
|
} catch (FileNotFoundException ex) {
|
||||||
|
throw new UserErrorException("Could not find sticker with ID: " + stickerId + " in pack " + packId, ex);
|
||||||
|
} catch (IOException ex) {
|
||||||
|
throw new UnexpectedErrorException("An error occurred reading sticker with ID: "
|
||||||
|
+ stickerId
|
||||||
|
+ " in pack "
|
||||||
|
+ packId, ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -38,6 +38,7 @@ import org.asamk.signal.manager.api.RecipientIdentifier;
|
|||||||
import org.asamk.signal.manager.api.SendGroupMessageResults;
|
import org.asamk.signal.manager.api.SendGroupMessageResults;
|
||||||
import org.asamk.signal.manager.api.SendMessageResults;
|
import org.asamk.signal.manager.api.SendMessageResults;
|
||||||
import org.asamk.signal.manager.api.StickerPack;
|
import org.asamk.signal.manager.api.StickerPack;
|
||||||
|
import org.asamk.signal.manager.api.StickerPackId;
|
||||||
import org.asamk.signal.manager.api.StickerPackInvalidException;
|
import org.asamk.signal.manager.api.StickerPackInvalidException;
|
||||||
import org.asamk.signal.manager.api.StickerPackUrl;
|
import org.asamk.signal.manager.api.StickerPackUrl;
|
||||||
import org.asamk.signal.manager.api.TypingAction;
|
import org.asamk.signal.manager.api.TypingAction;
|
||||||
@ -1069,6 +1070,26 @@ public class DbusManagerImpl implements Manager {
|
|||||||
throw new UnsupportedOperationException();
|
throw new UnsupportedOperationException();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public InputStream retrieveContactAvatar(final RecipientIdentifier.Single recipient) throws IOException, UnregisteredRecipientException {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public InputStream retrieveProfileAvatar(final RecipientIdentifier.Single recipient) throws IOException, UnregisteredRecipientException {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public InputStream retrieveGroupAvatar(final GroupId groupId) throws IOException {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public InputStream retrieveSticker(final StickerPackId stickerPackId, final int stickerId) throws IOException {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
private <T> T getValue(
|
private <T> T getValue(
|
||||||
final Map<String, Variant<?>> stringVariantMap, final String field
|
final Map<String, Variant<?>> stringVariantMap, final String field
|
||||||
|
Loading…
x
Reference in New Issue
Block a user