Java tutorial
/* * (C) Copyright 2015-2016 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * * Contributors: * qiong.wang () */ package com.wq.wqchat.core.push; import com.wq.wqchat.api.Message; import com.wq.wqchat.api.connection.Connection; import com.wq.wqchat.api.connection.SessionContext; import com.wq.wqchat.api.spi.push.IPushMessage; import com.wq.wqchat.common.condition.AwaysPassCondition; import com.wq.wqchat.api.common.Condition; import com.wq.wqchat.common.message.PushMessage; import com.wq.wqchat.common.qps.FlowControl; import com.wq.wqchat.common.qps.OverFlowException; import com.wq.wqchat.core.router.LocalRouter; import com.wq.wqchat.core.router.RouterCenter; import com.wq.wqchat.tools.common.TimeLine; import com.wq.wqchat.tools.log.Logs; import io.netty.channel.ChannelFuture; import io.netty.channel.ChannelFutureListener; import java.util.HashMap; import java.util.Iterator; import java.util.Map; import java.util.concurrent.ScheduledExecutorService; import java.util.concurrent.atomic.AtomicInteger; public final class BroadcastPushTask implements PushTask, ChannelFutureListener { private final long begin = System.currentTimeMillis(); private final AtomicInteger finishTasks = new AtomicInteger(0); private final FlowControl flowControl; private final IPushMessage message; private final Condition condition; private final TimeLine timeLine = new TimeLine(); //Iterator, ????????/ private final Iterator<Map.Entry<String, Map<Integer, LocalRouter>>> iterator; public BroadcastPushTask(IPushMessage message, FlowControl flowControl) { this.message = message; this.flowControl = flowControl; this.condition = message.getCondition(); this.iterator = RouterCenter.I.getLocalRouterManager().routers().entrySet().iterator(); this.timeLine.begin("push-center-begin"); } @Override public void run() { flowControl.reset(); boolean done = broadcast(); if (done) {//done ? if (finishTasks.addAndGet(flowControl.total()) == 0) { report(); } } else {//? TODO PushCenter.I.delayTask(flowControl.getDelay(), this); } flowControl.end(); } private boolean broadcast() { try { iterator.forEachRemaining(entry -> { String userId = entry.getKey(); entry.getValue().forEach((clientType, router) -> { Connection connection = router.getRouteValue(); if (checkCondition(condition, connection)) {//1.? if (connection.isConnected()) { if (connection.getChannel().isWritable()) { //TCP? PushMessage.build(connection).setContent(message.getContent()).send(this); //4. qps, ?????catch if (!flowControl.checkQps()) { throw new OverFlowException(false); } } } else { //2.?? Logs.PUSH.warn( "[Broadcast] find router in local but conn disconnect, message={}, conn={}", message, connection); //? RouterCenter.I.getLocalRouterManager().unRegister(userId, clientType); } } }); }); } catch (OverFlowException e) { //???? return e.isOverMaxLimit() || !iterator.hasNext(); } return !iterator.hasNext();//??, ? } private void report() { Logs.PUSH.info("[Broadcast] task finished, cost={}, message={}", (System.currentTimeMillis() - begin), message); PushCenter.I.getPushListener().onBroadcastComplete(message, timeLine.end().getTimePoints());//??? } private boolean checkCondition(Condition condition, Connection connection) { if (condition == AwaysPassCondition.I) return true; SessionContext sessionContext = connection.getSessionContext(); Map<String, Object> env = new HashMap<>(); env.put("userId", sessionContext.userId); env.put("tags", sessionContext.tags); env.put("clientVersion", sessionContext.clientVersion); env.put("osName", sessionContext.osName); env.put("osVersion", sessionContext.osVersion); return condition.test(env); } @Override public void operationComplete(ChannelFuture future) throws Exception { if (future.isSuccess()) {//?? Logs.PUSH.info("[Broadcast] push message to client success, userId={}, message={}", message.getUserId(), message); } else {//? Logs.PUSH.warn("[Broadcast] push message to client failure, userId={}, message={}, conn={}", message.getUserId(), message, future.channel()); } if (finishTasks.decrementAndGet() == 0) { report(); } } @Override public ScheduledExecutorService getExecutor() { return ((Message) message).getConnection().getChannel().eventLoop(); } }