What CDAP platform provides:
API used by apps for logging the debug data
...
language | java |
---|
...
This document explains the separation of responsibilities between CDAP platform and Applications.
CDAP Platform
- Java API: CDAP platform provides two sets of java API. External api is used by CDAP applications to interact with the preview system and internal api is used by preview REST handler.
API to be used by applications:
Get the instance of DebugLogger from program context. For example MapReduceContext will be updated to add new method as -
Code Block language java /** * MapReduce job execution context. */ public interface
MapReduceContext
... { /** *
Return
the
DebugLogger
*
@param
loggerName
the
name
of
the logger using which
the debug information
will be logged
*/
DebugLogger
getLogger(String loggerName); }
Use DebugLogger to log the useful information.
Code Block language java /** * Interface used by the CDAP applications to log the debug data. */ public interface DebugLogger { /** * Logs the data at INFO level. Multiple values can be logged against the same property. * @param propertyName the the name of the property * @param propertyValue the value associated with the property */ void info(String propertyName, Object propertyValue); /** * Return the name of the logger instance. */ String getName(); /** * Returns {@code true} if application is running in debug mode otherwise false is returned. */ boolean isEnabled(); } /** * DebugLoggerFactory will be injected in the Program context classes. This may not be directly used by Applications. */ public interface DebugLoggerFactory { /** * Get the {@link DebugLogger} used to log the debug data. * @param loggerName the name of the logger with which the log data to be associated * @return the instance of the DebugLogger */ DebugLogger getLogger(String loggerName); }
API to be used by REST handler: PreviewHttpHandler will be responsible for handling the REST calls (details below). This REST handler will also interact with the preview system through API exposed by PreviewManager. Note that this is internal API.
Code Block language java /** * Interface used to start preview and also retrieve the information associated with a preview. */ public interface PreviewManager { /** *
Start the
program
in
preview mode. *
@param namespaceId
- REST endpoints
To start a preview
Code Block language java POST /v3/namespaces/{namespace-id}/preview where namespace-id is the name of the namespace Response will contain the CDAP generated unique preview-id which can be used further to get the preview data.
To get the status of the preview
Code Block language java GET /v3/namespaces/{namespace-id}/previews/{preview-id}/status where namespace-id is the name of the namespace preview-id is the id of the preview for which status is to be requested
To get the data associated with the preview
Code Block language java GET /v3/namespaces/{namespace-id}/previews/{preview-id}/loggers/{logger-id} where namespace-id is the name of the namespace preview-id is the id of the preview for which data is to be requested logger-id is the unique name used to identify the logger
Platform specific CDAP configurations:
Code Block language java Application configuration will have preview related configurations which will be used by CDAP. Currently there are programId and programType configurations which will be used to identify the program to be executed as a part of preview. { "preview": { "programId": "MyProgram", "programType": "workflow", "logLevel": "info" } }
API used by CDAP platform to interact with the preview system:
Code Block /** * Interface used to start preview and also retrieve the information associated with a preview. */ public interface PreviewManager { /** * Start the preview of an application config provided as an input. * @param namespaceId the id of * @param config the config for the preview * @return the unique {@link PreviewId} generated for the preview run * @throws Exception if there were any error during starting */ PreviewId start(NamespaceId namespaceId, String config) throws Exception; /** * Get the status for the specified {@link PreviewId}. * @param previewId the id of the preview for which status is to be returned * @return the status associated with the preview * @throws NotFoundException if the previewId is not found */ PreviewStatus getStatus(PreviewId previewId) throws NotFoundException; /** * Stop the preview identified by previewId. * @param previewId id of the preview * @throws Exceptionlanguage java the id of the namespace * @param request the request for the preview. This includes details about artifact, application configs, and preview configurations used by CDAP(details below) * @return the unique {@link PreviewId} generated for the preview run * @throws Exception if there were any error during starting */ ApplicationId start(NamespaceId namespaceId, AppRequest<?> request) throws Exception; /** * Get the status for the specified {@link ApplicationId}. * @param preview the id of the preview for which status is to be returned * @return the status associated with the preview * @throws NotFoundException if the preview is not found */ PreviewStatus getStatus(ApplicationId preview) throws NotFoundException; /** * Stop the preview identified by preview. * @param preview id of the preview * @throws Exception if the preview is not found or if there were any error during stop */ void stop(ApplicationId previewId) throws Exception; /** * Get the list of loggers in this preview. * @param preview id of the preview * @return the {@link List} of list of loggers for a given preview * @throws NotFoundException if the previewId is not found */ List<String> getLoggers(ApplicationId previewId) throws NotFoundException; /** * Get the data associated with the specified logger name of the preview. * @param preview id of the preview * @param loggerName the name of the logger for which data is to be returned * @return the {@link Map} of property name to property value associated with the given logger for a given preview * @throws NotFoundException if the preview is not found */ Map<String, List<String>> getData(ApplicationId preview, String loggerName) throws NotFoundException; /** * Get metric associated with the preview. * @param preview the id of the preview * @return the {@link Collection} of metrics emitted during the preview run * @throws NotFoundException if the previewId is not found
*/
Collection<MetricTimeSeries>
getMetrics(
ApplicationId
preview) throws
NotFoundException; /** * Get the
logs
for the preview. * @param
preview the id
of
the preview for which logs to be fetched * @return the
logs
*
@throws
NotFoundException
if the
preview is not found */
List<LogEntry> getLogs(ApplicationId preview) throws NotFoundException; }
//
Instance of the
PreviewStatus
is
returned
by
the
getStatus
call
above.
The
details
are
as
follows /** *
Represents
the
state of the preview.
*/
public
class
PreviewStatus
{
public
enum
Status
{
RUNNING,
COMPLETED,
DEPLOY_FAILED,
RUNTIME_FAILED
};
Status previewStatus;
@Nullable
String
failureMessage;
// Represents the
request with which
the
preview
was started. AppRequest request; }
- REST API exposed by platform:
Start a preview
Code Block language java POST /v3/namespaces/{namespace-id}/previews where namespace-id is the name of the namespace Response will contain the CDAP generated unique preview-id which can be used further to get the preview data.
Get the status of preview
Code Block language java GET /v3/namespaces/{namespace-id}/previews/{preview-id}/status where namespace-id is the name of the
namespace
preview-id is the id
of
the
preview for which
status is to be
requested
Stop preview
Code Block language java POST /v3/namespaces/{namespace-id}/previews/{preview-id}/stop where namespace-id is the name of the namespace preview-id is the id of the preview which is to be stopped
Get the list of loggers in the preview
Code Block language java GET /v3/namespaces/{namespace-id}/previews/{preview-id}/loggers where namespace-id is the name of the namespace preview-id is the id of the preview which is to
be stopped
Get the data associated with the preview
Code Block language java GET /v3/namespaces/{namespace-id}/previews/{preview-id}/loggers/{logger-id} where namespace-id is the name of the namespace preview-id is the id of the preview logger-id is the id of the
logger for which logs to be fetched
Application level capabilities:
- Config changes which will be understood by the application. For hydrator following is an example of the application level preview configurations -
Get the logs generated for the preview
Code Block language java GET /v3/namespaces/{namespace-id}/previews/{preview-id}/logs where namespace-id is the name of the namespace preview-id is the id of the preview
Get the metrics associated with the preview
Code Block language java
GET /v3/namespaces/{namespace-id}/previews/{preview-id}/metrics where namespace-id is the name of the namespace preview-id is the id of the preview
Preview specific configurations understood by CDAP: When preview is started, CDAP needs to know which program need to be executed. Following is a sample request json -
Code Block language java { "scopeartifact":"SYSTEM"{ }, "name":"MyPipelinecdap-data-pipeline", "configversion":{ "3.5.0-SNAPSHOT", "connectionsscope":["SYSTEM" { }, "fromname":"FTPMyPipeline", "config":{ "to":"CSVParser" ..... application specific configurations }, "preview": { { "programId": "MyProgram", "fromprogramType": "CSVParserworkflow", "to":"Table" } ], "stages":[ } }
In the above config json, CDAP will look for "preview" key to figure out which program to be executed by preview.
Application responsibilities:
Application can use the API exposed by CDAP for getting the logger and logging the data.
Application specific configurations can be specified in the config section of the json. For example following are the preview related configurations for hydrator app -
Code Block language java { "artifact":{ "name":"FTPcdap-data-pipeline", "version":"3.5.0-SNAPSHOT", "pluginscope":{"SYSTEM" }, "name":"FTPMyPipeline", "config":{ "typeconnections":"batchsource", { ... "label":"FTP", }, "artifact"stages": { ... "name":"core-plugins"}, "appPreviewConfig": { "versionstartStages":"1.4.0-SNAPSHOT", ["MyCSVParser"], // stages from which pipeline execution is to be started "scopeendStages": ["SYSTEMMyTable"], // stages till which pipeline need to be executed }, "useRealDatasets": ["FTP"], // list of datasets to be used from the real "properties":{user space for READ only purpose "outputs": { "referenceName":"myfile", "FTP": { "path":"/tmp/myfile" "data": [ } {"offset": 1, "body": "100,bob"}, }, "outputSchema":"{\"fields\offset":[{\"name\ 2, "body":\ "offset\"200,\"type\":\"long\"},{\"name\":\"body\",\"type\":\"string\"}]}"rob"}, {"offset": 3, "body": "300,tom"} }, ] { } "name":"MyCSVParser", } "plugin":{ } "name":"CSVParser", }, "type"preview":"transform", { ..... CDAP specific preview configurations "label":"CSVParser", "artifact":{ "name":"transform-plugins", "version":"1.4.0-SNAPSHOT", "scope":"SYSTEM" }, "properties":{ "format":"DEFAULT", "schema":"{\"type\":\"record\",\"name\":\"etlSchemaBody\",\"fields\":[{\"name\":\"id\",\"type\":\"int\"},{\"name\":\"name\",\"type\":\"string\"}]}", "field":"body" } }, "outputSchema":"{\"type\":\"record\",\"name\":\"etlSchemaBody\",\"fields\":[{\"name\":\"id\",\"type\":\"int\"},{\"name\":\"name\",\"type\":\"string\"}]}" }, { "name":"MyTable", "plugin":{ "name":"Table", "type":"batchsink", "label":"Table", "artifact":{ "name":"core-plugins", "version":"1.4.0-SNAPSHOT", "scope":"SYSTEM" }, "properties":{ "schema":"{\"type\":\"record\",\"name\":\"etlSchemaBody\",\"fields\":[{\"name\":\"id\",\"type\":\"int\"},{\"name\":\"name\",\"type\":\"string\"}]}", "name":"mytable", "schema.row.field":"id" } }, "outputSchema":"{\"type\":\"record\",\"name\":\"etlSchemaBody\",\"fields\":[{\"name\":\"id\",\"type\":\"int\"},{\"name\":\"name\",\"type\":\"string\"}]}", "inputSchema":[ { "name":"id", "type":"int", "nullable":false }, { "name":"name", "type":"string", "nullable":false } ] } ], "appPreviewConfig": { "startStages": ["MyCSVParser"], "endStages": ["MyTable"], "useSinks": ["MyTable"], "outputs": { "FTP": { "data": [ {"offset": 1, "body": "100,bob"}, {"offset": 2, "body": "200,rob"}, {"offset": 3, "body": "300,tom"} ], "schema": { "type" : "record", "fields": [ {"name":"offset","type":"long"}, {"name":"body","type":"string"} ] } } } } } } Note that "appPreviewConfig" section above is the application specific configurations which will be handled by the application.
Handling application level preview configurations: Preview configurations mentioned in the above section "appPreviewConfig" are application level and are required to handle by the application.
End to End flow:
Request to the preview endpoint is given by user with the appropriate configurations. Note that the configurations will include the configs understood by CDAP and the configs understood by the app.
- CDAP will generate unique preview id for this request which is returned to the user. User can then use this preview id further to query the data generated during the preview run.
- Hydrator app will be configured based on the application configurations. For example for single stage preview configuration, we can add Worker in the app which will run the transform.
- CDAP platform will determine which program in the application is require to execute based on the preview configurations provided for CDAP.
- Based on the log level specified in the configurations, CDAP will write the preview data to the dataset.
Distributed:
Open Questions:
...
} }
Handling application level preview configurations: Preview configurations mentioned in the above section "appPreviewConfig" are application level and are required to handle by the application. More details TBD.