/* * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You 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. */ package org.apache.solr.update.processor; import java.io.IOException; import java.lang.invoke.MethodHandles; import java.util.ArrayList; import java.util.List; import org.apache.solr.common.params.SolrParams; import org.apache.solr.common.util.NamedList; import org.apache.solr.common.util.SimpleOrderedMap; import org.apache.solr.request.SolrQueryRequest; import org.apache.solr.response.SolrQueryResponse; import org.apache.solr.update.AddUpdateCommand; import org.apache.solr.update.CommitUpdateCommand; import org.apache.solr.update.DeleteUpdateCommand; import org.apache.solr.update.MergeIndexesCommand; import org.apache.solr.update.RollbackUpdateCommand; import org.slf4j.Logger; import org.slf4j.LoggerFactory; /** * <p> * A logging processor. This keeps track of all commands that have passed through * the chain and prints them on finish(). At the Debug (FINE) level, a message * will be logged for each command prior to the next stage in the chain. * </p> * <p> * If the Log level is not >= INFO the processor will not be created or added to the chain. * </p> * * @since solr 1.3 */ public class LogUpdateProcessorFactory extends UpdateRequestProcessorFactory implements UpdateRequestProcessorFactory.RunAlways { private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass()); int maxNumToLog = 10; int slowUpdateThresholdMillis = -1; @Override public void init( final NamedList args ) { if( args != null ) { SolrParams params = SolrParams.toSolrParams( args ); maxNumToLog = params.getInt( "maxNumToLog", maxNumToLog ); slowUpdateThresholdMillis = params.getInt("slowUpdateThresholdMillis", slowUpdateThresholdMillis); } } @Override public UpdateRequestProcessor getInstance(SolrQueryRequest req, SolrQueryResponse rsp, UpdateRequestProcessor next) { return log.isInfoEnabled() ? new LogUpdateProcessor(req, rsp, this, next) : null; } static class LogUpdateProcessor extends UpdateRequestProcessor { private final SolrQueryRequest req; private final SolrQueryResponse rsp; private final NamedList<Object> toLog; int numAdds; int numDeletes; // hold on to the added list for logging and the response private List<String> adds; private List<String> deletes; private final int maxNumToLog; private final int slowUpdateThresholdMillis; private final boolean logDebug = log.isDebugEnabled();//cache to avoid volatile-read public LogUpdateProcessor(SolrQueryRequest req, SolrQueryResponse rsp, LogUpdateProcessorFactory factory, UpdateRequestProcessor next) { super( next ); this.req = req; this.rsp = rsp; maxNumToLog = factory.maxNumToLog; // TODO: make configurable // TODO: make log level configurable as well, or is that overkill? // (ryan) maybe? I added it mostly to show that it *can* be configurable slowUpdateThresholdMillis = factory.slowUpdateThresholdMillis; this.toLog = new SimpleOrderedMap<>(); } @Override public void processAdd(AddUpdateCommand cmd) throws IOException { if (logDebug) { log.debug("PRE_UPDATE " + cmd.toString() + " " + req); } // call delegate first so we can log things like the version that get set later if (next != null) next.processAdd(cmd); // Add a list of added id's to the response if (adds == null) { adds = new ArrayList<>(); toLog.add("add",adds); } if (adds.size() < maxNumToLog) { long version = cmd.getVersion(); String msg = cmd.getPrintableId(); if (version != 0) msg = msg + " (" + version + ')'; adds.add(msg); } numAdds++; } @Override public void processDelete( DeleteUpdateCommand cmd ) throws IOException { if (logDebug) { log.debug("PRE_UPDATE " + cmd.toString() + " " + req); } if (next != null) next.processDelete(cmd); if (cmd.isDeleteById()) { if (deletes == null) { deletes = new ArrayList<>(); toLog.add("delete",deletes); } if (deletes.size() < maxNumToLog) { long version = cmd.getVersion(); String msg = cmd.getId(); if (version != 0) msg = msg + " (" + version + ')'; deletes.add(msg); } } else { if (toLog.size() < maxNumToLog) { long version = cmd.getVersion(); String msg = cmd.query; if (version != 0) msg = msg + " (" + version + ')'; toLog.add("deleteByQuery", msg); } } numDeletes++; } @Override public void processMergeIndexes(MergeIndexesCommand cmd) throws IOException { if (logDebug) { log.debug("PRE_UPDATE " + cmd.toString() + " " + req); } if (next != null) next.processMergeIndexes(cmd); toLog.add("mergeIndexes", cmd.toString()); } @Override public void processCommit( CommitUpdateCommand cmd ) throws IOException { if (logDebug) { log.debug("PRE_UPDATE " + cmd.toString() + " " + req); } if (next != null) next.processCommit(cmd); final String msg = cmd.optimize ? "optimize" : "commit"; toLog.add(msg, ""); } /** * @since Solr 1.4 */ @Override public void processRollback( RollbackUpdateCommand cmd ) throws IOException { if (logDebug) { log.debug("PRE_UPDATE " + cmd.toString() + " " + req); } if (next != null) next.processRollback(cmd); toLog.add("rollback", ""); } @Override public void finish() throws IOException { if (logDebug) { log.debug("PRE_UPDATE FINISH " + req); } if (next != null) next.finish(); // LOG A SUMMARY WHEN ALL DONE (INFO LEVEL) if (log.isInfoEnabled()) { log.info(getLogStringAndClearRspToLog()); } if (log.isWarnEnabled() && slowUpdateThresholdMillis >= 0) { final long elapsed = (long) req.getRequestTimer().getTime(); if (elapsed >= slowUpdateThresholdMillis) { log.warn("slow: " + getLogStringAndClearRspToLog()); } } } private String getLogStringAndClearRspToLog() { StringBuilder sb = new StringBuilder(rsp.getToLogAsString(req.getCore().getLogId())); rsp.getToLog().clear(); // make it so SolrCore.exec won't log this again // if id lists were truncated, show how many more there were if (adds != null && numAdds > maxNumToLog) { adds.add("... (" + numAdds + " adds)"); } if (deletes != null && numDeletes > maxNumToLog) { deletes.add("... (" + numDeletes + " deletes)"); } final long elapsed = (long) req.getRequestTimer().getTime(); sb.append(toLog).append(" 0 ").append(elapsed); return sb.toString(); } } }