assistants.php
3 years ago
chatbot.php
2 years ago
chatbot_legacy.php
2 years ago
discussions.php
2 years ago
security.php
3 years ago
discussions.php
274 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_admin = 'mwai/v1'; |
| 9 | private $namespace_ui = 'mwai-ui/v1'; |
| 10 | |
| 11 | public function __construct() { |
| 12 | global $wpdb; |
| 13 | global $mwai_core; |
| 14 | $this->core = $mwai_core; |
| 15 | $this->wpdb = $wpdb; |
| 16 | $this->table_chats = $wpdb->prefix . 'mwai_chats'; |
| 17 | |
| 18 | if ( $this->core->get_option( 'shortcode_chat_discussions' ) ) { |
| 19 | add_filter( 'mwai_chatbot_reply', [ $this, 'chatbot_reply' ], 10, 4 ); |
| 20 | add_action( 'rest_api_init', [ $this, 'rest_api_init' ] ); |
| 21 | } |
| 22 | } |
| 23 | |
| 24 | public function rest_api_init() { |
| 25 | |
| 26 | // Admin |
| 27 | register_rest_route( $this->namespace_admin, '/discussions/list', [ |
| 28 | 'methods' => 'POST', |
| 29 | 'callback' => [ $this, 'rest_discussions_list' ], |
| 30 | 'permission_callback' => [ $this->core, 'can_access_settings' ], |
| 31 | ] ); |
| 32 | register_rest_route( $this->namespace_admin, '/discussions/delete', [ |
| 33 | 'methods' => 'POST', |
| 34 | 'callback' => [ $this, 'rest_discussions_delete' ], |
| 35 | 'permission_callback' => [ $this->core, 'can_access_settings' ], |
| 36 | ] ); |
| 37 | |
| 38 | // UI |
| 39 | register_rest_route( $this->namespace_ui, '/discussions/list', [ |
| 40 | 'methods' => 'POST', |
| 41 | 'callback' => [ $this, 'rest_discussions_ui_list' ], |
| 42 | 'permission_callback' => '__return_true' |
| 43 | ] ); |
| 44 | } |
| 45 | |
| 46 | function rest_discussions_list( $request ) { |
| 47 | try { |
| 48 | $params = $request->get_json_params(); |
| 49 | $offset = $params['offset']; |
| 50 | $limit = $params['limit']; |
| 51 | $filters = $params['filters']; |
| 52 | $sort = $params['sort']; |
| 53 | $chats = $this->chats_query( [], $offset, $limit, $filters, $sort ); |
| 54 | return new WP_REST_Response([ 'success' => true, 'total' => $chats['total'], 'chats' => $chats['rows'] ], 200 ); |
| 55 | } |
| 56 | catch ( Exception $e ) { |
| 57 | return new WP_REST_Response([ 'success' => false, 'message' => $e->getMessage() ], 500 ); |
| 58 | } |
| 59 | } |
| 60 | |
| 61 | function rest_discussions_ui_list( $request ) { |
| 62 | try { |
| 63 | $params = $request->get_json_params(); |
| 64 | $offset = isset( $params['offset'] ) ? $params['offset'] : 0; |
| 65 | $limit = isset( $params['limit'] ) ? $params['limit'] : 10; |
| 66 | $botId = isset( $params['botId'] ) ? $params['botId'] : null; |
| 67 | |
| 68 | if ( is_null( $botId ) ) { |
| 69 | return new WP_REST_Response([ 'success' => false, 'message' => "Bot ID is required." ], 200 ); |
| 70 | } |
| 71 | |
| 72 | $userId = get_current_user_id(); |
| 73 | if ( !$userId ) { |
| 74 | return new WP_REST_Response([ 'success' => false, 'message' => "You need to be connected." ], 200 ); |
| 75 | } |
| 76 | $filters = [ |
| 77 | [ 'accessor' => 'user', 'value' => $userId ], |
| 78 | [ 'accessor' => 'botId', 'value' => $botId ], |
| 79 | ]; |
| 80 | $chats = $this->chats_query( [], $offset, $limit, $filters ); |
| 81 | return new WP_REST_Response([ 'success' => true, 'total' => $chats['total'], 'chats' => $chats['rows'] ], 200 ); |
| 82 | } |
| 83 | catch ( Exception $e ) { |
| 84 | return new WP_REST_Response([ 'success' => false, 'message' => $e->getMessage() ], 500 ); |
| 85 | } |
| 86 | } |
| 87 | |
| 88 | function rest_discussions_delete( $request ) { |
| 89 | try { |
| 90 | $params = $request->get_json_params(); |
| 91 | $chatsIds = $params['chatIds']; |
| 92 | if ( is_array( $chatsIds ) ) { |
| 93 | if ( count( $chatsIds ) === 0 ) { |
| 94 | $this->wpdb->query( "TRUNCATE TABLE $this->table_chats" ); |
| 95 | } |
| 96 | foreach( $chatsIds as $chatId ) { |
| 97 | $this->wpdb->delete( $this->table_chats, [ 'chatId' => $chatId ] ); |
| 98 | } |
| 99 | } |
| 100 | return new WP_REST_Response([ 'success' => true ], 200 ); |
| 101 | } |
| 102 | catch ( Exception $e ) { |
| 103 | return new WP_REST_Response([ 'success' => false, 'message' => $e->getMessage() ], 500 ); |
| 104 | } |
| 105 | } |
| 106 | |
| 107 | function chats_query( $chats = [], $offset = 0, $limit = null, $filters = null, $sort = null ) { |
| 108 | $this->check_db(); |
| 109 | $offset = !empty( $offset ) ? intval( $offset ) : 0; |
| 110 | $limit = !empty( $limit ) ? intval( $limit ) : 5; |
| 111 | $filters = !empty( $filters ) ? $filters : []; |
| 112 | $sort = !empty( $sort ) ? $sort : [ 'accessor' => 'updated', 'by' => 'desc' ]; |
| 113 | $query = "SELECT * FROM $this->table_chats"; |
| 114 | |
| 115 | // Filters |
| 116 | if ( is_array( $filters ) ) { |
| 117 | $where = array(); |
| 118 | foreach ( $filters as $filter ) { |
| 119 | if ( $filter['accessor'] === 'user' ) { |
| 120 | $value = esc_sql( $filter['value'] ); |
| 121 | if ( is_null( $value ) || $value === '' ) { |
| 122 | continue; |
| 123 | } |
| 124 | $isIP = filter_var( $value, FILTER_VALIDATE_IP ); |
| 125 | if ( $isIP ) { |
| 126 | $where[] = "ip = '{$value}'"; |
| 127 | } |
| 128 | else { |
| 129 | $where[] = "userId = '{$value}'"; |
| 130 | } |
| 131 | } |
| 132 | if ( $filter['accessor'] === 'botId' ) { |
| 133 | $value = esc_sql( $filter['value'] ); |
| 134 | if ( is_null( $value ) || $value === '' ) { |
| 135 | continue; |
| 136 | } |
| 137 | $where[] = "botId = '{$value}'"; |
| 138 | } |
| 139 | if ( $filter['accessor'] === 'preview' ) { |
| 140 | $value = $filter['value']; |
| 141 | if ( empty( $value ) ) { |
| 142 | continue; |
| 143 | } |
| 144 | $where[] = "messages LIKE '%{$value}%'"; |
| 145 | } |
| 146 | } |
| 147 | if ( count( $where ) > 0 ) { |
| 148 | $query .= " WHERE " . implode( " AND ", $where ); |
| 149 | } |
| 150 | } |
| 151 | |
| 152 | // Count based on this query |
| 153 | $chats['total'] = $this->wpdb->get_var( "SELECT COUNT(*) FROM ($query) AS t" ); |
| 154 | |
| 155 | // Order by |
| 156 | $query .= " ORDER BY " . esc_sql( $sort['accessor'] ) . " " . esc_sql( $sort['by'] ); |
| 157 | |
| 158 | // Limits |
| 159 | if ( $limit > 0 ) { |
| 160 | $query .= " LIMIT $offset, $limit"; |
| 161 | } |
| 162 | |
| 163 | $chats['rows'] = $this->wpdb->get_results( $query, ARRAY_A ); |
| 164 | return $chats; |
| 165 | } |
| 166 | |
| 167 | function chatbot_reply( $rawText, $query, $params, $extra ) { |
| 168 | global $mwai_core; |
| 169 | $userIp = $mwai_core->get_ip_address(); |
| 170 | $userId = $mwai_core->get_user_id(); |
| 171 | $botId = isset( $params['botId'] ) ? $params['botId'] : null; |
| 172 | $chatId = isset( $params['clientId'] ) ? $params['clientId'] : $query->session; |
| 173 | $newMessage = isset( $params['newMessage'] ) ? $params['newMessage'] : $query->prompt; |
| 174 | //$chatId = hash( 'sha256', $userIp . $userId . $clientChatId ); |
| 175 | $this->check_db(); |
| 176 | $chat = $this->wpdb->get_row( $this->wpdb->prepare( "SELECT * FROM $this->table_chats WHERE chatId = %s", $chatId ) ); |
| 177 | $extra = [ |
| 178 | 'embeddings' => isset( $extra['embeddings'] ) ? $extra['embeddings'] : null |
| 179 | ]; |
| 180 | if ( $chat ) { |
| 181 | $chat->messages = json_decode( $chat->messages ); |
| 182 | $chat->messages[] = [ |
| 183 | 'role' => 'user', |
| 184 | 'content' => $newMessage |
| 185 | ]; |
| 186 | $chat->messages[] = [ |
| 187 | 'role' => 'assistant', |
| 188 | 'content' => $rawText, |
| 189 | 'extra' => $extra |
| 190 | ]; |
| 191 | $chat->messages = json_encode( $chat->messages ); |
| 192 | $this->wpdb->update( $this->table_chats, [ |
| 193 | 'messages' => $chat->messages, |
| 194 | 'updated' => date( 'Y-m-d H:i:s' ) |
| 195 | ], [ 'id' => $chat->id ] ); |
| 196 | } |
| 197 | else { |
| 198 | $chat = [ |
| 199 | 'userId' => $userId, |
| 200 | 'ip' => $userIp, |
| 201 | 'messages' => json_encode( [ |
| 202 | [ |
| 203 | 'role' => 'user', |
| 204 | 'content' => $newMessage |
| 205 | ], |
| 206 | [ |
| 207 | 'role' => 'assistant', |
| 208 | 'content' => $rawText, |
| 209 | 'extra' => $extra |
| 210 | ] |
| 211 | ] ), |
| 212 | 'extra' => json_encode( [ |
| 213 | 'session' => $query->session, |
| 214 | 'model' => $query->model, |
| 215 | 'temperature' => $query->temperature, |
| 216 | 'context' => $query->context, |
| 217 | ] ), |
| 218 | 'botId' => $botId, |
| 219 | 'chatId' => $chatId, |
| 220 | 'created' => date( 'Y-m-d H:i:s' ), |
| 221 | 'updated' => date( 'Y-m-d H:i:s' ) |
| 222 | ]; |
| 223 | $this->wpdb->insert( $this->table_chats, $chat ); |
| 224 | } |
| 225 | return $rawText; |
| 226 | } |
| 227 | |
| 228 | function check_db() { |
| 229 | if ( $this->db_check ) { |
| 230 | return true; |
| 231 | } |
| 232 | $this->db_check = !( strtolower( |
| 233 | $this->wpdb->get_var( "SHOW TABLES LIKE '$this->table_chats'" ) ) != strtolower( $this->table_chats ) |
| 234 | ); |
| 235 | if ( !$this->db_check ) { |
| 236 | $this->create_db(); |
| 237 | $this->db_check = !( strtolower( |
| 238 | $this->wpdb->get_var( "SHOW TABLES LIKE '$this->table_chats'" ) ) != strtolower( $this->table_chats ) |
| 239 | ); |
| 240 | } |
| 241 | |
| 242 | // LATER: REMOVE THIS AFTER SEPTEMBER 2023 |
| 243 | // Make sure the column "userId" and "ip "exist in the $this->table_chats table |
| 244 | $this->db_check = $this->db_check && $this->wpdb->get_var( "SHOW COLUMNS FROM $this->table_chats LIKE 'userId'" ); |
| 245 | if ( !$this->db_check ) { |
| 246 | $this->wpdb->query( "ALTER TABLE $this->table_chats ADD COLUMN userId BIGINT(20) NULL" ); |
| 247 | $this->wpdb->query( "ALTER TABLE $this->table_chats ADD COLUMN ip VARCHAR(64) NULL" ); |
| 248 | $this->wpdb->query( "ALTER TABLE $this->table_chats ADD COLUMN botId VARCHAR(64) NULL" ); |
| 249 | $this->db_check = true; |
| 250 | } |
| 251 | |
| 252 | return $this->db_check; |
| 253 | } |
| 254 | |
| 255 | function create_db() { |
| 256 | $charset_collate = $this->wpdb->get_charset_collate(); |
| 257 | $sqlLogs = "CREATE TABLE $this->table_chats ( |
| 258 | id BIGINT(20) NOT NULL AUTO_INCREMENT, |
| 259 | userId BIGINT(20) NULL, |
| 260 | ip VARCHAR(64) NULL, |
| 261 | messages TEXT NOT NULL NULL, |
| 262 | extra TEXT NOT NULL NULL, |
| 263 | botId VARCHAR(64) NULL, |
| 264 | chatId VARCHAR(64) NOT NULL, |
| 265 | created DATETIME NOT NULL, |
| 266 | updated DATETIME NOT NULL, |
| 267 | PRIMARY KEY (id), |
| 268 | INDEX chatId (chatId) |
| 269 | ) $charset_collate;"; |
| 270 | require_once( ABSPATH . 'wp-admin/includes/upgrade.php' ); |
| 271 | dbDelta( $sqlLogs ); |
| 272 | } |
| 273 | |
| 274 | } |