/* Copyright (c) 2018 The Johns Hopkins University/Applied Physics Laboratory * All Rights Reserved. * * This file is dual licensed under the terms of the Apache 2.0 License and * the BSD 3-Clause License. See the LICENSE file in the root of this * repository for more information. */ #include #include #include #include #include #if OPENSSL_VERSION_NUMBER < 0x10100003L #define TLS_client_method SSLv23_client_method #define OPENSSL_init_ssl(a,b) SSL_library_init() #endif #include "kmip.h" #include "kmip_bio.h" #include "kmip_memset.h" void print_help(const char *app) { printf("Usage: %s [flag value | flag] ...\n\n", app); printf("Flags:\n"); printf("-a addr : the IP address of the KMIP server\n"); printf("-c path : path to client certificate file\n"); printf("-h : print this help info\n"); printf("-i id : the ID of the symmetric key to retrieve\n"); printf("-k path : path to client key file\n"); printf("-p port : the port number of the KMIP server\n"); printf("-r path : path to CA certificate file\n"); printf("-s pass : the password for KMIP server authentication\n"); printf("-u user : the username for KMIP server authentication\n"); } int parse_arguments(int argc, char **argv, char **server_address, char **server_port, char **client_certificate, char **client_key, char **ca_certificate, char **username, char **password, char **id, int *print_usage) { if(argc <= 1) { print_help(argv[0]); return(-1); } for(int i = 1; i < argc; i++) { if(strncmp(argv[i], "-a", 2) == 0) { *server_address = argv[++i]; } else if(strncmp(argv[i], "-c", 2) == 0) { *client_certificate = argv[++i]; } else if(strncmp(argv[i], "-h", 2) == 0) { *print_usage = 1; } else if(strncmp(argv[i], "-i", 2) == 0) { *id = argv[++i]; } else if(strncmp(argv[i], "-k", 2) == 0) { *client_key = argv[++i]; } else if(strncmp(argv[i], "-p", 2) == 0) { *server_port = argv[++i]; } else if(strncmp(argv[i], "-r", 2) == 0) { *ca_certificate = argv[++i]; } else if(strncmp(argv[i], "-s", 2) == 0) { *password = argv[++i]; } else if(strncmp(argv[i], "-u", 2) == 0) { *username = argv[++i]; } else { printf("Invalid option: '%s'\n", argv[i]); print_help(argv[0]); return(-1); } } return(0); } void * demo_calloc(void *state, size_t num, size_t size) { printf("demo_calloc called: state = %p, num = %zu, size = %zu\n", state, num, size); return(calloc(num, size)); } void * demo_realloc(void *state, void *ptr, size_t size) { printf("demo_realloc called: state = %p, ptr = %p, size = %zu\n", state, ptr, size); return(realloc(ptr, size)); } void demo_free(void *state, void *ptr) { printf("demo_free called: state = %p, ptr = %p\n", state, ptr); free(ptr); return; } int use_mid_level_api(char *server_address, char *server_port, char *client_certificate, char *client_key, char *ca_certificate, char *username, char *password, char *id) { /* Set up the TLS connection to the KMIP server. */ SSL_CTX *ctx = NULL; SSL *ssl = NULL; OPENSSL_init_ssl(0, NULL); ctx = SSL_CTX_new(TLS_client_method()); printf("\n"); printf("Loading the client certificate: %s\n", client_certificate); if(SSL_CTX_use_certificate_file(ctx, client_certificate, SSL_FILETYPE_PEM) != 1) { fprintf(stderr, "Loading the client certificate failed\n"); ERR_print_errors_fp(stderr); SSL_CTX_free(ctx); return(-1); } printf("Loading the client key: %s\n", client_key); if(SSL_CTX_use_PrivateKey_file(ctx, client_key, SSL_FILETYPE_PEM) != 1) { fprintf(stderr, "Loading the client key failed\n"); ERR_print_errors_fp(stderr); SSL_CTX_free(ctx); return(-1); } printf("Loading the CA certificate: %s\n", ca_certificate); if(SSL_CTX_load_verify_locations(ctx, ca_certificate, NULL) != 1) { fprintf(stderr, "Loading the CA file failed\n"); ERR_print_errors_fp(stderr); SSL_CTX_free(ctx); return(-1); } BIO *bio = NULL; bio = BIO_new_ssl_connect(ctx); if(bio == NULL) { printf("BIO_new_ssl_connect failed\n"); SSL_CTX_free(ctx); return(-1); } BIO_get_ssl(bio, &ssl); SSL_set_mode(ssl, SSL_MODE_AUTO_RETRY); BIO_set_conn_hostname(bio, server_address); BIO_set_conn_port(bio, server_port); if(BIO_do_connect(bio) != 1) { fprintf(stderr, "BIO_do_connect failed\n"); ERR_print_errors_fp(stderr); BIO_free_all(bio); SSL_CTX_free(ctx); return(-1); } printf("\n"); char *key = NULL; int key_size = 0; size_t id_size = kmip_strnlen_s(id, 50); /* Set up the KMIP context and send the request message. */ KMIP kmip_context = {0}; kmip_context.calloc_func = &demo_calloc; kmip_context.realloc_func = &demo_realloc; kmip_context.free_func = &demo_free; kmip_init(&kmip_context, NULL, 0, KMIP_1_0); TextString u = {0}; u.value = username; u.size = kmip_strnlen_s(username, 50); TextString p = {0}; p.value = password; p.size = kmip_strnlen_s(password, 50); UsernamePasswordCredential upc = {0}; upc.username = &u; upc.password = &p; Credential credential = {0}; credential.credential_type = KMIP_CRED_USERNAME_AND_PASSWORD; credential.credential_value = &upc; int result = kmip_add_credential(&kmip_context, &credential); if(result != KMIP_OK) { printf("Failed to add credential to the KMIP context.\n"); BIO_free_all(bio); SSL_CTX_free(ctx); kmip_destroy(&kmip_context); return(result); } result = kmip_bio_get_symmetric_key_with_context(&kmip_context, bio, id, id_size, &key, &key_size); BIO_free_all(bio); SSL_CTX_free(ctx); /* Handle the response results. */ printf("\n"); if(result < 0) { printf("An error occurred while creating the symmetric key."); printf("Error Code: %d\n", result); printf("Error Name: "); kmip_print_error_string(result); printf("\n"); printf("Context Error: %s\n", kmip_context.error_message); printf("Stack trace:\n"); kmip_print_stack_trace(&kmip_context); } else if(result >= 0) { printf("The KMIP operation was executed with no errors.\n"); printf("Result: "); kmip_print_result_status_enum(result); printf(" (%d)\n", result); if(result == KMIP_STATUS_SUCCESS) { printf("Symmetric Key ID: %s\n", id); printf("Symmetric Key Size: %d bits\n", key_size * 8); printf("Symmetric Key:"); kmip_print_buffer(key, key_size); printf("\n"); } } printf("\n"); if(key != NULL) { kmip_memset(key, 0, key_size); kmip_free(NULL, key); } /* Clean up the KMIP context and return the results. */ kmip_destroy(&kmip_context); return(result); } int main(int argc, char **argv) { char *server_address = NULL; char *server_port = NULL; char *client_certificate = NULL; char *client_key = NULL; char *ca_certificate = NULL; char *username = NULL; char *password = NULL; char *id = NULL; int help = 0; int error = parse_arguments(argc, argv, &server_address, &server_port, &client_certificate, &client_key, &ca_certificate, &username, &password, &id, &help); if(error) { return(error); } if(help) { print_help(argv[0]); return(0); } int result = use_mid_level_api(server_address, server_port, client_certificate, client_key, ca_certificate, username, password, id); return(result); }