/* * Copyright (c) 2015-2018 Nitrokey UG * * This file is part of libnitrokey. * * libnitrokey is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * libnitrokey is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with libnitrokey. If not, see . * * SPDX-License-Identifier: LGPL-3.0 */ #include "NK_C_API.h" #ifndef NO_LOG #include #endif #include #include "libnitrokey/NitrokeyManager.h" #include #include "libnitrokey/LibraryException.h" #include "libnitrokey/cxx_semantics.h" #include "libnitrokey/stick20_commands.h" #include "libnitrokey/device_proto.h" #include "libnitrokey/version.h" #include "nk_strndup.h" using namespace nitrokey; uint8_t NK_last_command_status = 0; #include "NK_C_API_helpers.h" #include "NitrokeyManagerOTP.h" #include "NitrokeyManagerPWS.h" #ifdef __cplusplus extern "C" { #endif NK_C_API uint8_t NK_get_last_command_status() { auto _copy = NK_last_command_status; NK_last_command_status = 0; return _copy; } NK_C_API int NK_login(const char *device_model) { auto m = NitrokeyManager::instance(); try { NK_last_command_status = 0; return m->connect(device_model); } catch (CommandFailedException & commandFailedException) { NK_last_command_status = commandFailedException.last_command_status; return commandFailedException.last_command_status; } catch (const DeviceCommunicationException &deviceException){ NK_last_command_status = 256-deviceException.getType(); #ifndef NO_LOG cerr << deviceException.what() << endl; #endif return 0; } catch (std::runtime_error &e) { #ifndef NO_LOG cerr << e.what() << endl; #endif return 0; } return 0; } NK_C_API int NK_login_enum(NK_device_model device_model) { const char *model_string; switch (device_model) { case NK_PRO: model_string = "P"; break; case NK_STORAGE: model_string = "S"; break; case NK_LIBREM: model_string = "L"; break; case NK_DISCONNECTED: default: /* no such enum value -- return error code */ return 0; } return NK_login(model_string); } NK_C_API int NK_logout() { auto m = NitrokeyManager::instance(); return get_without_result([&]() { m->disconnect(); }); } NK_C_API int NK_first_authenticate(const char* admin_password, const char* admin_temporary_password) { auto m = NitrokeyManager::instance(); return get_without_result([&]() { return m->first_authenticate(admin_password, admin_temporary_password); }); } NK_C_API int NK_user_authenticate(const char* user_password, const char* user_temporary_password) { auto m = NitrokeyManager::instance(); return get_without_result([&]() { m->user_authenticate(user_password, user_temporary_password); }); } NK_C_API int NK_factory_reset(const char* admin_password) { auto m = NitrokeyManager::instance(); return get_without_result([&]() { m->factory_reset(admin_password); }); } NK_C_API int NK_build_aes_key(const char* admin_password) { auto m = NitrokeyManager::instance(); return get_without_result([&]() { m->build_aes_key(admin_password); }); } NK_C_API int NK_unlock_user_password(const char *admin_password, const char *new_user_password) { auto m = NitrokeyManager::instance(); return get_without_result([&]() { m->unlock_user_password(admin_password, new_user_password); }); } NK_C_API enum NK_device_model NK_get_device_model() { auto m = NitrokeyManager::instance(); try { auto model = m->get_connected_device_model(); switch (model) { case DeviceModel::PRO: return NK_PRO; case DeviceModel::STORAGE: return NK_STORAGE; case DeviceModel::LIBREM: return NK_LIBREM; default: /* unknown or not connected device */ return NK_device_model::NK_DISCONNECTED; } } catch (const DeviceNotConnected& e) { return NK_device_model::NK_DISCONNECTED; } } NK_C_API char * NK_status() { return NK_get_status_as_string(); } NK_C_API char * NK_get_status_as_string() { auto m = NitrokeyManager::instance(); return get_with_string_result([&]() { string && s = m->get_status_as_string(); char * rs = strndup(s.c_str(), MAXIMUM_STR_REPLY_LENGTH); clear_string(s); return rs; }); } NK_C_API int NK_get_status(struct NK_status* out) { if (out == nullptr) { return -1; } auto m = NitrokeyManager::instance(); auto result = get_with_status([&]() { return m->get_status(); }, proto::stick10::GetStatus::ResponsePayload()); auto error_code = std::get<0>(result); if (error_code != 0) { return error_code; } auto status = std::get<1>(result); out->firmware_version_major = status.firmware_version_st.major; out->firmware_version_minor = status.firmware_version_st.minor; out->serial_number_smart_card = status.card_serial_u32; out->config_numlock = status.numlock; out->config_capslock = status.capslock; out->config_scrolllock = status.scrolllock; out->otp_user_password = status.enable_user_password != 0; return 0; } NK_C_API char * NK_device_serial_number() { auto m = NitrokeyManager::instance(); return get_with_string_result([&]() { string && s = m->get_serial_number(); char * rs = strndup(s.c_str(), max_string_field_length); clear_string(s); return rs; }); } NK_C_API uint32_t NK_device_serial_number_as_u32() { auto m = NitrokeyManager::instance(); return get_with_result([&]() { return m->get_serial_number_as_u32(); }); } NK_C_API void NK_set_debug(bool state) { auto m = NitrokeyManager::instance(); m->set_debug(state); } NK_C_API void NK_set_debug_level(const int level) { auto m = NitrokeyManager::instance(); m->set_loglevel(level); } NK_C_API unsigned int NK_get_major_library_version() { return get_major_library_version(); } NK_C_API unsigned int NK_get_minor_library_version() { return get_minor_library_version(); } NK_C_API const char* NK_get_library_version() { return get_library_version(); } NK_C_API int NK_change_admin_PIN(const char *current_PIN, const char *new_PIN) { auto m = NitrokeyManager::instance(); return get_without_result([&]() { m->change_admin_PIN(current_PIN, new_PIN); }); } NK_C_API int NK_change_user_PIN(const char *current_PIN, const char *new_PIN) { auto m = NitrokeyManager::instance(); return get_without_result([&]() { m->change_user_PIN(current_PIN, new_PIN); }); } NK_C_API uint8_t NK_get_user_retry_count() { auto m = NitrokeyManager::instance(); return get_with_result([&]() { return m->get_user_retry_count(); }); } NK_C_API uint8_t NK_get_admin_retry_count() { auto m = NitrokeyManager::instance(); return get_with_result([&]() { return m->get_admin_retry_count(); }); } NK_C_API int NK_lock_device() { auto m = NitrokeyManager::instance(); return get_without_result([&]() { m->lock_device(); }); } NK_C_API int NK_is_AES_supported(const char *user_password) { auto m = NitrokeyManager::instance(); return get_with_result([&]() { return (uint8_t)m->is_AES_supported(user_password); }); } NK_C_API int NK_login_auto() { auto m = NitrokeyManager::instance(); return get_with_result([&]() { return (uint8_t)m->connect(); }); } NK_C_API uint8_t NK_get_major_firmware_version() { auto m = NitrokeyManager::instance(); return get_with_result([&]() { return m->get_major_firmware_version(); }); } NK_C_API uint8_t NK_get_minor_firmware_version() { auto m = NitrokeyManager::instance(); return get_with_result([&]() { return m->get_minor_firmware_version(); }); } NK_C_API char* NK_list_devices_by_cpuID() { auto nm = NitrokeyManager::instance(); return get_with_string_result([&]() { auto v = nm->list_devices_by_cpuID(); std::string res; for (const auto a : v){ res += a+";"; } if (res.size()>0) res.pop_back(); // remove last delimiter char return strndup(res.c_str(), MAXIMUM_STR_REPLY_LENGTH); }); } bool copy_device_info(const DeviceInfo& source, NK_device_info* target) { switch (source.m_deviceModel) { case DeviceModel::PRO: target->model = NK_PRO; break; case DeviceModel::STORAGE: target->model = NK_STORAGE; break; case DeviceModel::LIBREM: target->model = NK_LIBREM; break; default: return false; } target->path = strndup(source.m_path.c_str(), MAXIMUM_STR_REPLY_LENGTH); target->serial_number = strndup(source.m_serialNumber.c_str(), MAXIMUM_STR_REPLY_LENGTH); target->next = nullptr; return target->path && target->serial_number; } NK_C_API struct NK_device_info* NK_list_devices() { auto nm = NitrokeyManager::instance(); return get_with_result([&]() -> NK_device_info* { auto v = nm->list_devices(); if (v.empty()) return nullptr; auto result = new NK_device_info(); auto ptr = result; auto first = v.begin(); if (!copy_device_info(*first, ptr)) { NK_free_device_info(result); return nullptr; } v.erase(first); for (auto& info : v) { ptr->next = new NK_device_info(); ptr = ptr->next; if (!copy_device_info(info, ptr)) { NK_free_device_info(result); return nullptr; } } return result; }); } NK_C_API void NK_free_device_info(struct NK_device_info* device_info) { if (!device_info) return; if (device_info->next) NK_free_device_info(device_info->next); free(device_info->path); free(device_info->serial_number); delete device_info; } NK_C_API int NK_connect_with_ID(const char* id) { auto m = NitrokeyManager::instance(); return get_with_result([&]() { return m->connect_with_ID(id) ? 1 : 0; }); } NK_C_API int NK_connect_with_path(const char* path) { auto m = NitrokeyManager::instance(); return get_with_result([&]() { return m->connect_with_path(path) ? 1 : 0; }); } NK_C_API int NK_wink() { auto m = NitrokeyManager::instance(); return get_without_result([&]() { return m->wink(); }); } NK_C_API int NK_enable_firmware_update_pro(const char* update_password){ auto m = NitrokeyManager::instance(); return get_without_result([&]() { m->enable_firmware_update_pro(update_password); }); } NK_C_API int NK_change_firmware_password_pro(const char *current_firmware_password, const char *new_firmware_password) { auto m = NitrokeyManager::instance(); return get_without_result([&]() { m->change_firmware_update_password_pro(current_firmware_password, new_firmware_password); }); } #ifdef __cplusplus } #endif