assistants.php
3 years ago
chatbot.php
3 years ago
chatbot_legacy.php
3 years ago
chatbot_logs.php
3 years ago
discussions.php
3 years ago
security.php
3 years ago
discussions.php
212 lines
| 1 | <?php |
| 2 | |
| 3 | class Meow_MWAI_Modules_Discussions { |
| 4 | private $wpdb = null; |
| 5 | private $core = null; |
| 6 | private $table_chats = null; |
| 7 | private $db_check = false; |
| 8 | private $namespace = 'mwai/v1'; |
| 9 | |
| 10 | public function __construct() { |
| 11 | global $wpdb; |
| 12 | global $mwai_core; |
| 13 | $this->core = $mwai_core; |
| 14 | $this->wpdb = $wpdb; |
| 15 | $this->table_chats = $wpdb->prefix . 'mwai_chats'; |
| 16 | |
| 17 | if ( $this->core->get_option( 'shortcode_chat_discussions' ) ) { |
| 18 | add_filter( 'mwai_chatbot_reply', [ $this, 'chatbot_reply' ], 10, 4 ); |
| 19 | add_action( 'rest_api_init', array( $this, 'rest_api_init' ) ); |
| 20 | } |
| 21 | } |
| 22 | |
| 23 | public function rest_api_init() { |
| 24 | register_rest_route( $this->namespace, '/discussions/list', array( |
| 25 | 'methods' => 'POST', |
| 26 | 'callback' => [ $this, 'rest_discussions_list' ], |
| 27 | 'permission_callback' => '__return_true' |
| 28 | ) ); |
| 29 | register_rest_route( $this->namespace, '/discussions/delete', array( |
| 30 | 'methods' => 'POST', |
| 31 | 'callback' => [ $this, 'rest_discussions_delete' ], |
| 32 | 'permission_callback' => '__return_true' |
| 33 | ) ); |
| 34 | } |
| 35 | |
| 36 | function rest_discussions_list( $request ) { |
| 37 | try { |
| 38 | $params = $request->get_json_params(); |
| 39 | $offset = $params['offset']; |
| 40 | $limit = $params['limit']; |
| 41 | $filters = $params['filters']; |
| 42 | $sort = $params['sort']; |
| 43 | $chats = $this->chats_query( [], $offset, $limit, $filters, $sort ); |
| 44 | return new WP_REST_Response([ 'success' => true, 'total' => $chats['total'], 'chats' => $chats['rows'] ], 200 ); |
| 45 | } |
| 46 | catch ( Exception $e ) { |
| 47 | return new WP_REST_Response([ 'success' => false, 'message' => $e->getMessage() ], 500 ); |
| 48 | } |
| 49 | } |
| 50 | |
| 51 | function rest_discussions_delete( $request ) { |
| 52 | try { |
| 53 | $params = $request->get_json_params(); |
| 54 | $chatsIds = $params['chatIds']; |
| 55 | if ( is_array( $chatsIds ) ) { |
| 56 | if ( count( $chatsIds ) === 0 ) { |
| 57 | $this->wpdb->query( "TRUNCATE TABLE $this->table_chats" ); |
| 58 | } |
| 59 | foreach( $chatsIds as $chatId ) { |
| 60 | $this->wpdb->delete( $this->table_chats, [ 'chatId' => $chatId ] ); |
| 61 | } |
| 62 | } |
| 63 | return new WP_REST_Response([ 'success' => true ], 200 ); |
| 64 | } |
| 65 | catch ( Exception $e ) { |
| 66 | return new WP_REST_Response([ 'success' => false, 'message' => $e->getMessage() ], 500 ); |
| 67 | } |
| 68 | } |
| 69 | |
| 70 | function chats_query( $chats = [], $offset = 0, $limit = null, $filters = null, $sort = null ) { |
| 71 | $this->check_db(); |
| 72 | $offset = !empty( $offset ) ? intval( $offset ) : 0; |
| 73 | $limit = !empty( $limit ) ? intval( $limit ) : 5; |
| 74 | $filters = !empty( $filters ) ? $filters : []; |
| 75 | $sort = !empty( $sort ) ? $sort : [ 'accessor' => 'time', 'by' => 'desc' ]; |
| 76 | $query = "SELECT * FROM $this->table_chats"; |
| 77 | |
| 78 | // Filters |
| 79 | if ( is_array( $filters ) ) { |
| 80 | $where = array(); |
| 81 | foreach ( $filters as $filter ) { |
| 82 | if ( $filter['accessor'] === 'user' ) { |
| 83 | $value = esc_sql( $filter['value'] ); |
| 84 | if ( empty( $value ) ) { |
| 85 | continue; |
| 86 | } |
| 87 | $isIP = filter_var( $value, FILTER_VALIDATE_IP ); |
| 88 | if ( $isIP ) { |
| 89 | $where[] = "extra LIKE '%\"ip\":\"{$value}\"%'"; |
| 90 | } |
| 91 | else { |
| 92 | $where[] = "extra LIKE '%\"userId\":{$value}%'"; |
| 93 | } |
| 94 | } |
| 95 | if ( $filter['accessor'] === 'preview' ) { |
| 96 | $value = $filter['value']; |
| 97 | if ( empty( $value ) ) { |
| 98 | continue; |
| 99 | } |
| 100 | $where[] = "messages LIKE '%{$value}%'"; |
| 101 | } |
| 102 | } |
| 103 | if ( count( $where ) > 0 ) { |
| 104 | $query .= " WHERE " . implode( " AND ", $where ); |
| 105 | } |
| 106 | } |
| 107 | |
| 108 | // Count based on this query |
| 109 | $chats['total'] = $this->wpdb->get_var( "SELECT COUNT(*) FROM ($query) AS t" ); |
| 110 | |
| 111 | // Order by |
| 112 | $query .= " ORDER BY " . esc_sql( $sort['accessor'] ) . " " . esc_sql( $sort['by'] ); |
| 113 | |
| 114 | // Limits |
| 115 | if ( $limit > 0 ) { |
| 116 | $query .= " LIMIT $offset, $limit"; |
| 117 | } |
| 118 | |
| 119 | $chats['rows'] = $this->wpdb->get_results( $query, ARRAY_A ); |
| 120 | return $chats; |
| 121 | } |
| 122 | |
| 123 | function chatbot_reply( $rawText, $query, $params, $extra ) { |
| 124 | global $mwai_core; |
| 125 | $userIp = $mwai_core->get_ip_address(); |
| 126 | $userId = $mwai_core->get_user_id(); |
| 127 | $chatClientId = isset( $params['clientId'] ) ? $params['clientId'] : $query->session; |
| 128 | $ssChatId = hash( 'sha256', $userIp . $userId . $chatClientId ); |
| 129 | $this->check_db(); |
| 130 | $chat = $this->wpdb->get_row( $this->wpdb->prepare( "SELECT * FROM $this->table_chats WHERE chatId = %s", $ssChatId ) ); |
| 131 | $extra = [ |
| 132 | 'embeddings' => isset( $extra['embeddings'] ) ? $extra['embeddings'] : null |
| 133 | ]; |
| 134 | if ( $chat ) { |
| 135 | $chat->messages = json_decode( $chat->messages ); |
| 136 | $chat->messages[] = [ |
| 137 | 'type' => 'user', |
| 138 | 'text' => $params['newMessage'] |
| 139 | ]; |
| 140 | $chat->messages[] = [ |
| 141 | 'type' => 'ai', |
| 142 | 'text' => $rawText, |
| 143 | 'extra' => $extra |
| 144 | ]; |
| 145 | $chat->messages = json_encode( $chat->messages ); |
| 146 | $this->wpdb->update( $this->table_chats, [ |
| 147 | 'messages' => $chat->messages, |
| 148 | 'updated' => date( 'Y-m-d H:i:s' ) |
| 149 | ], [ 'id' => $chat->id ] ); |
| 150 | } |
| 151 | else { |
| 152 | $chat = [ |
| 153 | 'chatId' => $ssChatId, |
| 154 | 'messages' => json_encode( [ |
| 155 | [ |
| 156 | 'type' => 'user', |
| 157 | 'text' => $params['newMessage'] |
| 158 | ], |
| 159 | [ |
| 160 | 'type' => 'ai', |
| 161 | 'text' => $rawText, |
| 162 | 'extra' => $extra |
| 163 | ] |
| 164 | ] ), |
| 165 | 'extra' => json_encode( [ |
| 166 | 'ip' => $userIp, |
| 167 | 'userId' => $userId, |
| 168 | 'session' => $query->session, |
| 169 | 'model' => $query->model, |
| 170 | 'temperature' => $query->temperature |
| 171 | ] ), |
| 172 | 'created' => date( 'Y-m-d H:i:s' ), |
| 173 | 'updated' => date( 'Y-m-d H:i:s' ) |
| 174 | ]; |
| 175 | $this->wpdb->insert( $this->table_chats, $chat ); |
| 176 | } |
| 177 | return $rawText; |
| 178 | } |
| 179 | |
| 180 | function check_db() { |
| 181 | if ( $this->db_check ) { |
| 182 | return true; |
| 183 | } |
| 184 | $this->db_check = !( strtolower( |
| 185 | $this->wpdb->get_var( "SHOW TABLES LIKE '$this->table_chats'" ) ) != strtolower( $this->table_chats ) |
| 186 | ); |
| 187 | if ( !$this->db_check ) { |
| 188 | $this->create_db(); |
| 189 | $this->db_check = !( strtolower( |
| 190 | $this->wpdb->get_var( "SHOW TABLES LIKE '$this->table_chats'" ) ) != strtolower( $this->table_chats ) |
| 191 | ); |
| 192 | } |
| 193 | return $this->db_check; |
| 194 | } |
| 195 | |
| 196 | function create_db() { |
| 197 | $charset_collate = $this->wpdb->get_charset_collate(); |
| 198 | $sqlLogs = "CREATE TABLE $this->table_chats ( |
| 199 | id BIGINT(20) NOT NULL AUTO_INCREMENT, |
| 200 | chatId VARCHAR(64) NOT NULL NULL, |
| 201 | messages TEXT NOT NULL NULL, |
| 202 | extra TEXT NOT NULL NULL, |
| 203 | created DATETIME NOT NULL, |
| 204 | updated DATETIME NOT NULL, |
| 205 | PRIMARY KEY (id), |
| 206 | INDEX chatId (chatId) |
| 207 | ) $charset_collate;"; |
| 208 | require_once( ABSPATH . 'wp-admin/includes/upgrade.php' ); |
| 209 | dbDelta( $sqlLogs ); |
| 210 | } |
| 211 | |
| 212 | } |