001/*
002 *  Copyright (c) 2022-2023, Mybatis-Flex (fuhai999@gmail.com).
003 *  <p>
004 *  Licensed under the Apache License, Version 2.0 (the "License");
005 *  you may not use this file except in compliance with the License.
006 *  You may obtain a copy of the License at
007 *  <p>
008 *  http://www.apache.org/licenses/LICENSE-2.0
009 *  <p>
010 *  Unless required by applicable law or agreed to in writing, software
011 *  distributed under the License is distributed on an "AS IS" BASIS,
012 *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013 *  See the License for the specific language governing permissions and
014 *  limitations under the License.
015 */
016package com.mybatisflex.core.audit;
017
018import java.util.ArrayList;
019import java.util.Collections;
020import java.util.List;
021import java.util.concurrent.Executors;
022import java.util.concurrent.ScheduledExecutorService;
023import java.util.concurrent.TimeUnit;
024import java.util.concurrent.locks.ReentrantReadWriteLock;
025
026/**
027 * 默认的审计消息收集器,其收集消息后,定时通过消息发送器{@link MessageReporter}把消息发送过去
028 */
029public class ScheduledMessageCollector implements MessageCollector, Runnable {
030
031    private final ScheduledExecutorService scheduler;
032    private final MessageReporter messageSender;
033
034    private final List<AuditMessage> messages = Collections.synchronizedList(new ArrayList<>());
035    private final ReentrantReadWriteLock rrwLock = new ReentrantReadWriteLock();
036
037    public ScheduledMessageCollector() {
038        this(10, new ConsoleMessageReporter());
039    }
040
041
042    public ScheduledMessageCollector(long period, MessageReporter messageSender) {
043        this.messageSender = messageSender;
044        this.scheduler = Executors.newSingleThreadScheduledExecutor(runnable -> {
045            Thread thread = new Thread(runnable, "ScheduledMessageCollector");
046            thread.setDaemon(true);
047            return thread;
048        });
049        this.scheduler.scheduleAtFixedRate(this, period, period, TimeUnit.SECONDS);
050    }
051
052
053    @Override
054    public void collect(AuditMessage message) {
055        try {
056            rrwLock.readLock().lock();
057            messages.add(message);
058        } finally {
059            rrwLock.readLock().unlock();
060        }
061    }
062
063
064    @Override
065    public void run() {
066        if (messages.isEmpty()) {
067            return;
068        }
069        List<AuditMessage> sendMessages;
070        try {
071            rrwLock.writeLock().lock();
072            sendMessages = new ArrayList<>(messages);
073            messages.clear();
074        } finally {
075            rrwLock.writeLock().unlock();
076        }
077        messageSender.sendMessages(sendMessages);
078    }
079
080    public void release() {
081        run(); //clear the messages
082        scheduler.shutdown();
083    }
084
085}