SOLR-13292: Provide extended per-segment status of a collection.

This commit is contained in:
Andrzej Bialecki 2019-03-09 14:08:01 +01:00
parent 83ab355772
commit f6f5f995ef
11 changed files with 866 additions and 16 deletions

View File

@ -53,6 +53,8 @@ New Features
* SOLR-13271: Read-only mode for SolrCloud collections (ab, shalin) * SOLR-13271: Read-only mode for SolrCloud collections (ab, shalin)
* SOLR-13292: Provide extended per-segment status of a collection. (ab)
Bug Fixes Bug Fixes
---------------------- ----------------------

View File

@ -0,0 +1,197 @@
/*
* 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.handler.admin;
import java.io.IOException;
import java.lang.invoke.MethodHandles;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import java.util.TreeMap;
import java.util.TreeSet;
import org.apache.http.client.HttpClient;
import org.apache.solr.client.solrj.SolrClient;
import org.apache.solr.client.solrj.SolrServerException;
import org.apache.solr.client.solrj.io.SolrClientCache;
import org.apache.solr.client.solrj.request.QueryRequest;
import org.apache.solr.common.cloud.ClusterState;
import org.apache.solr.common.cloud.DocCollection;
import org.apache.solr.common.cloud.Replica;
import org.apache.solr.common.cloud.RoutingRule;
import org.apache.solr.common.cloud.Slice;
import org.apache.solr.common.cloud.ZkCoreNodeProps;
import org.apache.solr.common.cloud.ZkNodeProps;
import org.apache.solr.common.cloud.ZkStateReader;
import org.apache.solr.common.params.CommonParams;
import org.apache.solr.common.params.ModifiableSolrParams;
import org.apache.solr.common.util.NamedList;
import org.apache.solr.common.util.SimpleOrderedMap;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Report low-level details of collection.
*/
public class ColStatus {
private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
private final ClusterState clusterState;
private final ZkNodeProps props;
private final SolrClientCache solrClientCache;
public static final String CORE_INFO_PROP = SegmentsInfoRequestHandler.WITH_CORE_INFO;
public static final String FIELD_INFO_PROP = SegmentsInfoRequestHandler.WITH_FIELD_INFO;
public static final String SIZE_INFO_PROP = SegmentsInfoRequestHandler.WITH_SIZE_INFO;
public static final String SEGMENTS_PROP = "segments";
public ColStatus(HttpClient httpClient, ClusterState clusterState, ZkNodeProps props) {
this.props = props;
this.solrClientCache = new SolrClientCache(httpClient);
this.clusterState = clusterState;
}
public void getColStatus(NamedList<Object> results) {
Collection<String> collections;
String col = props.getStr(ZkStateReader.COLLECTION_PROP);
if (col == null) {
collections = new HashSet<>(clusterState.getCollectionsMap().keySet());
} else {
collections = Collections.singleton(col);
}
boolean withFieldInfo = props.getBool(FIELD_INFO_PROP, false);
boolean withSegments = props.getBool(SEGMENTS_PROP, false);
boolean withCoreInfo = props.getBool(CORE_INFO_PROP, false);
boolean withSizeInfo = props.getBool(SIZE_INFO_PROP, false);
if (withFieldInfo || withSizeInfo) {
withSegments = true;
}
for (String collection : collections) {
DocCollection coll = clusterState.getCollectionOrNull(collection);
if (coll == null) {
continue;
}
SimpleOrderedMap<Object> colMap = new SimpleOrderedMap<>();
colMap.add("stateFormat", coll.getStateFormat());
colMap.add("znodeVersion", coll.getZNodeVersion());
Map<String, Object> props = new TreeMap<>(coll.getProperties());
props.remove("shards");
colMap.add("properties", props);
colMap.add("activeShards", coll.getActiveSlices().size());
colMap.add("inactiveShards", coll.getSlices().size() - coll.getActiveSlices().size());
results.add(collection, colMap);
Set<String> nonCompliant = new TreeSet<>();
SimpleOrderedMap<Object> shards = new SimpleOrderedMap<>();
for (Slice s : coll.getSlices()) {
SimpleOrderedMap<Object> sliceMap = new SimpleOrderedMap<>();
shards.add(s.getName(), sliceMap);
SimpleOrderedMap<Object> replicaMap = new SimpleOrderedMap<>();
int totalReplicas = s.getReplicas().size();
int activeReplicas = 0;
int downReplicas = 0;
int recoveringReplicas = 0;
int recoveryFailedReplicas = 0;
for (Replica r : s.getReplicas()) {
switch (r.getState()) {
case ACTIVE:
activeReplicas++;
break;
case DOWN:
downReplicas++;
break;
case RECOVERING:
recoveringReplicas++;
break;
case RECOVERY_FAILED:
recoveryFailedReplicas++;
break;
}
}
replicaMap.add("total", totalReplicas);
replicaMap.add("active", activeReplicas);
replicaMap.add("down", downReplicas);
replicaMap.add("recovering", recoveringReplicas);
replicaMap.add("recovery_failed", recoveryFailedReplicas);
sliceMap.add("state", s.getState().toString());
sliceMap.add("range", s.getRange().toString());
Map<String, RoutingRule> rules = s.getRoutingRules();
if (rules != null && !rules.isEmpty()) {
sliceMap.add("routingRules", rules);
}
sliceMap.add("replicas", replicaMap);
Replica leader = s.getLeader();
if (leader == null) { // pick the first one
leader = s.getReplicas().size() > 0 ? s.getReplicas().iterator().next() : null;
}
if (leader == null) {
continue;
}
SimpleOrderedMap<Object> leaderMap = new SimpleOrderedMap<>();
sliceMap.add("leader", leaderMap);
leaderMap.add("coreNode", leader.getName());
leaderMap.addAll(leader.getProperties());
String url = ZkCoreNodeProps.getCoreUrl(leader);
try (SolrClient client = solrClientCache.getHttpSolrClient(url)) {
ModifiableSolrParams params = new ModifiableSolrParams();
params.add(CommonParams.QT, "/admin/segments");
params.add(FIELD_INFO_PROP, "true");
params.add(CORE_INFO_PROP, String.valueOf(withCoreInfo));
params.add(SIZE_INFO_PROP, String.valueOf(withSizeInfo));
QueryRequest req = new QueryRequest(params);
NamedList<Object> rsp = client.request(req);
rsp.remove("responseHeader");
leaderMap.add("segInfos", rsp);
NamedList<Object> segs = (NamedList<Object>)rsp.get("segments");
if (segs != null) {
for (Map.Entry<String, Object> entry : segs) {
NamedList<Object> fields = (NamedList<Object>)((NamedList<Object>)entry.getValue()).get("fields");
if (fields != null) {
for (Map.Entry<String, Object> fEntry : fields) {
Object nc = ((NamedList<Object>)fEntry.getValue()).get("nonCompliant");
if (nc != null) {
nonCompliant.add(fEntry.getKey());
}
}
}
if (!withFieldInfo) {
((NamedList<Object>)entry.getValue()).remove("fields");
}
}
}
if (!withSegments) {
rsp.remove("segments");
}
if (!withFieldInfo) {
rsp.remove("fieldInfoLegend");
}
} catch (SolrServerException | IOException e) {
log.warn("Error getting details of replica segments from " + url, e);
}
}
if (nonCompliant.isEmpty()) {
nonCompliant.add("(NONE)");
}
colMap.add("schemaNonCompliant", nonCompliant);
colMap.add("shards", shards);
}
}
}

