|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451 |
- /*
- * Carla REST API Server
- * Copyright (C) 2018 Filipe Coelho <falktx@falktx.com>
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public License as
- * published by the Free Software Foundation; either version 2 of
- * the License, or any later version.
- *
- * This program 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.
- *
- * For a full copy of the GNU General Public License see the doc/GPL.txt file.
- */
-
- /* NOTE
- * Even though Carla is GPL, restbed if AGPL.
- * As such, the resulting binary will be AGPL.
- * Take this into consideration before deploying it to any servers.
- */
-
- #include "common.hpp"
-
- #include "carla-host.cpp"
- #include "carla-utils.cpp"
-
- #include "CarlaMutex.hpp"
- #include "CarlaStringList.hpp"
-
- // -------------------------------------------------------------------------------------------------------------------
-
- #include <map>
- #include <restbed>
- #include <system_error>
- #include <openssl/sha.h>
- #include <openssl/hmac.h>
- #include <openssl/evp.h>
- #include <openssl/bio.h>
- #include <openssl/buffer.h>
-
- using namespace std;
- using namespace restbed;
- using namespace std::chrono;
-
- // std::vector<std::shared_ptr<Session>> gSessions;
-
- CarlaStringList gSessionMessages;
- CarlaMutex gSessionMessagesMutex;
-
- std::map< string, shared_ptr< WebSocket > > sockets = { };
-
- // -------------------------------------------------------------------------------------------------------------------
-
- void send_server_side_message(const char* const message)
- {
- const CarlaMutexLocker cml(gSessionMessagesMutex);
-
- gSessionMessages.append(message);
- }
-
- // -------------------------------------------------------------------------------------------------------------------
-
- static void event_stream_handler(void)
- {
- static bool firstInit = true;
-
- if (firstInit)
- {
- firstInit = false;
- carla_stdout("Carla REST-API Server started");
- }
-
- const bool running = carla_is_engine_running();
-
- if (running)
- carla_engine_idle();
-
- CarlaStringList messages;
-
- {
- const CarlaMutexLocker cml(gSessionMessagesMutex);
-
- if (gSessionMessages.count() > 0)
- gSessionMessages.moveTo(messages);
- }
-
- for (auto message : messages)
- {
- for (auto entry : sockets)
- {
- auto socket = entry.second;
-
- if (socket->is_open())
- socket->send(message);
- }
- }
-
- if (running)
- {
- if (const uint count = carla_get_current_plugin_count())
- {
- char msgBuf[1024];
- float* peaks;
-
- for (uint i=0; i<count; ++i)
- {
- peaks = carla_get_peak_values(i);
- CARLA_SAFE_ASSERT_BREAK(peaks != nullptr);
-
- std::snprintf(msgBuf, 1023, "Peaks: %u %f %f %f %f", i, peaks[0], peaks[1], peaks[2], peaks[3]);
- msgBuf[1023] = '\0';
-
- for (auto entry : sockets)
- {
- auto socket = entry.second;
-
- if (socket->is_open())
- socket->send(msgBuf);
- }
- }
- }
- }
-
- for (auto entry : sockets)
- {
- auto socket = entry.second;
-
- if (socket->is_open())
- socket->send("Keep-Alive");
- }
- }
-
-
- // -------------------------------------------------------------------------------------------------------------------
-
- string base64_encode( const unsigned char* input, int length )
- {
- BIO* bmem, *b64;
- BUF_MEM* bptr;
-
- b64 = BIO_new( BIO_f_base64( ) );
- bmem = BIO_new( BIO_s_mem( ) );
- b64 = BIO_push( b64, bmem );
- BIO_write( b64, input, length );
- ( void ) BIO_flush( b64 );
- BIO_get_mem_ptr( b64, &bptr );
-
- char* buff = ( char* )malloc( bptr->length );
- memcpy( buff, bptr->data, bptr->length - 1 );
- buff[ bptr->length - 1 ] = 0;
-
- BIO_free_all( b64 );
-
- return buff;
- }
-
- multimap< string, string > build_websocket_handshake_response_headers( const shared_ptr< const Request >& request )
- {
- auto key = request->get_header( "Sec-WebSocket-Key" );
- key.append( "258EAFA5-E914-47DA-95CA-C5AB0DC85B11" );
-
- Byte hash[ SHA_DIGEST_LENGTH ];
- SHA1( reinterpret_cast< const unsigned char* >( key.data( ) ), key.length( ), hash );
-
- multimap< string, string > headers;
- headers.insert( make_pair( "Upgrade", "websocket" ) );
- headers.insert( make_pair( "Connection", "Upgrade" ) );
- headers.insert( make_pair( "Sec-WebSocket-Accept", base64_encode( hash, SHA_DIGEST_LENGTH ) ) );
-
- return headers;
- }
-
- void close_handler( const shared_ptr< WebSocket > socket )
- {
- carla_stdout("CLOSE %i", __LINE__);
-
- if ( socket->is_open( ) )
- {
- auto response = make_shared< WebSocketMessage >( WebSocketMessage::CONNECTION_CLOSE_FRAME, Bytes( { 10, 00 } ) );
- socket->send( response );
- }
- carla_stdout("CLOSE %i", __LINE__);
-
- const auto key = socket->get_key( );
- sockets.erase( key );
-
- fprintf( stderr, "Closed connection to %s.\n", key.data( ) );
- }
-
- void error_handler( const shared_ptr< WebSocket > socket, const error_code error )
- {
- const auto key = socket->get_key( );
- fprintf( stderr, "WebSocket Errored '%s' for %s.\n", error.message( ).data( ), key.data( ) );
- }
-
- void message_handler( const shared_ptr< WebSocket > source, const shared_ptr< WebSocketMessage > message )
- {
- const auto opcode = message->get_opcode( );
-
- if ( opcode == WebSocketMessage::PING_FRAME )
- {
- auto response = make_shared< WebSocketMessage >( WebSocketMessage::PONG_FRAME, message->get_data( ) );
- source->send( response );
- }
- else if ( opcode == WebSocketMessage::PONG_FRAME )
- {
- //Ignore PONG_FRAME.
- //
- //Every time the ping_handler is scheduled to run, it fires off a PING_FRAME to each
- //WebSocket. The client, if behaving correctly, will respond with a PONG_FRAME.
- //
- //On each occasion the underlying TCP socket sees any packet data transfer, whether
- //a PING, PONG, TEXT, or BINARY... frame. It will automatically reset the timeout counter
- //leaving the connection active; see also Settings::set_connection_timeout.
- return;
- }
- else if ( opcode == WebSocketMessage::CONNECTION_CLOSE_FRAME )
- {
- source->close( );
- }
- else if ( opcode == WebSocketMessage::BINARY_FRAME )
- {
- //We don't support binary data.
- auto response = make_shared< WebSocketMessage >( WebSocketMessage::CONNECTION_CLOSE_FRAME, Bytes( { 10, 03 } ) );
- source->send( response );
- }
- else if ( opcode == WebSocketMessage::TEXT_FRAME )
- {
- auto response = make_shared< WebSocketMessage >( *message );
- response->set_mask( 0 );
-
- for ( auto socket : sockets )
- {
- auto destination = socket.second;
- destination->send( response );
- }
-
- const auto key = source->get_key( );
- const auto data = String::format( "Received message '%.*s' from %s\n", message->get_data( ).size( ), message->get_data( ).data( ), key.data( ) );
- fprintf( stderr, "%s", data.data( ) );
- }
- }
-
- void get_method_handler(const shared_ptr<Session> session)
- {
- carla_stdout("HERE %i", __LINE__);
- const auto request = session->get_request();
- const auto connection_header = request->get_header("connection", String::lowercase);
- carla_stdout("HERE %i", __LINE__);
-
- if ( connection_header.find( "upgrade" ) not_eq string::npos )
- {
- if ( request->get_header( "upgrade", String::lowercase ) == "websocket" )
- {
- const auto headers = build_websocket_handshake_response_headers( request );
-
- session->upgrade( SWITCHING_PROTOCOLS, headers, [ ]( const shared_ptr< WebSocket > socket )
- {
- if ( socket->is_open( ) )
- {
- socket->set_close_handler( close_handler );
- socket->set_error_handler( error_handler );
- socket->set_message_handler( message_handler );
-
- socket->send("Welcome to Corvusoft Chat!");
-
- auto key = socket->get_key( );
- sockets[key] = socket;
- }
- else
- {
- fprintf( stderr, "WebSocket Negotiation Failed: Client closed connection.\n" );
- }
- } );
-
- return;
- }
- }
-
- session->close( BAD_REQUEST );
- }
-
- void ping_handler( void )
- {
- for ( auto entry : sockets )
- {
- auto key = entry.first;
- auto socket = entry.second;
-
- if ( socket->is_open( ) )
- {
- socket->send( WebSocketMessage::PING_FRAME );
- }
- else
- {
- socket->close( );
- }
- }
- }
-
- // -------------------------------------------------------------------------------------------------------------------
-
- static void make_resource(Service& service,
- const char* const path,
- const std::function<void (const std::shared_ptr<Session>)>& callback)
- {
- std::shared_ptr<Resource> resource = std::make_shared<Resource>();
- resource->set_path(path);
- resource->set_method_handler("GET", callback);
- service.publish(resource);
- }
-
- // -------------------------------------------------------------------------------------------------------------------
-
- int main(int, const char**)
- {
- Service service;
-
- // websocket
- {
- std::shared_ptr<Resource> resource = std::make_shared<Resource>();
- resource->set_path("/ws");
- resource->set_method_handler("GET", get_method_handler);
- service.publish(resource);
- }
-
- // carla-host
- make_resource(service, "/get_engine_driver_count", handle_carla_get_engine_driver_count);
- make_resource(service, "/get_engine_driver_name", handle_carla_get_engine_driver_name);
- make_resource(service, "/get_engine_driver_device_names", handle_carla_get_engine_driver_device_names);
- make_resource(service, "/get_engine_driver_device_info", handle_carla_get_engine_driver_device_info);
-
- make_resource(service, "/engine_init", handle_carla_engine_init);
- make_resource(service, "/engine_close", handle_carla_engine_close);
- make_resource(service, "/is_engine_running", handle_carla_is_engine_running);
- make_resource(service, "/set_engine_about_to_close", handle_carla_set_engine_about_to_close);
-
- make_resource(service, "/set_engine_option", handle_carla_set_engine_option);
- make_resource(service, "/load_file", handle_carla_load_file);
- make_resource(service, "/load_project", handle_carla_load_project);
- make_resource(service, "/save_project", handle_carla_save_project);
-
- make_resource(service, "/patchbay_connect", handle_carla_patchbay_connect);
- make_resource(service, "/patchbay_disconnect", handle_carla_patchbay_disconnect);
- make_resource(service, "/patchbay_refresh", handle_carla_patchbay_refresh);
-
- make_resource(service, "/transport_play", handle_carla_transport_play);
- make_resource(service, "/transport_pause", handle_carla_transport_pause);
- make_resource(service, "/transport_bpm", handle_carla_transport_bpm);
- make_resource(service, "/transport_relocate", handle_carla_transport_relocate);
- make_resource(service, "/get_current_transport_frame", handle_carla_get_current_transport_frame);
- make_resource(service, "/get_transport_info", handle_carla_get_transport_info);
-
- make_resource(service, "/get_current_plugin_count", handle_carla_get_current_plugin_count);
- make_resource(service, "/get_max_plugin_number", handle_carla_get_max_plugin_number);
- make_resource(service, "/add_plugin", handle_carla_add_plugin);
- make_resource(service, "/remove_plugin", handle_carla_remove_plugin);
- make_resource(service, "/remove_all_plugins", handle_carla_remove_all_plugins);
-
- make_resource(service, "/rename_plugin", handle_carla_rename_plugin);
- make_resource(service, "/clone_plugin", handle_carla_clone_plugin);
- make_resource(service, "/replace_plugin", handle_carla_replace_plugin);
- make_resource(service, "/switch_plugins", handle_carla_switch_plugins);
-
- make_resource(service, "/load_plugin_state", handle_carla_load_plugin_state);
- make_resource(service, "/save_plugin_state", handle_carla_save_plugin_state);
- make_resource(service, "/export_plugin_lv2", handle_carla_export_plugin_lv2);
-
- make_resource(service, "/get_plugin_info", handle_carla_get_plugin_info);
- make_resource(service, "/get_audio_port_count_info", handle_carla_get_audio_port_count_info);
- make_resource(service, "/get_midi_port_count_info", handle_carla_get_midi_port_count_info);
- make_resource(service, "/get_parameter_count_info", handle_carla_get_parameter_count_info);
- make_resource(service, "/get_parameter_info", handle_carla_get_parameter_info);
- make_resource(service, "/get_parameter_scalepoint_info", handle_carla_get_parameter_scalepoint_info);
-
- make_resource(service, "/get_parameter_data", handle_carla_get_parameter_data);
- make_resource(service, "/get_parameter_ranges", handle_carla_get_parameter_ranges);
- make_resource(service, "/get_midi_program_data", handle_carla_get_midi_program_data);
- make_resource(service, "/get_custom_data", handle_carla_get_custom_data);
- make_resource(service, "/get_custom_data_value", handle_carla_get_custom_data_value);
- make_resource(service, "/get_chunk_data", handle_carla_get_chunk_data);
-
- make_resource(service, "/get_parameter_count", handle_carla_get_parameter_count);
- make_resource(service, "/get_program_count", handle_carla_get_program_count);
- make_resource(service, "/get_midi_program_count", handle_carla_get_midi_program_count);
- make_resource(service, "/get_custom_data_count", handle_carla_get_custom_data_count);
-
- make_resource(service, "/get_parameter_text", handle_carla_get_parameter_text);
- make_resource(service, "/get_program_name", handle_carla_get_program_name);
- make_resource(service, "/get_midi_program_name", handle_carla_get_midi_program_name);
- make_resource(service, "/get_real_plugin_name", handle_carla_get_real_plugin_name);
-
- make_resource(service, "/get_current_program_index", handle_carla_get_current_program_index);
- make_resource(service, "/get_current_midi_program_index", handle_carla_get_current_midi_program_index);
-
- make_resource(service, "/get_default_parameter_value", handle_carla_get_default_parameter_value);
- make_resource(service, "/get_current_parameter_value", handle_carla_get_current_parameter_value);
- make_resource(service, "/get_internal_parameter_value", handle_carla_get_internal_parameter_value);
- make_resource(service, "/get_input_peak_value", handle_carla_get_input_peak_value);
- make_resource(service, "/get_output_peak_value", handle_carla_get_output_peak_value);
-
- make_resource(service, "/set_active", handle_carla_set_active);
- make_resource(service, "/set_drywet", handle_carla_set_drywet);
- make_resource(service, "/set_volume", handle_carla_set_volume);
- make_resource(service, "/set_balance_left", handle_carla_set_balance_left);
- make_resource(service, "/set_balance_right", handle_carla_set_balance_right);
- make_resource(service, "/set_panning", handle_carla_set_panning);
- make_resource(service, "/set_ctrl_channel", handle_carla_set_ctrl_channel);
- make_resource(service, "/set_option", handle_carla_set_option);
-
- make_resource(service, "/set_parameter_value", handle_carla_set_parameter_value);
- make_resource(service, "/set_parameter_midi_channel", handle_carla_set_parameter_midi_channel);
- make_resource(service, "/set_parameter_midi_cc", handle_carla_set_parameter_midi_cc);
- make_resource(service, "/set_program", handle_carla_set_program);
- make_resource(service, "/set_midi_program", handle_carla_set_midi_program);
- make_resource(service, "/set_custom_data", handle_carla_set_custom_data);
- make_resource(service, "/set_chunk_data", handle_carla_set_chunk_data);
-
- make_resource(service, "/prepare_for_save", handle_carla_prepare_for_save);
- make_resource(service, "/reset_parameters", handle_carla_reset_parameters);
- make_resource(service, "/randomize_parameters", handle_carla_randomize_parameters);
- make_resource(service, "/send_midi_note", handle_carla_send_midi_note);
-
- make_resource(service, "/get_buffer_size", handle_carla_get_buffer_size);
- make_resource(service, "/get_sample_rate", handle_carla_get_sample_rate);
- make_resource(service, "/get_last_error", handle_carla_get_last_error);
- make_resource(service, "/get_host_osc_url_tcp", handle_carla_get_host_osc_url_tcp);
- make_resource(service, "/get_host_osc_url_udp", handle_carla_get_host_osc_url_udp);
-
- // carla-utils
- make_resource(service, "/get_complete_license_text", handle_carla_get_complete_license_text);
- make_resource(service, "/get_supported_file_extensions", handle_carla_get_supported_file_extensions);
- make_resource(service, "/get_supported_features", handle_carla_get_supported_features);
- make_resource(service, "/get_cached_plugin_count", handle_carla_get_cached_plugin_count);
- make_resource(service, "/get_cached_plugin_info", handle_carla_get_cached_plugin_info);
-
- // schedule events
- service.schedule(event_stream_handler, std::chrono::milliseconds(33));
- service.schedule(ping_handler, milliseconds(5000));
-
- std::shared_ptr<Settings> settings = std::make_shared<Settings>();
- settings->set_port(2228);
- settings->set_default_header("Connection", "close");
-
- service.start(settings);
- return 0;
- }
-
- // -------------------------------------------------------------------------------------------------------------------
|