Search in sources :

Example 1 with MutexSyncMode

use of net.robinfriedli.exec.modes.MutexSyncMode in project aiode by robinfriedli.

the class Artist method getOrCreateArtist.

/**
 * Get the existing artist with the provided or create a new artist. This method runs synchronised on the artist id
 * to avoid unique constraint violations during concurrent creation.
 *
 * @param artist  the spotify artist
 * @param session the hibernate session
 */
public static Artist getOrCreateArtist(ArtistSimplified artist, Session session) {
    QueryBuilderFactory queryBuilderFactory = Aiode.get().getQueryBuilderFactory();
    Mode mode = Mode.create().with(new MutexSyncMode<>(artist.getId(), ARTIST_SYNC));
    return HibernateInvoker.create(session).invokeFunction(mode, currentSession -> {
        Optional<Artist> existingArtist = queryBuilderFactory.find(Artist.class).where((cb, root) -> cb.equal(root.get("id"), artist.getId())).build(currentSession).uniqueResultOptional();
        return existingArtist.orElseGet(() -> {
            Artist newArtist = new Artist(artist.getId(), artist.getName());
            currentSession.persist(newArtist);
            currentSession.flush();
            return newArtist;
        });
    });
}
Also used : QueryBuilderFactory(net.robinfriedli.aiode.persist.qb.QueryBuilderFactory) Mode(net.robinfriedli.exec.Mode) MutexSyncMode(net.robinfriedli.exec.modes.MutexSyncMode)

Example 2 with MutexSyncMode

use of net.robinfriedli.exec.modes.MutexSyncMode in project aiode by robinfriedli.

the class ConnectCommand method awaitPrivateMessage.

