is_connected() ) { $this->set_param( 'new_connection', 1 ); } $this->output_controls_html( 'sendinblue-v3' ); } /** * should handle: read data from post / get, test connection and save the details * * on error, it should register an error message (and redirect?) */ public function read_credentials() { $ajax_call = defined( 'DOING_AJAX' ) && DOING_AJAX; $key = ! empty( $_POST['connection']['key'] ) ? $_POST['connection']['key'] : ''; if ( empty( $key ) ) { $message = 'You must provide a valid Brevo key'; return $ajax_call ? __( $message, 'thrive-dash' ) : $this->error( __( $message, 'thrive-dash' ) ); } $this->set_credentials( $_POST['connection'] ); $result = $this->test_connection(); if ( $result !== true ) { $message = 'Could not connect to Brevo using the provided key (%s)'; return $ajax_call ? sprintf( __( $message, 'thrive-dash' ), $result ) : $this->error( sprintf( __( $message, 'thrive-dash' ), $result ) ); } /** * finally, save the connection details */ $this->save(); /** @var Thrive_Dash_List_Connection_SendinblueEmail $related_api */ $related_api = Thrive_Dash_List_Manager::connection_instance( 'sendinblueemail' ); if ( isset( $_POST['connection']['new_connection'] ) && (int) $_POST['connection']['new_connection'] ) { /** * Try to connect to the email service too */ $r_result = true; if ( ! $related_api->is_connected() ) { $r_result = $related_api->read_credentials(); } if ( $r_result !== true ) { $this->disconnect(); return $this->error( $r_result ); } } else { /** * let's make sure that the api was not edited and disconnect it */ $related_api->set_credentials( array() ); Thrive_Dash_List_Manager::save( $related_api ); } $this->success( __( 'Brevo connected successfully', 'thrive-dash' ) ); if ( $ajax_call ) { return true; } } /** * Test if a connection can be made to the service using the stored credentials * * @return bool|string true for success or error message for failure */ public function test_connection() { if ( ! $this->is_v3() ) { return parent::test_connection(); } $sendinblue = $this->get_api(); try { $sendinblue->getAccount(); } catch ( Exception $e ) { return $e->getMessage(); } return true; } /** * Instantiate the API code required for this connection * * @return Thrive_Dash_Api_SendinblueV3 * @throws Exception */ protected function get_api_instance() { if ( ! $this->is_v3() ) { return parent::get_api_instance(); } return new Thrive_Dash_Api_SendinblueV3( 'https://api.sendinblue.com/v3', $this->param( 'key' ) ); } /** * Get all Subscriber Lists from this API service * * @return array|bool for error */ protected function _get_lists() { if ( ! $this->is_v3() ) { return parent::_get_lists(); } /** @var Thrive_Dash_Api_Sendinblue $sendinblue */ $sendinblue = $this->get_api(); $data = array( 'limit' => 50, 'offset' => 0, ); try { $lists = array(); $raw = $sendinblue->getLists( $data ); if ( empty( $raw['lists'] ) ) { return array(); } if ( ! empty( $raw['count'] ) && $raw['count'] > $data['limit'] ) { $total_loaded = count( $raw['lists'] ); $remaining_pages = (int) ( $raw['count'] / $data['limit'] ) + 1; if ( $remaining_pages >= 2 ) { for ( $i = 2; $i <= $remaining_pages; $i ++ ) { $data['offset'] = $total_loaded; $response = $sendinblue->getLists( $data ); if ( isset( $response['lists'] ) && is_array( $response['lists'] ) ) { $raw['lists'] = array_merge( $raw['lists'], $response['lists'] ); } $total_loaded = count( $raw['lists'] ); } } } foreach ( $raw['lists'] as $item ) { $lists [] = array( 'id' => $item['id'], 'name' => $item['name'], ); } return $lists; } catch ( Exception $e ) { $this->_error = $e->getMessage() . ' ' . __( 'Please re-check your API connection details.', 'thrive-dash' ); return false; } } /** * Add a contact to a list * * @param mixed $list_identifier * @param array $arguments * * @return mixed */ public function add_subscriber( $list_identifier, $arguments ) { if ( ! is_array( $arguments ) ) { $arguments = (array) $arguments; } if ( ! $this->is_v3() ) { return parent::add_subscriber( $list_identifier, $arguments ); } $merge_tags = array(); if ( ! empty( $arguments['name'] ) ) { list( $first_name, $last_name ) = $this->get_name_parts( $arguments['name'] ); $merge_tags = array( 'NAME' => $first_name, 'FIRSTNAME' => $first_name, 'SURNAME' => $last_name, 'VORNAME' => $first_name, 'NACHNAME' => $last_name, 'LASTNAME' => $last_name, 'NOM' => $last_name, 'PRENOM' => $first_name, ); } /** @var Thrive_Dash_Api_SendinblueV3 $api */ $api = $this->get_api(); if ( ! empty( $arguments['phone'] ) ) { // SendinBlue does not accept phone numbers starting with 0 or other special chars $the_phone = ltrim( ( preg_replace( '/[^0-9]/', '', $arguments['phone'] ) ), '0' ); $merge_tags['SMS'] = $the_phone; $merge_tags['PHONE'] = $the_phone; $merge_tags['TELEFON'] = $the_phone; } if ( empty( $arguments['automator_custom_fields'] ) ) { $attributes = array_merge( $merge_tags, $this->_generate_custom_fields( $arguments ) ); } else { $attributes = array_merge( $merge_tags, $arguments['automator_custom_fields'] ); } $data = array( 'email' => $arguments['email'], 'listIds' => array( (int) $list_identifier ), 'updateEnabled' => true, ); if ( ! empty( $attributes ) ) { $data['attributes'] = $attributes; } try { $api->create_update_user( $data ); return true; } catch ( Exception $e ) { return $e->getMessage() ?: __( 'Unknown Error', 'thrive-dash' ); } } /** * Disconnect (remove) this API connection */ public function disconnect() { if ( ! $this->is_v3() ) { return parent::disconnect(); } $this->set_credentials( array() ); Thrive_Dash_List_Manager::save( $this ); /** * disconnect the email service too */ $related_api = Thrive_Dash_List_Manager::connection_instance( 'sendinblueemail' ); $related_api->set_credentials( array() ); Thrive_Dash_List_Manager::save( $related_api ); return $this; } /** * Return the connection email merge tag * * @return String */ public static function get_email_merge_tag() { return '{{ contact.EMAIL }}'; } /** * @param array $params which may contain `list_id` * @param bool $force make a call to API and invalidate cache * @param bool $get_all where to get lists with their custom fields * * @return array */ public function get_api_custom_fields( $params, $force = false, $get_all = true ) { if ( ! $this->is_v3() ) { return parent::get_api_custom_fields( $params, $force, $get_all ); } $cached_data = $this->get_cached_custom_fields(); if ( false === $force && ! empty( $cached_data ) ) { return $cached_data; } /** @var Thrive_Dash_Api_Sendinblue $api */ $api = $this->get_api(); try { $attributes = $api->get_attributes(); } catch ( Thrive_Dash_Api_SendinBlue_Exception $e ) { // Maybe log this } $custom_fields = array(); $excluded_fields = array( 'NAME', 'FIRSTNAME', 'SURNAME', 'VORNAME', 'LASTNAME', 'SMS', 'PHONE', 'TELEFON', ); if ( ! empty( $attributes['attributes'] ) ) { foreach ( $attributes['attributes'] as $attribute ) { if ( ! empty( $attribute['category'] ) && $attribute['category'] === 'normal' && ! empty( $attribute['type'] ) && ! in_array( $attribute['name'], $excluded_fields ) && 'text' === $attribute['type'] ) { $custom_fields[] = $this->normalize_custom_field( $attribute ); } } } $this->_save_custom_fields( $custom_fields ); return $custom_fields; } /** * @param array $field * * @return array */ protected function normalize_custom_field( $field = array() ) { return array( 'id' => ! empty( $field['name'] ) ? $field['name'] : '', 'name' => ! empty( $field['name'] ) ? $field['name'] : '', 'type' => ! empty( $field['type'] ) ? $field['type'] : '', 'label' => ! empty( $field['name'] ) ? $field['name'] : '', ); } /** * Generate custom fields array * * @param array $args * * @return array */ private function _generate_custom_fields( $args ) { $custom_fields = $this->get_api_custom_fields( array() ); $ids = $this->build_mapped_custom_fields( $args ); $result = array(); foreach ( $ids as $key => $id ) { $field = array_filter( $custom_fields, function ( $item ) use ( $id ) { return $item['id'] === $id['value']; } ); $field = array_values( $field ); if ( ! isset( $field[0] ) ) { continue; } $name = strpos( $id['type'], 'mapping_' ) !== false ? $id['type'] . '_' . $key : $key; $cf_form_name = str_replace( '[]', '', $name ); $result[ $field[0]['name'] ] = $this->process_field( $args[ $cf_form_name ] ); } return $result; } /** * Build custom fields mapping for automations * * @param $automation_data * * @return object */ public function build_automation_custom_fields( $automation_data ) { $mapped_data = []; $fields = $this->get_api_custom_fields( array() ); foreach ( $automation_data['api_fields'] as $pair ) { foreach ( $fields as $field ) { if ( $field['id'] == $pair['key'] ) { $value = sanitize_text_field( $pair['value'] ); if ( $value ) { $mapped_data[ $field['name'] ] = $value; } } } } return $mapped_data; } /** * Build mapped custom fields array based on form params * * @param $args * * @return array */ public function build_mapped_custom_fields( $args ) { $mapped_data = array(); // Should be always base_64 encoded of a serialized array if ( empty( $args['tve_mapping'] ) || ! tve_dash_is_bas64_encoded( $args['tve_mapping'] ) ) { return $mapped_data; } $form_data = thrive_safe_unserialize( base64_decode( $args['tve_mapping'] ) ); $mapped_fields = $this->get_mapped_field_ids(); foreach ( $mapped_fields as $mapped_field_name ) { // Extract an array with all custom fields (siblings) names from form data // {ex: [mapping_url_0, .. mapping_url_n] / [mapping_text_0, .. mapping_text_n]} $cf_form_fields = preg_grep( "#^{$mapped_field_name}#i", array_keys( $form_data ) ); if ( ! empty( $cf_form_fields ) && is_array( $cf_form_fields ) ) { foreach ( $cf_form_fields as $cf_form_name ) { if ( empty( $form_data[ $cf_form_name ][ $this->_key ] ) ) { continue; } $field_id = str_replace( $mapped_field_name . '_', '', $cf_form_name ); $mapped_data[ $field_id ] = array( 'type' => $mapped_field_name, 'value' => $form_data[ $cf_form_name ][ $this->_key ], ); } } } return $mapped_data; } /** * @param $email * @param array $custom_fields * @param array $extra * * @return int */ public function add_custom_fields( $email, $custom_fields = array(), $extra = array() ) { if ( ! $this->is_v3() ) { return parent::add_custom_fields( $email, $custom_fields = array(), $extra = array() ); } try { /** @var Thrive_Dash_Api_Sendinblue $api */ $api = $this->get_api(); $list_id = ! empty( $extra['list_identifier'] ) ? $extra['list_identifier'] : null; $args = array( 'email' => $email, ); if ( ! empty( $extra['name'] ) ) { $args['name'] = $extra['name']; } $this->add_subscriber( $list_id, $args ); $args['attributes'] = $this->prepare_custom_fields_for_api( $custom_fields ); $subscriber = $api->create_update_user( $args ); return ! empty( $subscriber['data']['id'] ) ? $subscriber['data']['id'] : 0; } catch ( Exception $e ) { return false; } } /** * Get available custom fields for this api connection * * @param null $list_id * * @return array */ public function get_available_custom_fields( $list_id = null ) { return $this->get_api_custom_fields( null, true ); } /** * Prepare custom fields for api call * * @param array $custom_fields * @param null $list_identifier * * @return array */ public function prepare_custom_fields_for_api( $custom_fields = array(), $list_identifier = null ) { $prepared_fields = array(); $api_fields = $this->get_api_custom_fields( null, true ); foreach ( $api_fields as $field ) { foreach ( $custom_fields as $key => $custom_field ) { if ( $custom_field && $field['id'] === $key ) { $prepared_fields[ $key ] = $custom_field; } } if ( empty( $custom_fields ) ) { break; } } return $prepared_fields; } public function get_automator_add_autoresponder_mapping_fields() { return array( 'autoresponder' => array( 'mailing_list', 'api_fields' ) ); } public function has_custom_fields() { return true; } }