View File

@ -520,6 +520,22 @@ public class CollectionsHandler extends RequestHandlerBase implements Permission
return copyPropertiesWithPrefix(req.getParams(), props, "router."); return copyPropertiesWithPrefix(req.getParams(), props, "router.");
}), }),
COLSTATUS_OP(COLSTATUS, (req, rsp, h) -> {
Map<String, Object> props = copy(req.getParams(), null,
COLLECTION_PROP,
ColStatus.CORE_INFO_PROP,
ColStatus.SEGMENTS_PROP,
ColStatus.FIELD_INFO_PROP,
ColStatus.SIZE_INFO_PROP);
// make sure we can get the name if there's "name" but not "collection"
if (props.containsKey(CoreAdminParams.NAME) && !props.containsKey(COLLECTION_PROP)) {
props.put(COLLECTION_PROP, props.get(CoreAdminParams.NAME));
}
new ColStatus(h.coreContainer.getUpdateShardHandler().getDefaultHttpClient(),
h.coreContainer.getZkController().getZkStateReader().getClusterState(), new ZkNodeProps(props))
.getColStatus(rsp.getValues());
return null;
}),
DELETE_OP(DELETE, (req, rsp, h) -> copy(req.getParams().required(), null, NAME)), DELETE_OP(DELETE, (req, rsp, h) -> copy(req.getParams().required(), null, NAME)),
RELOAD_OP(RELOAD, (req, rsp, h) -> copy(req.getParams().required(), null, NAME)), RELOAD_OP(RELOAD, (req, rsp, h) -> copy(req.getParams().required(), null, NAME)),

View File

