com.petersamokhin.bots.sdk.longpoll.UpdatesHandler Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of vk-bot-java-sdk Show documentation
Show all versions of vk-bot-java-sdk Show documentation
SDK for interacting with VK.com API
The newest version!
package com.petersamokhin.bots.sdk.longpoll;
import com.petersamokhin.bots.sdk.callbacks.Callback;
import com.petersamokhin.bots.sdk.callbacks.messages.*;
import com.petersamokhin.bots.sdk.clients.Client;
import com.petersamokhin.bots.sdk.objects.Message;
import org.json.JSONArray;
import org.json.JSONObject;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.TimeUnit;
import static com.petersamokhin.bots.sdk.clients.Client.service;
import static com.petersamokhin.bots.sdk.clients.Client.sheduler;
/**
* Class for handling all updates in other thread
*/
public class UpdatesHandler extends Thread {
private volatile Queue queue = new Queue();
volatile boolean sendTyping = false;
/**
* Map with callbacks
*/
private ConcurrentHashMap callbacks = new ConcurrentHashMap<>();
/**
* Client with access_token
*/
private Client client;
UpdatesHandler(Client client) {
this.client = client;
}
@Override
public void run() {
sheduler.scheduleWithFixedDelay(this::handleCurrentUpdate, 0, 1, TimeUnit.MILLISECONDS);
}
/**
* Handle the array of updates
*/
void handle(JSONArray updates) {
this.queue.putAll(updates);
}
/**
* Handle one event from longpoll server
*/
private void handleCurrentUpdate() {
JSONArray currentUpdate;
if (this.queue.updates.isEmpty()) {
return;
} else {
currentUpdate = this.queue.shift();
}
int updateType = currentUpdate.getInt(0);
switch (updateType) {
// Handling new message
case 4: {
int messageFlags = currentUpdate.getInt(2);
// check if message is received
if ((messageFlags & 2) == 0) {
service.submit(() -> handleMessageUpdate(currentUpdate));
}
break;
}
// Handling update (user started typing)
case 61: {
handleTypingUpdate(currentUpdate);
break;
}
}
}
/**
* Handle new message
*/
private void handleMessageUpdate(JSONArray updateObject) {
boolean messageIsAlreadyHandled = false;
Message message = new Message(
this.client,
updateObject.getInt(1),
updateObject.getInt(2),
updateObject.getInt(3),
updateObject.getInt(4),
updateObject.getString(5),
(updateObject.length() > 6 ? (updateObject.get(6).toString().startsWith("{") ? new JSONObject(updateObject.get(6).toString()) : null) : null),
updateObject.length() > 7 ? updateObject.getInt(7) : null
);
if (sendTyping) {
this.client.api().call("messages.setActivity", "{type:'typing',peer_id:" + message.authorId() + "}", response -> {
});
}
// check for commands
if (this.client.commands.size() > 0) {
messageIsAlreadyHandled = handleCommands(message);
}
if (message.hasFwds()) {
if (callbacks.containsKey("OnMessageWithFwdsCallback")) {
((OnMessageWithFwdsCallback) callbacks.get("OnMessageWithFwdsCallback")).onMessage(message);
messageIsAlreadyHandled = true;
}
}
if (!messageIsAlreadyHandled) {
switch (message.messageType()) {
case "voiceMessage": {
if (callbacks.containsKey("OnVoiceMessageCallback")) {
((OnVoiceMessageCallback) callbacks.get("OnVoiceMessageCallback")).OnVoiceMessage(message);
messageIsAlreadyHandled = true;
}
break;
}
case "stickerMessage": {
if (callbacks.containsKey("OnStickerMessageCallback")) {
((OnStickerMessageCallback) callbacks.get("OnStickerMessageCallback")).OnStickerMessage(message);
messageIsAlreadyHandled = true;
}
break;
}
case "gifMessage": {
if (callbacks.containsKey("OnGifMessageCallback")) {
((OnGifMessageCallback) callbacks.get("OnGifMessageCallback")).OnGifMessage(message);
messageIsAlreadyHandled = true;
}
break;
}
case "audioMessage": {
if (callbacks.containsKey("OnAudioMessageCallback")) {
((OnAudioMessageCallback) callbacks.get("OnAudioMessageCallback")).onAudioMessage(message);
messageIsAlreadyHandled = true;
}
break;
}
case "videoMessage": {
if (callbacks.containsKey("OnVideoMessageCallback")) {
((OnVideoMessageCallback) callbacks.get("OnVideoMessageCallback")).onVideoMessage(message);
messageIsAlreadyHandled = true;
}
break;
}
case "docMessage": {
if (callbacks.containsKey("OnDocMessageCallback")) {
((OnDocMessageCallback) callbacks.get("OnDocMessageCallback")).OnDocMessage(message);
messageIsAlreadyHandled = true;
}
break;
}
case "wallMessage": {
if (callbacks.containsKey("OnWallMessageCallback")) {
((OnVoiceMessageCallback) callbacks.get("OnWallMessageCallback")).OnVoiceMessage(message);
messageIsAlreadyHandled = true;
}
break;
}
case "photoMessage": {
if (callbacks.containsKey("OnPhotoMessageCallback")) {
((OnPhotoMessageCallback) callbacks.get("OnPhotoMessageCallback")).onPhotoMessage(message);
messageIsAlreadyHandled = true;
}
break;
}
case "linkMessage": {
if (callbacks.containsKey("OnLinkMessageCallback")) {
((OnLinkMessageCallback) callbacks.get("OnLinkMessageCallback")).OnLinkMessage(message);
messageIsAlreadyHandled = true;
}
break;
}
case "simpleTextMessage": {
if (callbacks.containsKey("OnSimpleTextMessageCallback")) {
((OnSimpleTextMessageCallback) callbacks.get("OnSimpleTextMessageCallback")).OnSimpleTextMessage(message);
messageIsAlreadyHandled = true;
}
break;
}
}
}
if (callbacks.containsKey("OnMessageCallback") && !messageIsAlreadyHandled) {
((OnMessageCallback) callbacks.get("OnMessageCallback")).onMessage(message);
}
if (callbacks.containsKey("OnEveryMessageCallback")) {
((OnEveryMessageCallback) callbacks.get("OnEveryMessageCallback")).OnEveryMessage(message);
}
}
/**
* Handle dialog with typing user
*/
private void handleTypingUpdate(JSONArray updateObject) {
if (callbacks.containsKey("OnTypingCallback")) {
((OnTypingCallback) callbacks.get("OnTypingCallback")).OnTyping(updateObject.getInt(1));
}
}
/**
* Add callback to the map
*
* @param name Callback name
* @param callback Callback
*/
public void registerCallback(String name, Callback callback) {
this.callbacks.put(name, callback);
}
/**
* Returns count of callbacks
*/
int callbacksCount() {
return this.callbacks.size();
}
/**
* Returns count of commands
*/
int commandsCount() {
return this.client.commands.size();
}
/**
* Handle message and call back if it contains any command
*
* @param message received message
*/
private boolean handleCommands(Message message) {
boolean is = false;
for (Client.Command command : this.client.commands) {
for (int i = 0; i < command.getCommands().length; i++) {
if (message.getText().toLowerCase().contains(command.getCommands()[i].toString().toLowerCase())) {
command.getCallback().OnCommand(message);
is = true;
}
}
}
return is;
}
}