/* -*- Mode: C; tab-width: 4; c-basic-offset: 4; indent-tabs-mode: nil -*- */ /* * Copyright 2018-2020 Couchbase, Inc. * * Licensed 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. */ /** CFLAGS="-I$(realpath ../../include) -I$(realpath ../../build/generated)" LDFLAGS="-L$(realpath ../../build/lib) -lcouchbase -Wl,-rpath=$(realpath ../../build/lib)" make analytics */ #include #include #include #include #include #include "cJSON.h" static void fail(const char *msg) { printf("[\x1b[31mERROR\x1b[0m] %s\n", msg); exit(EXIT_FAILURE); } static void check(lcb_STATUS err, const char *msg) { if (err != LCB_SUCCESS) { char buf[1024] = {0}; snprintf(buf, sizeof(buf), "%s: %s\n", msg, lcb_strerror_short(err)); fail(buf); } } static int err2color(lcb_STATUS err) { switch (err) { case LCB_SUCCESS: return 49; default: return 31; } } static void row_callback(lcb_INSTANCE *instance, int type, const lcb_RESPANALYTICS *resp) { int *idx; const char *row; size_t nrow; lcb_STATUS rc = lcb_respanalytics_status(resp); lcb_respanalytics_cookie(resp, (void **)&idx); lcb_respanalytics_row(resp, &row, &nrow); if (rc != LCB_SUCCESS) { const lcb_RESPHTTP *http; printf("\x1b[31m%s\x1b[0m", lcb_strerror_short(rc)); lcb_respanalytics_http_response(resp, &http); if (http) { uint16_t status; lcb_resphttp_http_status(http, &status); printf(", HTTP status: %d", (int)status); } printf("\n"); if (nrow) { cJSON *json; char *data = calloc(nrow + 1, sizeof(char)); memcpy(data, row, nrow); json = cJSON_Parse(data); if (json && json->type == cJSON_Object) { cJSON *errors = cJSON_GetObjectItem(json, "errors"); if (errors && errors->type == cJSON_Array) { int ii, nerrors = cJSON_GetArraySize(errors); for (ii = 0; ii < nerrors; ii++) { cJSON *err = cJSON_GetArrayItem(errors, ii); if (err && err->type == cJSON_Object) { cJSON *code, *msg; code = cJSON_GetObjectItem(err, "code"); msg = cJSON_GetObjectItem(err, "msg"); if (code && code->type == cJSON_Number && msg && msg->type == cJSON_String) { printf( "\x1b[1mcode\x1b[0m: \x1b[31m%d\x1b[0m, \x1b[1mmessage\x1b[0m: \x1b[31m%s\x1b[0m\n", code->valueint, msg->valuestring); } } } } } free(data); } } if (lcb_respanalytics_is_final(resp)) { printf("\x1b[1mMETA:\x1b[0m "); } else { printf("\x1b[1mR%d:\x1b[0m ", (*idx)++); } printf("%.*s\n", (int)nrow, (char *)row); if (lcb_respanalytics_is_final(resp)) { printf("\n"); } lcb_DEFERRED_HANDLE *handle; lcb_respanalytics_deferred_handle_extract(resp, &handle); if (handle) { const char *status; size_t status_len; lcb_deferred_handle_status(handle, &status, &status_len); printf("\x1b[1mDEFERRED:\x1b[0m %.*s\n", (int)status_len, status); lcb_deferred_handle_callback(handle, row_callback); check(lcb_deferred_handle_poll(instance, idx, handle), "poll deferred query status"); lcb_deferred_handle_destroy(handle); } } int main(int argc, char *argv[]) { lcb_STATUS err; lcb_INSTANCE *instance; char *bucket = NULL; size_t ii; if (argc < 2) { printf("Usage: %s couchbase://host/beer-sample [ password [ username ] ]\n", argv[0]); exit(EXIT_FAILURE); } { lcb_CREATEOPTS *create_options = NULL; lcb_createopts_create(&create_options, LCB_TYPE_BUCKET); lcb_createopts_connstr(create_options, argv[1], strlen(argv[1])); if (argc > 3) { lcb_createopts_credentials(create_options, argv[3], strlen(argv[3]), argv[2], strlen(argv[2])); } check(lcb_create(&instance, create_options), "create couchbase handle"); lcb_createopts_destroy(create_options); check(lcb_connect(instance), "schedule connection"); lcb_wait(instance, LCB_WAIT_DEFAULT); check(lcb_get_bootstrap_status(instance), "bootstrap from cluster"); check(lcb_cntl(instance, LCB_CNTL_GET, LCB_CNTL_BUCKETNAME, &bucket), "get bucket name"); if (strcmp(bucket, "beer-sample") != 0) { fail("expected bucket to be \"beer-sample\""); } } { const char *stmt = "SELECT * FROM breweries LIMIT 2"; lcb_CMDANALYTICS *cmd; int idx = 0; lcb_cmdanalytics_create(&cmd); lcb_cmdanalytics_callback(cmd, row_callback); lcb_cmdanalytics_statement(cmd, stmt, strlen(stmt)); lcb_cmdanalytics_deferred(cmd, 1); check(lcb_analytics(instance, &idx, cmd), "schedule analytics query"); printf("----> \x1b[36m%s\x1b[0m\n", stmt); lcb_cmdanalytics_destroy(cmd); lcb_wait(instance, LCB_WAIT_DEFAULT); } /* Now that we're all done, close down the connection handle */ lcb_destroy(instance); return 0; }