|  | --- | 
|  | id: functions-debug | 
|  | title: Debug Pulsar Functions | 
|  | sidebar_label: "How-to: Debug" | 
|  | original_id: functions-debug | 
|  | --- | 
|  |  | 
|  | You can use the following methods to debug Pulsar Functions: | 
|  |  | 
|  | * [Captured stderr](functions-debug.md#captured-stderr) | 
|  | * [Use unit test](functions-debug.md#use-unit-test) | 
|  | * [Debug with localrun mode](functions-debug.md#debug-with-localrun-mode) | 
|  | * [Use log topic](functions-debug.md#use-log-topic) | 
|  | * [Use Functions CLI](functions-debug.md#use-functions-cli) | 
|  |  | 
|  | ## Captured stderr | 
|  |  | 
|  | Function startup information and captured stderr output is written to `logs/functions/<tenant>/<namespace>/<function>/<function>-<instance>.log` | 
|  |  | 
|  | This is useful for debugging why a function fails to start. | 
|  |  | 
|  | ## Use unit test | 
|  |  | 
|  | A Pulsar Function is a function with inputs and outputs, you can test a Pulsar Function in a similar way as you test any function. | 
|  |  | 
|  | For example, if you have the following Pulsar Function: | 
|  |  | 
|  | ```java | 
|  |  | 
|  | import java.util.function.Function; | 
|  |  | 
|  | public class JavaNativeExclamationFunction implements Function<String, String> { | 
|  | @Override | 
|  | public String apply(String input) { | 
|  | return String.format("%s!", input); | 
|  | } | 
|  | } | 
|  |  | 
|  | ``` | 
|  |  | 
|  | You can write a simple unit test to test Pulsar Function. | 
|  |  | 
|  | :::tip | 
|  |  | 
|  | Pulsar uses testng for testing. | 
|  |  | 
|  | ::: | 
|  |  | 
|  | ```java | 
|  |  | 
|  | @Test | 
|  | public void testJavaNativeExclamationFunction() { | 
|  | JavaNativeExclamationFunction exclamation = new JavaNativeExclamationFunction(); | 
|  | String output = exclamation.apply("foo"); | 
|  | Assert.assertEquals(output, "foo!"); | 
|  | } | 
|  |  | 
|  | ``` | 
|  |  | 
|  | The following Pulsar Function implements the `org.apache.pulsar.functions.api.Function` interface. | 
|  |  | 
|  | ```java | 
|  |  | 
|  | import org.apache.pulsar.functions.api.Context; | 
|  | import org.apache.pulsar.functions.api.Function; | 
|  |  | 
|  | public class ExclamationFunction implements Function<String, String> { | 
|  | @Override | 
|  | public String process(String input, Context context) { | 
|  | return String.format("%s!", input); | 
|  | } | 
|  | } | 
|  |  | 
|  | ``` | 
|  |  | 
|  | In this situation, you can write a unit test for this function as well. Remember to mock the `Context` parameter. The following is an example. | 
|  |  | 
|  | :::tip | 
|  |  | 
|  | Pulsar uses testng for testing. | 
|  |  | 
|  | ::: | 
|  |  | 
|  | ```java | 
|  |  | 
|  | @Test | 
|  | public void testExclamationFunction() { | 
|  | ExclamationFunction exclamation = new ExclamationFunction(); | 
|  | String output = exclamation.process("foo", mock(Context.class)); | 
|  | Assert.assertEquals(output, "foo!"); | 
|  | } | 
|  |  | 
|  | ``` | 
|  |  | 
|  | ## Debug with localrun mode | 
|  | When you run a Pulsar Function in localrun mode, it launches an instance of the Function on your local machine as a thread. | 
|  |  | 
|  | In this mode, a Pulsar Function consumes and produces actual data to a Pulsar cluster, and mirrors how the function actually runs in a Pulsar cluster. | 
|  |  | 
|  | :::note | 
|  |  | 
|  | Currently, debugging with localrun mode is only supported by Pulsar Functions written in Java. You need Pulsar version 2.4.0 or later to do the following. Even though localrun is available in versions earlier than Pulsar 2.4.0, you cannot debug with localrun mode programmatically or run Functions as threads. | 
|  |  | 
|  | ::: | 
|  |  | 
|  | You can launch your function in the following manner. | 
|  |  | 
|  | ```java | 
|  |  | 
|  | FunctionConfig functionConfig = new FunctionConfig(); | 
|  | functionConfig.setName(functionName); | 
|  | functionConfig.setInputs(Collections.singleton(sourceTopic)); | 
|  | functionConfig.setClassName(ExclamationFunction.class.getName()); | 
|  | functionConfig.setRuntime(FunctionConfig.Runtime.JAVA); | 
|  | functionConfig.setOutput(sinkTopic); | 
|  |  | 
|  | LocalRunner localRunner = LocalRunner.builder().functionConfig(functionConfig).build(); | 
|  | localRunner.start(true); | 
|  |  | 
|  | ``` | 
|  |  | 
|  | So you can debug functions using an IDE easily. Set breakpoints and manually step through a function to debug with real data. | 
|  |  | 
|  | The following example illustrates how to programmatically launch a function in localrun mode. | 
|  |  | 
|  | ```java | 
|  |  | 
|  | public class ExclamationFunction implements Function<String, String> { | 
|  |  | 
|  | @Override | 
|  | public String process(String s, Context context) throws Exception { | 
|  | return s + "!"; | 
|  | } | 
|  |  | 
|  | public static void main(String[] args) throws Exception { | 
|  | FunctionConfig functionConfig = new FunctionConfig(); | 
|  | functionConfig.setName("exclamation"); | 
|  | functionConfig.setInputs(Collections.singleton("input")); | 
|  | functionConfig.setClassName(ExclamationFunction.class.getName()); | 
|  | functionConfig.setRuntime(FunctionConfig.Runtime.JAVA); | 
|  | functionConfig.setOutput("output"); | 
|  |  | 
|  | LocalRunner localRunner = LocalRunner.builder().functionConfig(functionConfig).build(); | 
|  | localRunner.start(false); | 
|  | } | 
|  |  | 
|  | ``` | 
|  |  | 
|  | To use localrun mode programmatically, add the following dependency. | 
|  |  | 
|  | ```xml | 
|  |  | 
|  | <dependency> | 
|  | <groupId>org.apache.pulsar</groupId> | 
|  | <artifactId>pulsar-functions-local-runner</artifactId> | 
|  | <version>${pulsar.version}</version> | 
|  | </dependency> | 
|  |  | 
|  | ``` | 
|  |  | 
|  | For complete code samples, see [here](https://github.com/jerrypeng/pulsar-functions-demos/tree/master/debugging). | 
|  |  | 
|  | :::note | 
|  |  | 
|  | Debugging with localrun mode for Pulsar Functions written in other languages will be supported soon. | 
|  |  | 
|  | ::: | 
|  |  | 
|  | ## Use log topic | 
|  |  | 
|  | In Pulsar Functions, you can generate log information defined in functions to a specified log topic. You can configure consumers to consume messages from a specified log topic to check the log information. | 
|  |  | 
|  |  | 
|  |  | 
|  | **Example** | 
|  |  | 
|  | ```java | 
|  |  | 
|  | import org.apache.pulsar.functions.api.Context; | 
|  | import org.apache.pulsar.functions.api.Function; | 
|  | import org.slf4j.Logger; | 
|  |  | 
|  | public class LoggingFunction implements Function<String, Void> { | 
|  | @Override | 
|  | public void apply(String input, Context context) { | 
|  | Logger LOG = context.getLogger(); | 
|  | String messageId = new String(context.getMessageId()); | 
|  |  | 
|  | if (input.contains("danger")) { | 
|  | LOG.warn("A warning was received in message {}", messageId); | 
|  | } else { | 
|  | LOG.info("Message {} received\nContent: {}", messageId, input); | 
|  | } | 
|  |  | 
|  | return null; | 
|  | } | 
|  | } | 
|  |  | 
|  | ``` | 
|  |  | 
|  | As shown in the example above, you can get the logger via `context.getLogger()` and assign the logger to the `LOG` variable of `slf4j`, so you can define your desired log information in a function using the `LOG` variable. Meanwhile, you need to specify the topic to which the log information is produced. | 
|  |  | 
|  | **Example** | 
|  |  | 
|  | ```bash | 
|  |  | 
|  | $ bin/pulsar-admin functions create \ | 
|  | --log-topic persistent://public/default/logging-function-logs \ | 
|  | # Other function configs | 
|  |  | 
|  | ``` | 
|  |  | 
|  | ## Use Functions CLI | 
|  |  | 
|  | With [Pulsar Functions CLI](reference-pulsar-admin.md#functions), you can debug Pulsar Functions with the following subcommands: | 
|  |  | 
|  | * `get` | 
|  | * `status` | 
|  | * `stats` | 
|  | * `list` | 
|  | * `trigger` | 
|  |  | 
|  | :::tip | 
|  |  | 
|  | For complete commands of **Pulsar Functions CLI**, see [here](reference-pulsar-admin.md#functions)。 | 
|  |  | 
|  | ::: | 
|  |  | 
|  | ### `get` | 
|  |  | 
|  | Get information about a Pulsar Function. | 
|  |  | 
|  | **Usage** | 
|  |  | 
|  | ```bash | 
|  |  | 
|  | $ pulsar-admin functions get options | 
|  |  | 
|  | ``` | 
|  |  | 
|  | **Options** | 
|  |  | 
|  | |Flag|Description | 
|  | |---|--- | 
|  | |`--fqfn`|The Fully Qualified Function Name (FQFN) of a Pulsar Function. | 
|  | |`--name`|The name of a Pulsar Function. | 
|  | |`--namespace`|The namespace of a Pulsar Function. | 
|  | |`--tenant`|The tenant of a Pulsar Function. | 
|  |  | 
|  | :::tip | 
|  |  | 
|  | `--fqfn` consists of `--name`, `--namespace` and `--tenant`, so you can specify either `--fqfn` or `--name`, `--namespace` and `--tenant`. | 
|  |  | 
|  | ::: | 
|  |  | 
|  | **Example** | 
|  |  | 
|  | You can specify `--fqfn` to get information about a Pulsar Function. | 
|  |  | 
|  | ```bash | 
|  |  | 
|  | $ ./bin/pulsar-admin functions get public/default/ExclamationFunctio6 | 
|  |  | 
|  | ``` | 
|  |  | 
|  | Optionally, you can specify `--name`, `--namespace` and `--tenant` to get information about a Pulsar Function. | 
|  |  | 
|  | ```bash | 
|  |  | 
|  | $ ./bin/pulsar-admin functions get \ | 
|  | --tenant public \ | 
|  | --namespace default \ | 
|  | --name ExclamationFunctio6 | 
|  |  | 
|  | ``` | 
|  |  | 
|  | As shown below, the `get` command shows input, output, runtime, and other information about the _ExclamationFunctio6_ function. | 
|  |  | 
|  | ```json | 
|  |  | 
|  | { | 
|  | "tenant": "public", | 
|  | "namespace": "default", | 
|  | "name": "ExclamationFunctio6", | 
|  | "className": "org.example.test.ExclamationFunction", | 
|  | "inputSpecs": { | 
|  | "persistent://public/default/my-topic-1": { | 
|  | "isRegexPattern": false | 
|  | } | 
|  | }, | 
|  | "output": "persistent://public/default/test-1", | 
|  | "processingGuarantees": "ATLEAST_ONCE", | 
|  | "retainOrdering": false, | 
|  | "userConfig": {}, | 
|  | "runtime": "JAVA", | 
|  | "autoAck": true, | 
|  | "parallelism": 1 | 
|  | } | 
|  |  | 
|  | ``` | 
|  |  | 
|  | ### `status` | 
|  |  | 
|  | Check the current status of a Pulsar Function. | 
|  |  | 
|  | **Usage** | 
|  |  | 
|  | ```bash | 
|  |  | 
|  | $ pulsar-admin functions status options | 
|  |  | 
|  | ``` | 
|  |  | 
|  | **Options** | 
|  |  | 
|  | |Flag|Description | 
|  | |---|--- | 
|  | |`--fqfn`|The Fully Qualified Function Name (FQFN) of a Pulsar Function. | 
|  | |`--instance-id`|The instance ID of a Pulsar Function <br />If the `--instance-id` is not specified, it gets the IDs of all instances.<br /> | 
|  | |`--name`|The name of a Pulsar Function. | 
|  | |`--namespace`|The namespace of a Pulsar Function. | 
|  | |`--tenant`|The tenant of a Pulsar Function. | 
|  |  | 
|  | **Example** | 
|  |  | 
|  | ```bash | 
|  |  | 
|  | $ ./bin/pulsar-admin functions status \ | 
|  | --tenant public \ | 
|  | --namespace default \ | 
|  | --name ExclamationFunctio6 \ | 
|  |  | 
|  | ``` | 
|  |  | 
|  | As shown below, the `status` command shows the number of instances, running instances, the instance running under the _ExclamationFunctio6_ function, received messages, successfully processed messages, system exceptions, the average latency and so on. | 
|  |  | 
|  | ```json | 
|  |  | 
|  | { | 
|  | "numInstances" : 1, | 
|  | "numRunning" : 1, | 
|  | "instances" : [ { | 
|  | "instanceId" : 0, | 
|  | "status" : { | 
|  | "running" : true, | 
|  | "error" : "", | 
|  | "numRestarts" : 0, | 
|  | "numReceived" : 1, | 
|  | "numSuccessfullyProcessed" : 1, | 
|  | "numUserExceptions" : 0, | 
|  | "latestUserExceptions" : [ ], | 
|  | "numSystemExceptions" : 0, | 
|  | "latestSystemExceptions" : [ ], | 
|  | "averageLatency" : 0.8385, | 
|  | "lastInvocationTime" : 1557734137987, | 
|  | "workerId" : "c-standalone-fw-23ccc88ef29b-8080" | 
|  | } | 
|  | } ] | 
|  | } | 
|  |  | 
|  | ``` | 
|  |  | 
|  | ### `stats` | 
|  |  | 
|  | Get the current stats of a Pulsar Function. | 
|  |  | 
|  | **Usage** | 
|  |  | 
|  | ```bash | 
|  |  | 
|  | $ pulsar-admin functions stats options | 
|  |  | 
|  | ``` | 
|  |  | 
|  | **Options** | 
|  |  | 
|  | |Flag|Description | 
|  | |---|--- | 
|  | |`--fqfn`|The Fully Qualified Function Name (FQFN) of a Pulsar Function. | 
|  | |`--instance-id`|The instance ID of a Pulsar Function. <br />If the `--instance-id` is not specified, it gets the IDs of all instances.<br /> | 
|  | |`--name`|The name of a Pulsar Function. | 
|  | |`--namespace`|The namespace of a Pulsar Function. | 
|  | |`--tenant`|The tenant of a Pulsar Function. | 
|  |  | 
|  | **Example** | 
|  |  | 
|  | ```bash | 
|  |  | 
|  | $ ./bin/pulsar-admin functions stats \ | 
|  | --tenant public \ | 
|  | --namespace default \ | 
|  | --name ExclamationFunctio6 \ | 
|  |  | 
|  | ``` | 
|  |  | 
|  | The output is shown as follows: | 
|  |  | 
|  | ```json | 
|  |  | 
|  | { | 
|  | "receivedTotal" : 1, | 
|  | "processedSuccessfullyTotal" : 1, | 
|  | "systemExceptionsTotal" : 0, | 
|  | "userExceptionsTotal" : 0, | 
|  | "avgProcessLatency" : 0.8385, | 
|  | "1min" : { | 
|  | "receivedTotal" : 0, | 
|  | "processedSuccessfullyTotal" : 0, | 
|  | "systemExceptionsTotal" : 0, | 
|  | "userExceptionsTotal" : 0, | 
|  | "avgProcessLatency" : null | 
|  | }, | 
|  | "lastInvocation" : 1557734137987, | 
|  | "instances" : [ { | 
|  | "instanceId" : 0, | 
|  | "metrics" : { | 
|  | "receivedTotal" : 1, | 
|  | "processedSuccessfullyTotal" : 1, | 
|  | "systemExceptionsTotal" : 0, | 
|  | "userExceptionsTotal" : 0, | 
|  | "avgProcessLatency" : 0.8385, | 
|  | "1min" : { | 
|  | "receivedTotal" : 0, | 
|  | "processedSuccessfullyTotal" : 0, | 
|  | "systemExceptionsTotal" : 0, | 
|  | "userExceptionsTotal" : 0, | 
|  | "avgProcessLatency" : null | 
|  | }, | 
|  | "lastInvocation" : 1557734137987, | 
|  | "userMetrics" : { } | 
|  | } | 
|  | } ] | 
|  | } | 
|  |  | 
|  | ``` | 
|  |  | 
|  | ### `list` | 
|  |  | 
|  | List all Pulsar Functions running under a specific tenant and namespace. | 
|  |  | 
|  | **Usage** | 
|  |  | 
|  | ```bash | 
|  |  | 
|  | $ pulsar-admin functions list options | 
|  |  | 
|  | ``` | 
|  |  | 
|  | **Options** | 
|  |  | 
|  | |Flag|Description | 
|  | |---|--- | 
|  | |`--namespace`|The namespace of a Pulsar Function. | 
|  | |`--tenant`|The tenant of a Pulsar Function. | 
|  |  | 
|  | **Example** | 
|  |  | 
|  | ```bash | 
|  |  | 
|  | $ ./bin/pulsar-admin functions list \ | 
|  | --tenant public \ | 
|  | --namespace default | 
|  |  | 
|  | ``` | 
|  |  | 
|  | As shown below, the `list` command returns three functions running under the _public_ tenant and the _default_ namespace. | 
|  |  | 
|  | ```text | 
|  |  | 
|  | ExclamationFunctio1 | 
|  | ExclamationFunctio2 | 
|  | ExclamationFunctio3 | 
|  |  | 
|  | ``` | 
|  |  | 
|  | ### `trigger` | 
|  |  | 
|  | Trigger a specified Pulsar Function with a supplied value. This command simulates the execution process of a Pulsar Function and verifies it. | 
|  |  | 
|  | **Usage** | 
|  |  | 
|  | ```bash | 
|  |  | 
|  | $ pulsar-admin functions trigger options | 
|  |  | 
|  | ``` | 
|  |  | 
|  | **Options** | 
|  |  | 
|  | |Flag|Description | 
|  | |---|--- | 
|  | |`--fqfn`|The Fully Qualified Function Name (FQFN) of a Pulsar Function. | 
|  | |`--name`|The name of a Pulsar Function. | 
|  | |`--namespace`|The namespace of a Pulsar Function. | 
|  | |`--tenant`|The tenant of a Pulsar Function. | 
|  | |`--topic`|The topic name that a Pulsar Function consumes from. | 
|  | |`--trigger-file`|The path to a file that contains the data to trigger a Pulsar Function. | 
|  | |`--trigger-value`|The value to trigger a Pulsar Function. | 
|  |  | 
|  | **Example** | 
|  |  | 
|  | ```bash | 
|  |  | 
|  | $ ./bin/pulsar-admin functions trigger \ | 
|  | --tenant public \ | 
|  | --namespace default \ | 
|  | --name ExclamationFunctio6 \ | 
|  | --topic persistent://public/default/my-topic-1 \ | 
|  | --trigger-value "hello pulsar functions" | 
|  |  | 
|  | ``` | 
|  |  | 
|  | As shown below, the `trigger` command returns the following result: | 
|  |  | 
|  | ```text | 
|  |  | 
|  | This is my function! | 
|  |  | 
|  | ``` | 
|  |  | 
|  | :::note | 
|  |  | 
|  | You must specify the entire topic name when using the `--topic` option. Otherwise, the following error occurs. | 
|  |  | 
|  | ```text | 
|  |  | 
|  | Function in trigger function has unidentified topic | 
|  | Reason: Function in trigger function has unidentified topic | 
|  |  | 
|  | ``` | 
|  |  | 
|  | ::: | 
|  |  |