From c47a2c3841c5b920b4d592a0861e81c647746c8a Mon Sep 17 00:00:00 2001 From: Nick Dimiduk Date: Thu, 28 Apr 2022 17:19:24 +0200 Subject: [PATCH] HBASE-26986 Trace a one-shot execution of a Master procedure This one adds a tracing context to the Master's ProcedureV2 execution. This ties all the spans that result from a procedure execution back to a procedure by name. Maybe in a follow-on issue, we can serialize span context along with procedures and display procedure start/stop/retry events and parent-child hierarchy in tracing visualization. Signed-off-by: Andrew Purtell Signed-off-by: Duo Zhang --- .../trace/hamcrest/AttributesMatchers.java | 5 + .../hbase/procedure2/ProcedureExecutor.java | 52 +++-- .../trace/ProcedureSpanBuilder.java | 52 +++++ .../procedure/TestProcedureTracing.java | 187 ++++++++++++++++++ .../hbase/trace/OpenTelemetryClassRule.java | 2 +- 5 files changed, 276 insertions(+), 22 deletions(-) create mode 100644 hbase-procedure/src/main/java/org/apache/hadoop/hbase/procedure2/trace/ProcedureSpanBuilder.java create mode 100644 hbase-server/src/test/java/org/apache/hadoop/hbase/master/procedure/TestProcedureTracing.java diff --git a/hbase-client/src/test/java/org/apache/hadoop/hbase/client/trace/hamcrest/AttributesMatchers.java b/hbase-client/src/test/java/org/apache/hadoop/hbase/client/trace/hamcrest/AttributesMatchers.java index d73abba6907..aec96897179 100644 --- a/hbase-client/src/test/java/org/apache/hadoop/hbase/client/trace/hamcrest/AttributesMatchers.java +++ b/hbase-client/src/test/java/org/apache/hadoop/hbase/client/trace/hamcrest/AttributesMatchers.java @@ -45,6 +45,11 @@ public final class AttributesMatchers { return containsEntry(equalTo(key), equalTo(value)); } + public static Matcher containsEntry(AttributeKey key, + Matcher matcher) { + return containsEntry(equalTo(key), matcher); + } + public static Matcher containsEntry(String key, String value) { return containsEntry(AttributeKey.stringKey(key), value); } diff --git a/hbase-procedure/src/main/java/org/apache/hadoop/hbase/procedure2/ProcedureExecutor.java b/hbase-procedure/src/main/java/org/apache/hadoop/hbase/procedure2/ProcedureExecutor.java index 024c3341adf..530e5305ba2 100644 --- a/hbase-procedure/src/main/java/org/apache/hadoop/hbase/procedure2/ProcedureExecutor.java +++ b/hbase-procedure/src/main/java/org/apache/hadoop/hbase/procedure2/ProcedureExecutor.java @@ -46,8 +46,10 @@ import org.apache.hadoop.hbase.procedure2.Procedure.LockState; import org.apache.hadoop.hbase.procedure2.store.ProcedureStore; import org.apache.hadoop.hbase.procedure2.store.ProcedureStore.ProcedureIterator; import org.apache.hadoop.hbase.procedure2.store.ProcedureStore.ProcedureStoreListener; +import org.apache.hadoop.hbase.procedure2.trace.ProcedureSpanBuilder; import org.apache.hadoop.hbase.procedure2.util.StringUtils; import org.apache.hadoop.hbase.security.User; +import org.apache.hadoop.hbase.trace.TraceUtil; import org.apache.hadoop.hbase.util.EnvironmentEdgeManager; import org.apache.hadoop.hbase.util.IdLock; import org.apache.hadoop.hbase.util.NonceKey; @@ -1947,6 +1949,34 @@ public class ProcedureExecutor { scheduler.signalAll(); } + /** + * Encapsulates execution of the current {@link #activeProcedure} for easy tracing. + */ + private long runProcedure() throws IOException { + final Procedure proc = this.activeProcedure; + int activeCount = activeExecutorCount.incrementAndGet(); + int runningCount = store.setRunningProcedureCount(activeCount); + LOG.trace("Execute pid={} runningCount={}, activeCount={}", proc.getProcId(), runningCount, + activeCount); + executionStartTime.set(EnvironmentEdgeManager.currentTime()); + IdLock.Entry lockEntry = procExecutionLock.getLockEntry(proc.getProcId()); + try { + executeProcedure(proc); + } catch (AssertionError e) { + LOG.info("ASSERT pid=" + proc.getProcId(), e); + throw e; + } finally { + procExecutionLock.releaseLockEntry(lockEntry); + activeCount = activeExecutorCount.decrementAndGet(); + runningCount = store.setRunningProcedureCount(activeCount); + LOG.trace("Halt pid={} runningCount={}, activeCount={}", proc.getProcId(), runningCount, + activeCount); + this.activeProcedure = null; + executionStartTime.set(Long.MAX_VALUE); + } + return EnvironmentEdgeManager.currentTime(); + } + @Override public void run() { long lastUpdate = EnvironmentEdgeManager.currentTime(); @@ -1958,27 +1988,7 @@ public class ProcedureExecutor { continue; } this.activeProcedure = proc; - int activeCount = activeExecutorCount.incrementAndGet(); - int runningCount = store.setRunningProcedureCount(activeCount); - LOG.trace("Execute pid={} runningCount={}, activeCount={}", proc.getProcId(), - runningCount, activeCount); - executionStartTime.set(EnvironmentEdgeManager.currentTime()); - IdLock.Entry lockEntry = procExecutionLock.getLockEntry(proc.getProcId()); - try { - executeProcedure(proc); - } catch (AssertionError e) { - LOG.info("ASSERT pid=" + proc.getProcId(), e); - throw e; - } finally { - procExecutionLock.releaseLockEntry(lockEntry); - activeCount = activeExecutorCount.decrementAndGet(); - runningCount = store.setRunningProcedureCount(activeCount); - LOG.trace("Halt pid={} runningCount={}, activeCount={}", proc.getProcId(), runningCount, - activeCount); - this.activeProcedure = null; - lastUpdate = EnvironmentEdgeManager.currentTime(); - executionStartTime.set(Long.MAX_VALUE); - } + lastUpdate = TraceUtil.trace(this::runProcedure, new ProcedureSpanBuilder(proc)); } } catch (Throwable t) { LOG.warn("Worker terminating UNNATURALLY {}", this.activeProcedure, t); diff --git a/hbase-procedure/src/main/java/org/apache/hadoop/hbase/procedure2/trace/ProcedureSpanBuilder.java b/hbase-procedure/src/main/java/org/apache/hadoop/hbase/procedure2/trace/ProcedureSpanBuilder.java new file mode 100644 index 00000000000..90d9326548b --- /dev/null +++ b/hbase-procedure/src/main/java/org/apache/hadoop/hbase/procedure2/trace/ProcedureSpanBuilder.java @@ -0,0 +1,52 @@ +/* + * 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.hadoop.hbase.procedure2.trace; + +import io.opentelemetry.api.trace.Span; +import io.opentelemetry.api.trace.SpanKind; +import java.util.function.Supplier; +import org.apache.hadoop.hbase.procedure2.Procedure; +import org.apache.hadoop.hbase.trace.TraceUtil; +import org.apache.yetus.audience.InterfaceAudience; + +/** + * Construct a {@link Span} instance for a {@link Procedure} execution. + */ +@InterfaceAudience.Private +public class ProcedureSpanBuilder implements Supplier { + + private final String name; + private final long procId; + private final long parentProcId; + + public ProcedureSpanBuilder(final Procedure proc) { + name = proc.getProcName(); + procId = proc.getProcId(); + parentProcId = proc.getParentProcId(); + } + + @Override + public Span get() { + return build(); + } + + public Span build() { + return TraceUtil.getGlobalTracer().spanBuilder(name).setSpanKind(SpanKind.INTERNAL) + .setAttribute("procId", procId).setAttribute("parentProcId", parentProcId).startSpan(); + } +} diff --git a/hbase-server/src/test/java/org/apache/hadoop/hbase/master/procedure/TestProcedureTracing.java b/hbase-server/src/test/java/org/apache/hadoop/hbase/master/procedure/TestProcedureTracing.java new file mode 100644 index 00000000000..7fe09aad0d9 --- /dev/null +++ b/hbase-server/src/test/java/org/apache/hadoop/hbase/master/procedure/TestProcedureTracing.java @@ -0,0 +1,187 @@ +/* + * 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.hadoop.hbase.master.procedure; + +import static io.opentelemetry.api.common.AttributeKey.longKey; +import static org.apache.hadoop.hbase.client.trace.hamcrest.AttributesMatchers.containsEntry; +import static org.apache.hadoop.hbase.client.trace.hamcrest.SpanDataMatchers.hasAttributes; +import static org.apache.hadoop.hbase.client.trace.hamcrest.SpanDataMatchers.hasEnded; +import static org.apache.hadoop.hbase.client.trace.hamcrest.SpanDataMatchers.hasName; +import static org.apache.hadoop.hbase.client.trace.hamcrest.SpanDataMatchers.hasStatusWithCode; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.allOf; +import static org.hamcrest.Matchers.any; +import static org.hamcrest.Matchers.anyOf; +import static org.hamcrest.Matchers.containsString; +import static org.hamcrest.Matchers.hasItem; + +import io.opentelemetry.api.trace.StatusCode; +import io.opentelemetry.sdk.trace.data.SpanData; +import java.util.List; +import java.util.concurrent.TimeUnit; +import java.util.concurrent.atomic.AtomicReference; +import java.util.stream.Collectors; +import org.apache.hadoop.hbase.ConnectionRule; +import org.apache.hadoop.hbase.HBaseClassTestRule; +import org.apache.hadoop.hbase.HRegionLocation; +import org.apache.hadoop.hbase.MatcherPredicate; +import org.apache.hadoop.hbase.MiniClusterRule; +import org.apache.hadoop.hbase.StartMiniClusterOption; +import org.apache.hadoop.hbase.TableName; +import org.apache.hadoop.hbase.Waiter; +import org.apache.hadoop.hbase.client.AsyncAdmin; +import org.apache.hadoop.hbase.client.AsyncConnection; +import org.apache.hadoop.hbase.client.RegionInfo; +import org.apache.hadoop.hbase.client.Table; +import org.apache.hadoop.hbase.client.trace.StringTraceRenderer; +import org.apache.hadoop.hbase.testclassification.MasterTests; +import org.apache.hadoop.hbase.testclassification.MediumTests; +import org.apache.hadoop.hbase.trace.OpenTelemetryClassRule; +import org.apache.hadoop.hbase.trace.OpenTelemetryTestRule; +import org.apache.hadoop.hbase.trace.TraceUtil; +import org.apache.hadoop.hbase.util.Bytes; +import org.hamcrest.Matcher; +import org.hamcrest.Matchers; +import org.junit.ClassRule; +import org.junit.Rule; +import org.junit.Test; +import org.junit.experimental.categories.Category; +import org.junit.rules.RuleChain; +import org.junit.rules.TestName; +import org.junit.rules.TestRule; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * Test of master ProcedureV2 tracing. + */ +@Category({ MasterTests.class, MediumTests.class }) +public class TestProcedureTracing { + private static final Logger LOG = LoggerFactory.getLogger(TestProcedureTracing.class); + + @ClassRule + public static final HBaseClassTestRule CLASS_RULE = + HBaseClassTestRule.forClass(TestProcedureTracing.class); + + private static final OpenTelemetryClassRule otelClassRule = OpenTelemetryClassRule.create(); + private static final MiniClusterRule miniClusterRule = MiniClusterRule.newBuilder() + .setMiniClusterOption(StartMiniClusterOption.builder().numWorkers(3).build()).build(); + protected static final ConnectionRule connectionRule = + ConnectionRule.createAsyncConnectionRule(miniClusterRule::createAsyncConnection); + + @ClassRule + public static final TestRule classRule = + RuleChain.outerRule(otelClassRule).around(miniClusterRule).around(connectionRule); + + @Rule + public final OpenTelemetryTestRule otelTestRule = new OpenTelemetryTestRule(otelClassRule); + + @Rule + public TestName name = new TestName(); + + @Test + public void testCreateOpenDeleteTableSpans() throws Exception { + final TableName tableName = TableName.valueOf(name.getMethodName()); + final AsyncConnection conn = connectionRule.getAsyncConnection(); + final AsyncAdmin admin = conn.getAdmin(); + + final AtomicReference> tableRegionsRef = new AtomicReference<>(); + TraceUtil.trace(() -> { + try (final Table ignored = miniClusterRule.getTestingUtility() + .createMultiRegionTable(tableName, Bytes.toBytes("fam"), 5)) { + final List tableRegions = conn.getRegionLocator(tableName).getAllRegionLocations() + .get().stream().map(HRegionLocation::getRegion).map(RegionInfo::getEncodedName) + .collect(Collectors.toList()); + tableRegionsRef.set(tableRegions); + } + if (admin.tableExists(tableName).get()) { + if (admin.isTableEnabled(tableName).get()) { + admin.disableTable(tableName).get(); + } + admin.deleteTable(tableName).get(); + } + }, name.getMethodName()); + + final Matcher testSpanMatcher = allOf(hasName(name.getMethodName()), hasEnded()); + Waiter.waitFor(conn.getConfiguration(), TimeUnit.MINUTES.toMillis(3), + new MatcherPredicate<>(otelClassRule::getSpans, hasItem(testSpanMatcher))); + final List spans = otelClassRule.getSpans(); + final StringTraceRenderer renderer = new StringTraceRenderer(spans); + renderer.render(LOG::debug); + + // Expect to find a span for a CreateTableProcedure for the test table + final Matcher createTableProcedureSpanMatcher = allOf( + hasName(allOf(containsString("CreateTableProcedure"), + containsString("table=" + name.getMethodName()))), + hasEnded(), hasStatusWithCode(StatusCode.OK), + hasAttributes(allOf(containsEntry(longKey("procId"), any(Long.class)), + containsEntry(longKey("parentProcId"), any(Long.class))))); + assertThat("Expected to find a span for a CreateTableProcedure for the test table", spans, + hasItem(createTableProcedureSpanMatcher)); + + // Expect to find a span for a TransitRegionStateProcedure for the test table + final Matcher transitRegionStateProcedureSpanMatcher = allOf( + hasName(allOf(containsString("TransitRegionStateProcedure"), + containsString("table=" + name.getMethodName()))), + hasEnded(), hasStatusWithCode(StatusCode.OK), + hasAttributes(allOf(containsEntry(longKey("procId"), any(Long.class)), + containsEntry(longKey("parentProcId"), any(Long.class))))); + assertThat("Expected to find a span for a TransitRegionStateProcedure for the test table", + spans, hasItem(transitRegionStateProcedureSpanMatcher)); + + // Expect to find a span for an OpenRegionProcedure for a region of the test table + final List> tableRegionMatchers = + tableRegionsRef.get().stream().map(Matchers::containsString).collect(Collectors.toList()); + final Matcher openRegionProcedureSpanMatcher = + allOf(hasName(allOf(containsString("OpenRegionProcedure"), anyOf(tableRegionMatchers))), + hasEnded(), hasStatusWithCode(StatusCode.OK), + hasAttributes(allOf(containsEntry(longKey("procId"), any(Long.class)), + containsEntry(longKey("parentProcId"), any(Long.class))))); + assertThat("Expected to find a span for an OpenRegionProcedure for a region of the test table", + spans, hasItem(openRegionProcedureSpanMatcher)); + + // Expect to find a span for a CloseRegionProcedure for a region of the test table + final Matcher closeRegionProcedureSpanMatcher = + allOf(hasName(allOf(containsString("CloseRegionProcedure"), anyOf(tableRegionMatchers))), + hasEnded(), hasStatusWithCode(StatusCode.OK), + hasAttributes(allOf(containsEntry(longKey("procId"), any(Long.class)), + containsEntry(longKey("parentProcId"), any(Long.class))))); + assertThat("Expected to find a span for a CloseRegionProcedure for a region of the test table", + spans, hasItem(closeRegionProcedureSpanMatcher)); + + // Expect to find a span for a DisableTableProcedure for the test table + final Matcher disableTableProcedureSpanMatcher = allOf( + hasName(allOf(containsString("DisableTableProcedure"), + containsString("table=" + name.getMethodName()))), + hasEnded(), hasStatusWithCode(StatusCode.OK), + hasAttributes(allOf(containsEntry(longKey("procId"), any(Long.class)), + containsEntry(longKey("parentProcId"), any(Long.class))))); + assertThat("Expected to find a span for a DisableTableProcedure for the test table", spans, + hasItem(disableTableProcedureSpanMatcher)); + + // Expect to find a span for a DeleteTableProcedure for the test table + final Matcher deleteTableProcedureSpanMatcher = allOf( + hasName(allOf(containsString("DeleteTableProcedure"), + containsString("table=" + name.getMethodName()))), + hasEnded(), hasStatusWithCode(StatusCode.OK), + hasAttributes(allOf(containsEntry(longKey("procId"), any(Long.class)), + containsEntry(longKey("parentProcId"), any(Long.class))))); + assertThat("Expected to find a span for a DeleteTableProcedure for the test table", spans, + hasItem(deleteTableProcedureSpanMatcher)); + } +} diff --git a/hbase-server/src/test/java/org/apache/hadoop/hbase/trace/OpenTelemetryClassRule.java b/hbase-server/src/test/java/org/apache/hadoop/hbase/trace/OpenTelemetryClassRule.java index 0f7c2db4f0e..7b5455f2011 100644 --- a/hbase-server/src/test/java/org/apache/hadoop/hbase/trace/OpenTelemetryClassRule.java +++ b/hbase-server/src/test/java/org/apache/hadoop/hbase/trace/OpenTelemetryClassRule.java @@ -49,7 +49,7 @@ import org.junit.rules.ExternalResource; * private static final OpenTelemetryClassRule otelClassRule = OpenTelemetryClassRule.create(); * private static final MiniClusterRule miniClusterRule = MiniClusterRule.newBuilder().build(); * protected static final ConnectionRule connectionRule = - * new ConnectionRule(miniClusterRule::createConnection); + * ConnectionRule.createAsyncConnectionRule(miniClusterRule::createAsyncConnection); * * @ClassRule * public static final TestRule classRule =