blob: 6b9f44461f97f7cd3ddc5f4702c4ab6b7b0e3965 [file] [log] [blame]
// 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.doris.common.proc;
import org.apache.doris.alter.DecommissionBackendJob.DecommissionType;
import org.apache.doris.catalog.Catalog;
import org.apache.doris.cluster.Cluster;
import org.apache.doris.common.AnalysisException;
import org.apache.doris.common.Pair;
import org.apache.doris.common.util.DebugUtil;
import org.apache.doris.common.util.ListComparator;
import org.apache.doris.common.util.TimeUtils;
import org.apache.doris.service.FrontendOptions;
import org.apache.doris.system.Backend;
import org.apache.doris.system.SystemInfoService;
import com.google.common.base.Preconditions;
import com.google.common.base.Stopwatch;
import com.google.common.base.Strings;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Lists;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import java.util.ArrayList;
import java.util.Collections;
import java.util.LinkedList;
import java.util.List;
import java.util.concurrent.TimeUnit;
public class BackendsProcDir implements ProcDirInterface {
private static final Logger LOG = LogManager.getLogger(BackendsProcDir.class);
public static final ImmutableList<String> TITLE_NAMES = new ImmutableList.Builder<String>()
.add("BackendId").add("Cluster").add("IP").add("HostName").add("HeartbeatPort")
.add("BePort").add("HttpPort").add("BrpcPort").add("LastStartTime").add("LastHeartbeat").add("Alive")
.add("SystemDecommissioned").add("ClusterDecommissioned").add("TabletNum")
.add("DataUsedCapacity").add("AvailCapacity").add("TotalCapacity").add("UsedPct")
.add("MaxDiskUsedPct").add("ErrMsg").add("Version")
.build();
public static final int IP_INDEX = 2;
public static final int HOSTNAME_INDEX = 3;
private SystemInfoService clusterInfoService;
public BackendsProcDir(SystemInfoService clusterInfoService) {
this.clusterInfoService = clusterInfoService;
}
@Override
public ProcResult fetchResult() throws AnalysisException {
Preconditions.checkNotNull(clusterInfoService);
BaseProcResult result = new BaseProcResult();
result.setNames(TITLE_NAMES);
final List<List<String>> backendInfos = getClusterBackendInfos(null);
for (List<String> backendInfo : backendInfos) {
List<String> oneInfo = new ArrayList<String>(backendInfo.size());
for (String info : backendInfo) {
oneInfo.add(info);
}
result.addRow(oneInfo);
}
return result;
}
/**
* get backends of cluster
* @param clusterName
* @return
*/
public static List<List<String>> getClusterBackendInfos(String clusterName) {
final SystemInfoService clusterInfoService = Catalog.getCurrentSystemInfo();
List<List<String>> backendInfos = new LinkedList<List<String>>();
List<Long> backendIds = null;
if (!Strings.isNullOrEmpty(clusterName)) {
final Cluster cluster = Catalog.getInstance().getCluster(clusterName);
// root not in any cluster
if (null == cluster) {
return backendInfos;
}
backendIds = cluster.getBackendIdList();
} else {
backendIds = clusterInfoService.getBackendIds(false);
if (backendIds == null) {
return backendInfos;
}
}
long start = System.currentTimeMillis();
Stopwatch watch = Stopwatch.createUnstarted();
List<List<Comparable>> comparableBackendInfos = new LinkedList<List<Comparable>>();
for (long backendId : backendIds) {
Backend backend = clusterInfoService.getBackend(backendId);
if (backend == null) {
continue;
}
watch.start();
Integer tabletNum = Catalog.getCurrentInvertedIndex().getTabletNumByBackendId(backendId);
watch.stop();
List<Comparable> backendInfo = Lists.newArrayList();
backendInfo.add(String.valueOf(backendId));
backendInfo.add(backend.getOwnerClusterName());
backendInfo.add(backend.getHost());
if (Strings.isNullOrEmpty(clusterName)) {
backendInfo.add(FrontendOptions.getHostnameByIp(backend.getHost()));
backendInfo.add(String.valueOf(backend.getHeartbeatPort()));
backendInfo.add(String.valueOf(backend.getBePort()));
backendInfo.add(String.valueOf(backend.getHttpPort()));
backendInfo.add(String.valueOf(backend.getBrpcPort()));
}
backendInfo.add(TimeUtils.longToTimeString(backend.getLastStartTime()));
backendInfo.add(TimeUtils.longToTimeString(backend.getLastUpdateMs()));
backendInfo.add(String.valueOf(backend.isAlive()));
if (backend.isDecommissioned() && backend.getDecommissionType() == DecommissionType.ClusterDecommission) {
backendInfo.add(String.valueOf("false"));
backendInfo.add(String.valueOf("true"));
} else if (backend.isDecommissioned()
&& backend.getDecommissionType() == DecommissionType.SystemDecommission) {
backendInfo.add(String.valueOf("true"));
backendInfo.add(String.valueOf("false"));
} else {
backendInfo.add(String.valueOf("false"));
backendInfo.add(String.valueOf("false"));
}
backendInfo.add(tabletNum.toString());
// capacity
// data used
long dataUsedB = backend.getDataUsedCapacityB();
Pair<Double, String> usedCapacity = DebugUtil.getByteUint(dataUsedB);
backendInfo.add(DebugUtil.DECIMAL_FORMAT_SCALE_3.format(usedCapacity.first) + " " + usedCapacity.second);
// available
long availB = backend.getAvailableCapacityB();
Pair<Double, String> availCapacity = DebugUtil.getByteUint(availB);
backendInfo.add(DebugUtil.DECIMAL_FORMAT_SCALE_3.format(availCapacity.first) + " " + availCapacity.second);
// total
long totalB = backend.getTotalCapacityB();
Pair<Double, String> totalCapacity = DebugUtil.getByteUint(totalB);
backendInfo.add(DebugUtil.DECIMAL_FORMAT_SCALE_3.format(totalCapacity.first) + " " + totalCapacity.second);
// used percent
double used = 0.0;
if (totalB <= 0) {
used = 0.0;
} else {
used = (double) (totalB - availB) * 100 / totalB;
}
backendInfo.add(String.format("%.2f", used) + " %");
backendInfo.add(String.format("%.2f", backend.getMaxDiskUsedPct() * 100) + " %");
backendInfo.add(backend.getHeartbeatErrMsg());
backendInfo.add(backend.getVersion());
comparableBackendInfos.add(backendInfo);
}
// backends proc node get result too slow, add log to observer.
LOG.info("backends proc get tablet num cost: {}, total cost: {}",
watch.elapsed(TimeUnit.MILLISECONDS), (System.currentTimeMillis() - start));
// sort by cluster name, host name
ListComparator<List<Comparable>> comparator = new ListComparator<List<Comparable>>(1, 3);
Collections.sort(comparableBackendInfos, comparator);
for (List<Comparable> backendInfo : comparableBackendInfos) {
List<String> oneInfo = new ArrayList<String>(backendInfo.size());
for (Comparable element : backendInfo) {
oneInfo.add(element.toString());
}
backendInfos.add(oneInfo);
}
return backendInfos;
}
@Override
public boolean register(String name, ProcNodeInterface node) {
return false;
}
@Override
public ProcNodeInterface lookup(String beIdStr) throws AnalysisException {
if (Strings.isNullOrEmpty(beIdStr)) {
throw new AnalysisException("Backend id is null");
}
long backendId = -1L;
try {
backendId = Long.valueOf(beIdStr);
} catch (NumberFormatException e) {
throw new AnalysisException("Invalid backend id format: " + beIdStr);
}
Backend backend = clusterInfoService.getBackend(backendId);
if (backend == null) {
throw new AnalysisException("Backend[" + backendId + "] does not exist.");
}
return new BackendProcNode(backend);
}
}