private void awaitPrivateMessage(ClientSession clientSession, long userId, Guild guild, User user, int attemptNumber) {
    CompletableFuture<PrivateMessageReceivedEvent> futurePrivateMessage = getManager().getEventWaiter().awaitEvent(PrivateMessageReceivedEvent.class, event -> event.getAuthor().getIdLong() == userId).orTimeout(1, TimeUnit.MINUTES);
    CompletableFutures.handleWhenComplete(futurePrivateMessage, (event, error) -> {
        try {
            MessageService messageService = getMessageService();
            if (event != null) {
                String token = event.getMessage().getContentRaw();
                UUID sessionId;
                try {
                    sessionId = UUID.fromString(token);
                } catch (IllegalArgumentException e) {
                    String message = String.format("'%s' is not a valid token. ", token);
                    if (attemptNumber >= RETRY_COUNT) {
                        messageService.sendError(message + "Maximum retry count reached.", user);
                    } else {
                        messageService.sendError(message + String.format("Attempt %d / %d", attemptNumber, RETRY_COUNT), user);
                        awaitPrivateMessage(clientSession, userId, guild, user, attemptNumber + 1);
                    }
                    return;
                }
                QueryBuilderFactory queryBuilderFactory = getQueryBuilderFactory();
                MutexSyncMode<UUID> mutexSyncMode = new MutexSyncMode<>(sessionId, TOKEN_SYNC);
                HibernateInvoker.create().invokeConsumer(Mode.create().with(mutexSyncMode), session -> {
                    Optional<GeneratedToken> foundGeneratedToken = queryBuilderFactory.find(GeneratedToken.class).where((cb, root) -> cb.equal(root.get("token"), sessionId)).build(session).uniqueResultOptional();
                    if (foundGeneratedToken.isEmpty()) {
                        String message = "Token is invalid. Make sure it matches the token generated by your web client. Tokens may not be shared or reused. ";
                        if (attemptNumber >= RETRY_COUNT) {
                            messageService.sendError(message + "Maximum retry count reached.", user);
                        } else {
                            messageService.sendError(message + String.format("Attempt %d / %d", attemptNumber, RETRY_COUNT), user);
                            awaitPrivateMessage(clientSession, userId, guild, user, attemptNumber + 1);
                        }
                        return;
                    }
                    GeneratedToken generatedToken = foundGeneratedToken.get();
                    Long existingSessionCount = queryBuilderFactory.select(ClientSession.class, (from, cb) -> cb.count(from.get("pk")), Long.class).where((cb, root) -> cb.equal(root.get("sessionId"), sessionId)).build(session).uniqueResult();
                    if (existingSessionCount > 0) {
                        messageService.sendError("A session with this ID already exists. You are probably already signed in, try reloading your web client. If your client still can't sign in try generating a new token using the reload button and then try the connect command again.", user);
                        return;
                    }
                    clientSession.setLastRefresh(LocalDateTime.now());
                    clientSession.setSessionId(sessionId);
                    clientSession.setIpAddress(generatedToken.getIp());
                    session.persist(clientSession);
                    session.delete(generatedToken);
                    messageService.sendSuccess(String.format("Okay, a session connected to guild '%s' " + "has been prepared. You may return to the web client to complete the setup. The client should " + "connect automatically, else you can continue manually.", guild.getName()), user);
                });
            } else if (error != null) {
                if (error instanceof TimeoutException) {
                    messageService.sendError("Your connection attempt timed out", user);
                } else {
                    EmbedBuilder exceptionEmbed = ExceptionUtils.buildErrorEmbed(error);
                    exceptionEmbed.setTitle("Exception");
                    exceptionEmbed.setDescription("There has been an error awaiting private message to establish connection");
                    LoggerFactory.getLogger(getClass()).error("unexpected exception while awaiting private message", error);
                    sendMessage(exceptionEmbed.build());
                }
                setFailed(true);
            }
        } finally {
            USERS_WITH_PENDING_CONNECTION.remove(userId);
        }
    }, e -> {
        LoggerFactory.getLogger(getClass()).error("Unexpected error in whenComplete of event handler", e);
        EmbedBuilder embedBuilder = ExceptionUtils.buildErrorEmbed(e).setTitle("Exception").setDescription("There has been an unexpected exception while trying to establish the connection");
        getMessageService().send(embedBuilder.build(), user);
    });
}
Also used : LocalDateTime(java.time.LocalDateTime) LoggerFactory(org.slf4j.LoggerFactory) TimeoutException(java.util.concurrent.TimeoutException) CompletableFuture(java.util.concurrent.CompletableFuture) MutexSync(net.robinfriedli.exec.MutexSync) User(net.dv8tion.jda.api.entities.User) QueryBuilderFactory(net.robinfriedli.aiode.persist.qb.QueryBuilderFactory) Guild(net.dv8tion.jda.api.entities.Guild) Map(java.util.Map) CompletableFutures(net.robinfriedli.aiode.concurrent.CompletableFutures) CommandContribution(net.robinfriedli.aiode.entities.xml.CommandContribution) ClientSession(net.robinfriedli.aiode.entities.ClientSession) CommandManager(net.robinfriedli.aiode.command.CommandManager) Mode(net.robinfriedli.exec.Mode) MessageService(net.robinfriedli.aiode.discord.MessageService) InvalidCommandException(net.robinfriedli.aiode.exceptions.InvalidCommandException) ConcurrentHashMap(java.util.concurrent.ConcurrentHashMap) PrivateMessageReceivedEvent(net.dv8tion.jda.api.events.message.priv.PrivateMessageReceivedEvent) UUID(java.util.UUID) EmbedBuilder(net.dv8tion.jda.api.EmbedBuilder) HibernateInvoker(net.robinfriedli.aiode.function.HibernateInvoker) TimeUnit(java.util.concurrent.TimeUnit) AbstractCommand(net.robinfriedli.aiode.command.AbstractCommand) CommandContext(net.robinfriedli.aiode.command.CommandContext) GeneratedToken(net.robinfriedli.aiode.entities.GeneratedToken) Optional(java.util.Optional) MutexSyncMode(net.robinfriedli.exec.modes.MutexSyncMode) ExceptionUtils(net.robinfriedli.aiode.exceptions.ExceptionUtils) PrivateMessageReceivedEvent(net.dv8tion.jda.api.events.message.priv.PrivateMessageReceivedEvent) MessageService(net.robinfriedli.aiode.discord.MessageService) QueryBuilderFactory(net.robinfriedli.aiode.persist.qb.QueryBuilderFactory) EmbedBuilder(net.dv8tion.jda.api.EmbedBuilder) MutexSyncMode(net.robinfriedli.exec.modes.MutexSyncMode) GeneratedToken(net.robinfriedli.aiode.entities.GeneratedToken) ClientSession(net.robinfriedli.aiode.entities.ClientSession) UUID(java.util.UUID) TimeoutException(java.util.concurrent.TimeoutException)