@ -17,72 +17,187 @@
package org.apache.solr.handler.admin; package org.apache.solr.handler.admin;
import java.io.IOException; import java.io.IOException;
import java.lang.invoke.MethodHandles;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Date; import java.util.Date;
import java.util.LinkedHashMap;
import java.util.List; import java.util.List;
import java.util.function.BiConsumer;
import java.util.stream.Collectors;
import org.apache.lucene.index.DocValuesType;
import org.apache.lucene.index.FieldInfo;
import org.apache.lucene.index.FieldInfos;
import org.apache.lucene.index.FilterLeafReader;
import org.apache.lucene.index.IndexOptions;
import org.apache.lucene.index.IndexWriter; import org.apache.lucene.index.IndexWriter;
import org.apache.lucene.index.LeafMetaData;
import org.apache.lucene.index.LeafReader;
import org.apache.lucene.index.LeafReaderContext;
import org.apache.lucene.index.MergePolicy; import org.apache.lucene.index.MergePolicy;
import org.apache.lucene.index.MergePolicy.MergeSpecification; import org.apache.lucene.index.MergePolicy.MergeSpecification;
import org.apache.lucene.index.MergePolicy.OneMerge; import org.apache.lucene.index.MergePolicy.OneMerge;
import org.apache.lucene.index.MergeTrigger; import org.apache.lucene.index.MergeTrigger;
import org.apache.lucene.index.SegmentCommitInfo; import org.apache.lucene.index.SegmentCommitInfo;
import org.apache.lucene.index.SegmentInfos; import org.apache.lucene.index.SegmentInfos;
import org.apache.lucene.index.SegmentReader;
import org.apache.lucene.index.Terms;
import org.apache.lucene.store.Directory;
import org.apache.lucene.util.Accountable;
import org.apache.lucene.util.RamUsageEstimator;
import org.apache.lucene.util.Version;
import org.apache.solr.common.luke.FieldFlag;
import org.apache.solr.common.util.Pair;
import org.apache.solr.common.util.SimpleOrderedMap; import org.apache.solr.common.util.SimpleOrderedMap;
import org.apache.solr.core.SolrCore;
import org.apache.solr.handler.RequestHandlerBase; import org.apache.solr.handler.RequestHandlerBase;
import org.apache.solr.request.SolrQueryRequest; import org.apache.solr.request.SolrQueryRequest;
import org.apache.solr.response.SolrQueryResponse; import org.apache.solr.response.SolrQueryResponse;
import org.apache.solr.schema.IndexSchema;
import org.apache.solr.schema.SchemaField;
import org.apache.solr.search.SolrIndexSearcher; import org.apache.solr.search.SolrIndexSearcher;
import org.apache.solr.update.SolrIndexWriter;
import org.apache.solr.util.RefCounted; import org.apache.solr.util.RefCounted;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import static org.apache.lucene.index.IndexOptions.DOCS;
import static org.apache.lucene.index.IndexOptions.DOCS_AND_FREQS;
import static org.apache.lucene.index.IndexOptions.DOCS_AND_FREQS_AND_POSITIONS_AND_OFFSETS;
import static org.apache.solr.common.params.CommonParams.NAME; import static org.apache.solr.common.params.CommonParams.NAME;
/** /**
* This handler exposes information about last commit generation segments * This handler exposes information about last commit generation segments
*/ */
public class SegmentsInfoRequestHandler extends RequestHandlerBase { public class SegmentsInfoRequestHandler extends RequestHandlerBase {
private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
public static final String WITH_FIELD_INFO = "fieldInfo";
public static final String WITH_CORE_INFO = "coreInfo";
public static final String WITH_SIZE_INFO = "sizeInfo";
private static final List<String> FI_LEGEND;
static {
FI_LEGEND = Arrays.asList(
FieldFlag.INDEXED.toString(),
FieldFlag.DOC_VALUES.toString(),
"xxx - DocValues type",
FieldFlag.TERM_VECTOR_STORED.toString(),
FieldFlag.OMIT_NORMS.toString(),
FieldFlag.OMIT_TF.toString(),
FieldFlag.OMIT_POSITIONS.toString(),
FieldFlag.STORE_OFFSETS_WITH_POSITIONS.toString(),
"p - field has payloads",
"s - field uses soft deletes",
":x:x:x - point data dim : index dim : num bytes");
}
@Override @Override
public void handleRequestBody(SolrQueryRequest req, SolrQueryResponse rsp) public void handleRequestBody(SolrQueryRequest req, SolrQueryResponse rsp)
throws Exception { throws Exception {
rsp.add("segments", getSegmentsInfo(req, rsp)); getSegmentsInfo(req, rsp);
rsp.setHttpCaching(false); rsp.setHttpCaching(false);
} }
private SimpleOrderedMap<Object> getSegmentsInfo(SolrQueryRequest req, SolrQueryResponse rsp) private static final double GB = 1024.0 * 1024.0 * 1024.0;
private void getSegmentsInfo(SolrQueryRequest req, SolrQueryResponse rsp)
throws Exception { throws Exception {
boolean withFieldInfo = req.getParams().getBool(WITH_FIELD_INFO, false);
boolean withCoreInfo = req.getParams().getBool(WITH_CORE_INFO, false);
boolean withSizeInfo = req.getParams().getBool(WITH_SIZE_INFO, false);
SolrIndexSearcher searcher = req.getSearcher(); SolrIndexSearcher searcher = req.getSearcher();
SegmentInfos infos = SegmentInfos infos =
SegmentInfos.readLatestCommit(searcher.getIndexReader().directory()); SegmentInfos.readLatestCommit(searcher.getIndexReader().directory());
List<String> mergeCandidates = getMergeCandidatesNames(req, infos);
SimpleOrderedMap<Object> segmentInfos = new SimpleOrderedMap<>(); SimpleOrderedMap<Object> segmentInfos = new SimpleOrderedMap<>();
SolrCore core = req.getCore();
RefCounted<IndexWriter> iwRef = core.getSolrCoreState().getIndexWriter(core);
SimpleOrderedMap<Object> infosInfo = new SimpleOrderedMap<>();
Version minVersion = infos.getMinSegmentLuceneVersion();
if (minVersion != null) {
infosInfo.add("minSegmentLuceneVersion", minVersion.toString());
}
Version commitVersion = infos.getCommitLuceneVersion();
if (commitVersion != null) {
infosInfo.add("commitLuceneVersion", commitVersion.toString());
}
infosInfo.add("numSegments", infos.size());
infosInfo.add("segmentsFileName", infos.getSegmentsFileName());
infosInfo.add("totalMaxDoc", infos.totalMaxDoc());
infosInfo.add("userData", infos.userData);
if (withCoreInfo) {
SimpleOrderedMap<Object> coreInfo = new SimpleOrderedMap<>();
infosInfo.add("core", coreInfo);
coreInfo.add("startTime", core.getStartTimeStamp().getTime() + "(" + core.getStartTimeStamp() + ")");
coreInfo.add("dataDir", core.getDataDir());
coreInfo.add("indexDir", core.getIndexDir());
coreInfo.add("sizeInGB", (double)core.getIndexSize() / GB);
if (iwRef != null) {
try {
IndexWriter iw = iwRef.get();
String iwConfigStr = iw.getConfig().toString();
SimpleOrderedMap<Object> iwConfig = new SimpleOrderedMap<>();
// meh ...
String[] lines = iwConfigStr.split("\\n");
for (String line : lines) {
String[] parts = line.split("=");
if (parts.length < 2) {
continue;
}
iwConfig.add(parts[0], parts[1]);
}
coreInfo.add("indexWriterConfig", iwConfig);
} finally {
iwRef.decref();
}
}
}
SimpleOrderedMap<Object> segmentInfo = null; SimpleOrderedMap<Object> segmentInfo = null;
List<SegmentCommitInfo> sortable = new ArrayList<>(); List<SegmentCommitInfo> sortable = new ArrayList<>();
sortable.addAll(infos.asList()); sortable.addAll(infos.asList());
// Order by the number of live docs. The display is logarithmic so it is a little jumbled visually // Order by the number of live docs. The display is logarithmic so it is a little jumbled visually
sortable.sort((s1, s2) -> { sortable.sort((s1, s2) ->
return (s2.info.maxDoc() - s2.getDelCount()) - (s1.info.maxDoc() - s1.getDelCount()); (s2.info.maxDoc() - s2.getDelCount()) - (s1.info.maxDoc() - s1.getDelCount())
}); );
List<String> mergeCandidates = new ArrayList<>();
SimpleOrderedMap<Object> runningMerges = getMergeInformation(req, infos, mergeCandidates);
List<LeafReaderContext> leafContexts = searcher.getIndexReader().leaves();
IndexSchema schema = req.getSchema();
for (SegmentCommitInfo segmentCommitInfo : sortable) { for (SegmentCommitInfo segmentCommitInfo : sortable) {
segmentInfo = getSegmentInfo(segmentCommitInfo); segmentInfo = getSegmentInfo(segmentCommitInfo, withSizeInfo, withFieldInfo, leafContexts, schema);
if (mergeCandidates.contains(segmentCommitInfo.info.name)) { if (mergeCandidates.contains(segmentCommitInfo.info.name)) {
segmentInfo.add("mergeCandidate", true); segmentInfo.add("mergeCandidate", true);
} }
segmentInfos.add((String) segmentInfo.get(NAME), segmentInfo); segmentInfos.add((String) segmentInfo.get(NAME), segmentInfo);
} }
return segmentInfos; rsp.add("info", infosInfo);
if (runningMerges.size() > 0) {
rsp.add("runningMerges", runningMerges);
}
if (withFieldInfo) {
rsp.add("fieldInfoLegend", FI_LEGEND);
}
rsp.add("segments", segmentInfos);
} }
private SimpleOrderedMap<Object> getSegmentInfo( private SimpleOrderedMap<Object> getSegmentInfo(
SegmentCommitInfo segmentCommitInfo) throws IOException { SegmentCommitInfo segmentCommitInfo, boolean withSizeInfo, boolean withFieldInfos,
List<LeafReaderContext> leafContexts, IndexSchema schema) throws IOException {
SimpleOrderedMap<Object> segmentInfoMap = new SimpleOrderedMap<>(); SimpleOrderedMap<Object> segmentInfoMap = new SimpleOrderedMap<>();
segmentInfoMap.add(NAME, segmentCommitInfo.info.name); segmentInfoMap.add(NAME, segmentCommitInfo.info.name);
segmentInfoMap.add("delCount", segmentCommitInfo.getDelCount()); segmentInfoMap.add("delCount", segmentCommitInfo.getDelCount());
segmentInfoMap.add("softDelCount", segmentCommitInfo.getSoftDelCount());
segmentInfoMap.add("hasFieldUpdates", segmentCommitInfo.hasFieldUpdates());
segmentInfoMap.add("sizeInBytes", segmentCommitInfo.sizeInBytes()); segmentInfoMap.add("sizeInBytes", segmentCommitInfo.sizeInBytes());
segmentInfoMap.add("size", segmentCommitInfo.info.maxDoc()); segmentInfoMap.add("size", segmentCommitInfo.info.maxDoc());
Long timestamp = Long.parseLong(segmentCommitInfo.info.getDiagnostics() Long timestamp = Long.parseLong(segmentCommitInfo.info.getDiagnostics()
@ -91,15 +206,224 @@ public class SegmentsInfoRequestHandler extends RequestHandlerBase {
segmentInfoMap.add("source", segmentInfoMap.add("source",
segmentCommitInfo.info.getDiagnostics().get("source")); segmentCommitInfo.info.getDiagnostics().get("source"));
segmentInfoMap.add("version", segmentCommitInfo.info.getVersion().toString()); segmentInfoMap.add("version", segmentCommitInfo.info.getVersion().toString());
// don't open a new SegmentReader - try to find the right one from the leaf contexts
SegmentReader seg = null;
for (LeafReaderContext lrc : leafContexts) {
LeafReader leafReader = lrc.reader();
// unwrap
while (leafReader instanceof FilterLeafReader) {
leafReader = ((FilterLeafReader)leafReader).getDelegate();
}
if (leafReader instanceof SegmentReader) {
SegmentReader sr = (SegmentReader)leafReader;
if (sr.getSegmentInfo().info.equals(segmentCommitInfo.info)) {
seg = sr;
break;
}
}
}
if (seg != null) {
LeafMetaData metaData = seg.getMetaData();
if (metaData != null) {
segmentInfoMap.add("createdVersionMajor", metaData.getCreatedVersionMajor());
segmentInfoMap.add("minVersion", metaData.getMinVersion().toString());
if (metaData.getSort() != null) {
segmentInfoMap.add("sort", metaData.getSort().toString());
}
}
}
if (!segmentCommitInfo.info.getDiagnostics().isEmpty()) {
segmentInfoMap.add("diagnostics", segmentCommitInfo.info.getDiagnostics());
}
if (!segmentCommitInfo.info.getAttributes().isEmpty()) {
segmentInfoMap.add("attributes", segmentCommitInfo.info.getAttributes());
}
if (withSizeInfo) {
Directory dir = segmentCommitInfo.info.dir;
List<Pair<String, Long>> files = segmentCommitInfo.files().stream()
.map(f -> {
long size = -1;
try {
size = dir.fileLength(f);
} catch (IOException e) {
}
return new Pair<String, Long>(f, size);
}).sorted((p1, p2) -> {
if (p1.second() > p2.second()) {
return -1;
} else if (p1.second() < p2.second()) {
return 1;
} else {
return 0;
}
}).collect(Collectors.toList());
if (!files.isEmpty()) {
SimpleOrderedMap<Object> topFiles = new SimpleOrderedMap<>();
for (int i = 0; i < Math.min(files.size(), 5); i++) {
Pair<String, Long> p = files.get(i);
topFiles.add(p.first(), RamUsageEstimator.humanReadableUnits(p.second()));
}
segmentInfoMap.add("largestFiles", topFiles);
}
}
if (seg != null && withSizeInfo) {
SimpleOrderedMap<Object> ram = new SimpleOrderedMap<>();
ram.add("total", seg.ramBytesUsed());
for (Accountable ac : seg.getChildResources()) {
accountableToMap(ac, ram::add);
}
segmentInfoMap.add("ramBytesUsed", ram);
}
if (withFieldInfos) {
if (seg == null) {
log.debug("Skipping segment info - not available as a SegmentReader: " + segmentCommitInfo);
} else {
FieldInfos fis = seg.getFieldInfos();
SimpleOrderedMap<Object> fields = new SimpleOrderedMap<>();
for (FieldInfo fi : fis) {
fields.add(fi.name, getFieldInfo(seg, fi, schema));
}
segmentInfoMap.add("fields", fields);
}
}
return segmentInfoMap; return segmentInfoMap;
} }
private List<String> getMergeCandidatesNames(SolrQueryRequest req, SegmentInfos infos) throws IOException { private void accountableToMap(Accountable accountable, BiConsumer<String, Object> consumer) {
List<String> result = new ArrayList<String>(); Collection<Accountable> children = accountable.getChildResources();
if (children != null && !children.isEmpty()) {
LinkedHashMap<String, Object> map = new LinkedHashMap<>();
map.put("total", accountable.ramBytesUsed());
for (Accountable child : children) {
accountableToMap(child, map::put);
}
consumer.accept(accountable.toString(), map);
} else {
consumer.accept(accountable.toString(), accountable.ramBytesUsed());
}
}
private SimpleOrderedMap<Object> getFieldInfo(SegmentReader reader, FieldInfo fi, IndexSchema schema) {
SimpleOrderedMap<Object> fieldFlags = new SimpleOrderedMap<>();
StringBuilder flags = new StringBuilder();
IndexOptions opts = fi.getIndexOptions();
flags.append( (opts != IndexOptions.NONE) ? FieldFlag.INDEXED.getAbbreviation() : '-' );
DocValuesType dvt = fi.getDocValuesType();
if (dvt != DocValuesType.NONE) {
flags.append(FieldFlag.DOC_VALUES.getAbbreviation());
switch (dvt) {
case NUMERIC:
flags.append("num");
break;
case BINARY:
flags.append("bin");
break;
case SORTED:
flags.append("srt");
break;
case SORTED_NUMERIC:
flags.append("srn");
break;
case SORTED_SET:
flags.append("srs");
break;
default:
flags.append("???"); // should not happen
}
} else {
flags.append("----");
}
flags.append( (fi.hasVectors()) ? FieldFlag.TERM_VECTOR_STORED.getAbbreviation() : '-' );
flags.append( (fi.omitsNorms()) ? FieldFlag.OMIT_NORMS.getAbbreviation() : '-' );
flags.append( (DOCS == opts ) ?
FieldFlag.OMIT_TF.getAbbreviation() : '-' );
flags.append((DOCS_AND_FREQS == opts) ?
FieldFlag.OMIT_POSITIONS.getAbbreviation() : '-');
flags.append((DOCS_AND_FREQS_AND_POSITIONS_AND_OFFSETS == opts) ?
FieldFlag.STORE_OFFSETS_WITH_POSITIONS.getAbbreviation() : '-');
flags.append( (fi.hasPayloads() ? "p" : "-"));
flags.append( (fi.isSoftDeletesField() ? "s" : "-"));
if (fi.getPointDataDimensionCount() > 0 || fi.getPointIndexDimensionCount() > 0) {
flags.append(":");
flags.append(fi.getPointDataDimensionCount() + ":");
flags.append(fi.getPointIndexDimensionCount() + ":");
flags.append(fi.getPointNumBytes());
}
fieldFlags.add("flags", flags.toString());
try {
Terms terms = reader.terms(fi.name);
if (terms != null) {
fieldFlags.add("docCount", terms.getDocCount());
fieldFlags.add("sumDocFreq", terms.getSumDocFreq());
fieldFlags.add("sumTotalTermFreq", terms.getSumTotalTermFreq());
}
} catch (Exception e) {
log.debug("Exception retrieving term stats for field " + fi.name, e);
}
// probably too much detail?
// Map<String, String> attributes = fi.attributes();
// if (!attributes.isEmpty()) {
// fieldFlags.add("attributes", attributes);
// }
// check compliance of the index with the current schema
SchemaField sf = schema.getFieldOrNull(fi.name);
boolean hasPoints = fi.getPointDataDimensionCount() > 0 || fi.getPointIndexDimensionCount() > 0;
if (sf != null) {
fieldFlags.add("schemaType", sf.getType().getTypeName());
SimpleOrderedMap<Object> nonCompliant = new SimpleOrderedMap<>();
if (sf.hasDocValues() &&
fi.getDocValuesType() == DocValuesType.NONE &&
fi.getIndexOptions() != IndexOptions.NONE) {
nonCompliant.add("docValues", "schema=" + sf.getType().getUninversionType(sf) + ", segment=false");
}
if (!sf.hasDocValues() &&
fi.getDocValuesType() != DocValuesType.NONE &&
fi.getIndexOptions() != IndexOptions.NONE) {
nonCompliant.add("docValues", "schema=false, segment=" + fi.getDocValuesType().toString());
}
if (!sf.isPolyField()) { // difficult to find all sub-fields in a general way
if (sf.indexed() != ((fi.getIndexOptions() != IndexOptions.NONE) || hasPoints)) {
nonCompliant.add("indexed", "schema=" + sf.indexed() + ", segment=" + fi.getIndexOptions());
}
}
if (sf.omitNorms() != (fi.omitsNorms() || hasPoints)) {
nonCompliant.add("omitNorms", "schema=" + sf.omitNorms() + ", segment=" + fi.omitsNorms());
}
if (sf.storeTermVector() != fi.hasVectors()) {
nonCompliant.add("termVectors", "schema=" + sf.storeTermVector() + ", segment=" + fi.hasVectors());
}
if (sf.storeOffsetsWithPositions() != (fi.getIndexOptions() == IndexOptions.DOCS_AND_FREQS_AND_POSITIONS_AND_OFFSETS)) {
nonCompliant.add("storeOffsetsWithPositions", "schema=" + sf.storeOffsetsWithPositions() + ", segment=" + fi.getIndexOptions());
}
if (nonCompliant.size() > 0) {
nonCompliant.add("schemaField", sf.toString());
fieldFlags.add("nonCompliant", nonCompliant);
}
} else {
fieldFlags.add("schemaType", "(UNKNOWN)");
}
return fieldFlags;
}
// returns a map of currently running merges, and populates a list of candidate segments for merge
private SimpleOrderedMap<Object> getMergeInformation(SolrQueryRequest req, SegmentInfos infos, List<String> mergeCandidates) throws IOException {
SimpleOrderedMap<Object> result = new SimpleOrderedMap<>();
RefCounted<IndexWriter> refCounted = req.getCore().getSolrCoreState().getIndexWriter(req.getCore()); RefCounted<IndexWriter> refCounted = req.getCore().getSolrCoreState().getIndexWriter(req.getCore());
try { try {
IndexWriter indexWriter = refCounted.get(); IndexWriter indexWriter = refCounted.get();
if (indexWriter instanceof SolrIndexWriter) {
result.addAll(((SolrIndexWriter)indexWriter).getRunningMerges());
}
//get chosen merge policy //get chosen merge policy
MergePolicy mp = indexWriter.getConfig().getMergePolicy(); MergePolicy mp = indexWriter.getConfig().getMergePolicy();
//Find merges //Find merges
@ -108,7 +432,7 @@ public class SegmentsInfoRequestHandler extends RequestHandlerBase {
for (OneMerge merge : findMerges.merges) { for (OneMerge merge : findMerges.merges) {
//TODO: add merge grouping //TODO: add merge grouping
for (SegmentCommitInfo mergeSegmentInfo : merge.segments) { for (SegmentCommitInfo mergeSegmentInfo : merge.segments) {
result.add(mergeSegmentInfo.info.name); mergeCandidates.add(mergeSegmentInfo.info.name);
} }
} }
} }

View File

@ -18,8 +18,10 @@ package org.apache.solr.update;
import java.io.IOException; import java.io.IOException;
import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodHandles;
import java.util.Collections;
import java.util.HashMap; import java.util.HashMap;
import java.util.Map; import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicInteger; import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicLong; import java.util.concurrent.atomic.AtomicLong;
@ -88,6 +90,8 @@ public class SolrIndexWriter extends IndexWriter {
private final SolrMetricManager metricManager; private final SolrMetricManager metricManager;
private final String registryName; private final String registryName;
// merge diagnostics.
private final Map<String, Long> runningMerges = new ConcurrentHashMap<>();
public static SolrIndexWriter create(SolrCore core, String name, String path, DirectoryFactory directoryFactory, boolean create, IndexSchema schema, SolrIndexConfig config, IndexDeletionPolicy delPolicy, Codec codec) throws IOException { public static SolrIndexWriter create(SolrCore core, String name, String path, DirectoryFactory directoryFactory, boolean create, IndexSchema schema, SolrIndexConfig config, IndexDeletionPolicy delPolicy, Codec codec) throws IOException {
@ -192,12 +196,18 @@ public class SolrIndexWriter extends IndexWriter {
// we override this method to collect metrics for merges. // we override this method to collect metrics for merges.
@Override @Override
public void merge(MergePolicy.OneMerge merge) throws IOException { public void merge(MergePolicy.OneMerge merge) throws IOException {
String segString = merge.segString();
long totalNumDocs = merge.totalNumDocs();
runningMerges.put(segString, totalNumDocs);
if (!mergeTotals) { if (!mergeTotals) {
try {
super.merge(merge); super.merge(merge);
} finally {
runningMerges.remove(segString);
}
return; return;
} }
long deletedDocs = 0; long deletedDocs = 0;
long totalNumDocs = merge.totalNumDocs();
for (SegmentCommitInfo info : merge.segments) { for (SegmentCommitInfo info : merge.segments) {
totalNumDocs -= info.getDelCount(); totalNumDocs -= info.getDelCount();
deletedDocs += info.getDelCount(); deletedDocs += info.getDelCount();
@ -226,6 +236,7 @@ public class SolrIndexWriter extends IndexWriter {
mergeErrors.inc(); mergeErrors.inc();
throw t; throw t;
} finally { } finally {
runningMerges.remove(segString);
context.stop(); context.stop();
if (major) { if (major) {
runningMajorMerges.decrementAndGet(); runningMajorMerges.decrementAndGet();
@ -239,6 +250,10 @@ public class SolrIndexWriter extends IndexWriter {
} }
} }
public Map<String, Object> getRunningMerges() {
return Collections.unmodifiableMap(runningMerges);
}
@Override @Override
protected void doAfterFlush() throws IOException { protected void doAfterFlush() throws IOException {
if (flushMeter != null) { // this is null when writer is used only for snapshot cleanup if (flushMeter != null) { // this is null when writer is used only for snapshot cleanup

View File

@ -39,6 +39,7 @@ import java.util.concurrent.atomic.AtomicReference;
import org.apache.lucene.util.LuceneTestCase; import org.apache.lucene.util.LuceneTestCase;
import org.apache.lucene.util.TestUtil; import org.apache.lucene.util.TestUtil;
import org.apache.solr.client.solrj.SolrClient;
import org.apache.solr.client.solrj.SolrRequest; import org.apache.solr.client.solrj.SolrRequest;
import org.apache.solr.client.solrj.SolrServerException; import org.apache.solr.client.solrj.SolrServerException;
import org.apache.solr.client.solrj.impl.CloudSolrClient; import org.apache.solr.client.solrj.impl.CloudSolrClient;
@ -601,6 +602,35 @@ public class CollectionsAPISolrJTest extends SolrCloudTestCase {
fail("Timed out waiting for cluster property value"); fail("Timed out waiting for cluster property value");
} }
@Test
public void testColStatus() throws Exception {
final String collectionName = "collectionStatusTest";
CollectionAdminRequest.createCollection(collectionName, "conf", 2, 2)
.process(cluster.getSolrClient());
cluster.waitForActiveCollection(collectionName, 2, 4);
SolrClient client = cluster.getSolrClient();
// index some docs
for (int i = 0; i < 10; i++) {
client.add(collectionName, new SolrInputDocument("id", String.valueOf(i)));
}
client.commit(collectionName);
CollectionAdminRequest.ColStatus req = CollectionAdminRequest.collectionStatus(collectionName);
req.setWithFieldInfo(true);
req.setWithCoreInfo(true);
req.setWithSegments(true);
req.setWithSizeInfo(true);
CollectionAdminResponse rsp = req.process(cluster.getSolrClient());
assertEquals(0, rsp.getStatus());
NamedList<Object> segInfos = (NamedList<Object>) rsp.getResponse().findRecursive(collectionName, "shards", "shard1", "leader", "segInfos");
assertNotNull(Utils.toJSONString(rsp), segInfos.findRecursive("info", "core", "startTime"));
assertNotNull(Utils.toJSONString(rsp), segInfos.get("fieldInfoLegend"));
assertNotNull(Utils.toJSONString(rsp), segInfos.findRecursive("segments", "_0", "fields", "id", "flags"));
assertNotNull(Utils.toJSONString(rsp), segInfos.findRecursive("segments", "_0", "ramBytesUsed"));
}
private static final int NUM_DOCS = 10; private static final int NUM_DOCS = 10;
@Test @Test

View File

@ -120,4 +120,31 @@ public class SegmentsInfoRequestHandlerTest extends SolrTestCaseJ4 {
//#Deletes //#Deletes
DEL_COUNT+"=sum(//lst[@name='segments']/lst/int[@name='delCount'])"); DEL_COUNT+"=sum(//lst[@name='segments']/lst/int[@name='delCount'])");
} }
@Test
public void testCoreInfo() {
assertQ("Missing core info",
req("qt", "/admin/segments", "coreInfo", "true"),
"boolean(//lst[@name='info']/lst[@name='core'])");
}
@Test
public void testFieldInfo() throws Exception {
String[] segmentNamePatterns = new String[NUM_SEGMENTS];
h.getCore().withSearcher((searcher) -> {
int i = 0;
for (SegmentCommitInfo sInfo : SegmentInfos.readLatestCommit(searcher.getIndexReader().directory())) {
assertTrue("Unexpected number of segment in the index: " + i, i < NUM_SEGMENTS);
segmentNamePatterns[i] = "boolean(//lst[@name='segments']/lst[@name='" + sInfo.info.name + "']/lst[@name='fields']/lst[@name='id']/str[@name='flags'])";
i++;
}
return null;
});
assertQ("Unexpected field infos returned",
req("qt","/admin/segments", "fieldInfo", "true"),
segmentNamePatterns);
}
} }

View File

@ -1334,6 +1334,192 @@ http://localhost:8983/solr/admin/collections?action=COLLECTIONPROP&name=coll&pro
</response> </response>
---- ----
[[colstatus]]
== COLSTATUS: Detailed low-level status of collection's indexes
The COLSTATUS command provides a detailed description of the collection status, including low-level index
information about segments and field data.
This command also checks the compliance of Lucene index field types with the current Solr collection
schema and indicates the names of non-compliant fields, ie. Lucene fields with field types incompatible
(or different) from the corresponding Solr field types declared in the current schema. Such incompatibilities may
result from incompatible schema changes or after migration of
data to a different major Solr release.
`/admin/collections?action=COLSTATUS&collection=coll&coreInfo=true&segments=true&fieldInfo=true&sizeInfo=true`
=== COLSTATUS Parameters
`collection`::
Collection name (optional). If missing then it means all collections.
`coreInfo`::
Optional boolean. If true then additional information will be provided about
SolrCore of shard leaders.
`segments`::
Optional boolean. If true then segment information will be provided.
`fieldInfo`::
Optional boolean. If true then detailed Lucene field information will be provided
and their corresponding Solr schema types.
`sizeInfo`::
Optional boolean. If true then additional information about the index files
size and their RAM usage will be provided.
=== COLSTATUS Response
The response will include an overview of the collection status, the number of
active / inactive shards and replicas, and additional index information
of shard leaders.
=== Examples using COLSTATUS
*Input*
[source,text]
----
http://localhost:8983/solr/admin/collections?action=COLSTATUS&collection=gettingstarted&fieldInfo=true&sizeInfo=true
----
*Output*
[source,json]
----
{
"responseHeader": {
"status": 0,
"QTime": 50
},
"gettingstarted": {
"stateFormat": 2,
"znodeVersion": 16,
"properties": {
"autoAddReplicas": "false",
"maxShardsPerNode": "-1",
"nrtReplicas": "2",
"pullReplicas": "0",
"replicationFactor": "2",
"router": {
"name": "compositeId"
},
"tlogReplicas": "0"
},
"activeShards": 2,
"inactiveShards": 0,
"schemaNonCompliant": [
"(NONE)"
],
"shards": {
"shard1": {
"state": "active",
"range": "80000000-ffffffff",
"replicas": {
"total": 2,
"active": 2,
"down": 0,
"recovering": 0,
"recovery_failed": 0
},
"leader": {
"coreNode": "core_node4",
"core": "gettingstarted_shard1_replica_n1",
"base_url": "http://192.168.0.80:8983/solr",
"node_name": "192.168.0.80:8983_solr",
"state": "active",
"type": "NRT",
"force_set_state": "false",
"leader": "true",
"segInfos": {
"info": {
"minSegmentLuceneVersion": "9.0.0",
"commitLuceneVersion": "9.0.0",
"numSegments": 40,
"segmentsFileName": "segments_w",
"totalMaxDoc": 686953,
"userData": {
"commitCommandVer": "1627350608019193856",
"commitTimeMSec": "1551962478819"
}
},
"fieldInfoLegend": [
"I - Indexed",
"D - DocValues",
"xxx - DocValues type",
"V - TermVector Stored",
"O - Omit Norms",
"F - Omit Term Frequencies & Positions",
"P - Omit Positions",
"H - Store Offsets with Positions",
"p - field has payloads",
"s - field uses soft deletes",
":x:x:x - point data dim : index dim : num bytes"
],
"segments": {
"_i": {
"name": "_i",
"delCount": 738,
"softDelCount": 0,
"hasFieldUpdates": false,
"sizeInBytes": 109398213,
"size": 70958,
"age": "2019-03-07T12:34:24.761Z",
"source": "merge",
"version": "9.0.0",
"createdVersionMajor": 9,
"minVersion": "9.0.0",
"diagnostics": {
"os": "Mac OS X",
"java.vendor": "Oracle Corporation",
"java.version": "1.8.0_191",
"java.vm.version": "25.191-b12",
"lucene.version": "9.0.0",
"mergeMaxNumSegments": "-1",
"os.arch": "x86_64",
"java.runtime.version": "1.8.0_191-b12",
"source": "merge",
"mergeFactor": "10",
"os.version": "10.14.3",
"timestamp": "1551962064761"
},
"attributes": {
"Lucene50StoredFieldsFormat.mode": "BEST_SPEED"
},
"largestFiles": {
"_i.fdt": "42.5 MB",
"_i_Lucene80_0.dvd": "35.3 MB",
"_i_Lucene50_0.pos": "11.1 MB",
"_i_Lucene50_0.doc": "10 MB",
"_i_Lucene50_0.tim": "4.3 MB"
},
"ramBytesUsed": {
"total": 49153,
"postings [PerFieldPostings(segment=_i formats=1)]": {
"total": 31023,
...
"fields": {
"dc": {
"flags": "I-----------",
"schemaType": "text_general"
},
"dc_str": {
"flags": "-Dsrs-------",
"schemaType": "strings"
},
"dc.title": {
"flags": "I-----------",
"docCount": 70958,
"sumDocFreq": 646756,
"sumTotalTermFreq": 671817,
"schemaType": "text_general"
},
"dc.date": {
"flags": "-Dsrn-------:1:1:8",
"schemaType": "pdates"
},
...
----
[[migrate]] [[migrate]]
== MIGRATE: Migrate Documents to Another Collection == MIGRATE: Migrate Documents to Another Collection

View File

@ -783,6 +783,54 @@ public abstract class CollectionAdminRequest<T extends CollectionAdminResponse>
} }
/**
* Return a SolrRequest for low-level detailed status of the collection.
*/
public static ColStatus collectionStatus(String collection) {
return new ColStatus(collection);
}
public static class ColStatus extends AsyncCollectionSpecificAdminRequest {
protected Boolean withSegments = null;
protected Boolean withFieldInfo = null;
protected Boolean withCoreInfo = null;
protected Boolean withSizeInfo = null;
private ColStatus(String collection) {
super(CollectionAction.COLSTATUS, collection);
}
public ColStatus setWithSegments(boolean withSegments) {
this.withSegments = withSegments;
return this;
}
public ColStatus setWithFieldInfo(boolean withFieldInfo) {
this.withFieldInfo = withFieldInfo;
return this;
}
public ColStatus setWithCoreInfo(boolean withCoreInfo) {
this.withCoreInfo = withCoreInfo;
return this;
}
public ColStatus setWithSizeInfo(boolean withSizeInfo) {
this.withSizeInfo = withSizeInfo;
return this;
}
@Override
public SolrParams getParams() {
ModifiableSolrParams params = (ModifiableSolrParams)super.getParams();
params.setNonNull("segments", withSegments.toString());
params.setNonNull("fieldInfo", withFieldInfo.toString());
params.setNonNull("coreInfo", withCoreInfo.toString());
params.setNonNull("sizeInfo", withSizeInfo.toString());
return params;
}
}
/** /**
* Returns a SolrRequest to delete a collection * Returns a SolrRequest to delete a collection
*/ */

View File

@ -67,4 +67,8 @@ public enum FieldFlag {
public String getDisplay() { public String getDisplay() {
return display; return display;
} }
public String toString() {
return abbreviation + " - " + display;
}
} }

View File

@ -121,7 +121,8 @@ public interface CollectionParams {
MOCK_REPLICA_TASK(false, LockLevel.REPLICA), MOCK_REPLICA_TASK(false, LockLevel.REPLICA),
NONE(false, LockLevel.NONE), NONE(false, LockLevel.NONE),
// TODO: not implemented yet // TODO: not implemented yet
MERGESHARDS(true, LockLevel.SHARD) MERGESHARDS(true, LockLevel.SHARD),
COLSTATUS(true, LockLevel.NONE)
; ;
public final boolean isWrite; public final boolean isWrite;