Example 3 with MutexSyncMode

use of net.robinfriedli.exec.modes.MutexSyncMode in project aiode by robinfriedli.

the class AbstractCommand method invoke.

protected <E> E invoke(Callable<E> callable) {
    HibernateInvoker hibernateInvoker = HibernateInvoker.create(context.getSession());
    Mode mode = Mode.create().with(new MutexSyncMode<>(context.getGuild().getIdLong(), GUILD_MUTEX_SYNC));
    return hibernateInvoker.invoke(mode, callable);
}
Also used : HibernateInvoker(net.robinfriedli.aiode.function.HibernateInvoker) MutexSyncMode(net.robinfriedli.exec.modes.MutexSyncMode) Mode(net.robinfriedli.exec.Mode)

Example 4 with MutexSyncMode

use of net.robinfriedli.exec.modes.MutexSyncMode in project aiode by robinfriedli.

the class AbstractCommand method invokeWithSession.

protected <E> E invokeWithSession(Function<Session, E> function) {
    HibernateInvoker hibernateInvoker = HibernateInvoker.create(context.getSession());
    Mode mode = Mode.create().with(new MutexSyncMode<>(context.getGuild().getIdLong(), GUILD_MUTEX_SYNC));
    return hibernateInvoker.invokeFunction(mode, function);
}
Also used : HibernateInvoker(net.robinfriedli.aiode.function.HibernateInvoker) MutexSyncMode(net.robinfriedli.exec.modes.MutexSyncMode) Mode(net.robinfriedli.exec.Mode)

Aggregations

Mode (net.robinfriedli.exec.Mode)4 MutexSyncMode (net.robinfriedli.exec.modes.MutexSyncMode)4 HibernateInvoker (net.robinfriedli.aiode.function.HibernateInvoker)3 QueryBuilderFactory (net.robinfriedli.aiode.persist.qb.QueryBuilderFactory)2 LocalDateTime (java.time.LocalDateTime)1 Map (java.util.Map)1 Optional (java.util.Optional)1 UUID (java.util.UUID)1 CompletableFuture (java.util.concurrent.CompletableFuture)1 ConcurrentHashMap (java.util.concurrent.ConcurrentHashMap)1 TimeUnit (java.util.concurrent.TimeUnit)1 TimeoutException (java.util.concurrent.TimeoutException)1 EmbedBuilder (net.dv8tion.jda.api.EmbedBuilder)1 Guild (net.dv8tion.jda.api.entities.Guild)1 User (net.dv8tion.jda.api.entities.User)1 PrivateMessageReceivedEvent (net.dv8tion.jda.api.events.message.priv.PrivateMessageReceivedEvent)1 AbstractCommand (net.robinfriedli.aiode.command.AbstractCommand)1 CommandContext (net.robinfriedli.aiode.command.CommandContext)1 CommandManager (net.robinfriedli.aiode.command.CommandManager)1 CompletableFutures (net.robinfriedli.aiode.concurrent.CompletableFutures)1