/** * Retrieve a Google Access Token using the service account credentials stored in the admin option. * * @return mixed Access token string on success, false on failure. */ function get_google_access_token() { $credentials_json = get_option('decree_ai_google_credentials', ''); if (empty($credentials_json)) { return false; } $credentials = json_decode($credentials_json, true); if (!$credentials || !isset($credentials['client_email'], $credentials['private_key'], $credentials['token_uri'])) { return false; } $client_email = $credentials['client_email']; $private_key = $credentials['private_key']; $token_uri = $credentials['token_uri']; $now = time(); $header = ['alg' => 'RS256', 'typ' => 'JWT']; $payload = [ 'iss' => $client_email, 'scope' => 'https://www.googleapis.com/auth/spreadsheets', 'aud' => $token_uri, 'exp' => $now + 3600, 'iat' => $now ]; $base64UrlEncode = function ($data) { return rtrim(strtr(base64_encode(json_encode($data)), '+/', '-_'), '='); }; $header_encoded = $base64UrlEncode($header); $payload_encoded = $base64UrlEncode($payload); $signature_input = $header_encoded . '.' . $payload_encoded; $signature = ''; openssl_sign($signature_input, $signature, $private_key, 'SHA256'); $signature_encoded = rtrim(strtr(base64_encode($signature), '+/', '-_'), '='); $jwt = $signature_input . '.' . $signature_encoded; $post_fields = http_build_query([ 'grant_type' => 'urn:ietf:params:oauth:grant-type:jwt-bearer', 'assertion' => $jwt ]); $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, $token_uri); curl_setopt($ch, CURLOPT_POST, 1); curl_setopt($ch, CURLOPT_POSTFIELDS, $post_fields); curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); $response = curl_exec($ch); curl_close($ch); $responseData = json_decode($response, true); if (isset($responseData['access_token'])) { return $responseData['access_token']; } return false; } /** * Append query details (timestamp, current user ID, query, answer, Pinecone context) * to the designated Google Sheet. * * @param string $query The text submitted by the user. * @param string $answer The response from OpenAI. * @param string $pinecone_context The context returned from Pinecone. * @return mixed The result from the Sheets API call. */ function append_to_google_sheet($query, $answer, $pinecone_context) { $timestamp = date('c'); $user_id = get_current_user_id(); $values = [[ $timestamp, $user_id, $query, $answer, $pinecone_context ]]; $body = json_encode([ 'values' => $values ]); $spreadsheetId = '1QpNFetkT-_HlxuxZQrWtzUIIVQTwiFtaMIJJAjvsUwk'; $range = 'Sheet1!A:E'; $url = 'https://sheets.googleapis.com/v4/spreadsheets/' . $spreadsheetId . '/values/' . $range . ':append?valueInputOption=USER_ENTERED'; $accessToken = get_google_access_token(); if (!$accessToken) { return false; } $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, $url); curl_setopt($ch, CURLOPT_POST, 1); curl_setopt($ch, CURLOPT_HTTPHEADER, [ 'Content-Type: application/json', 'Authorization: Bearer ' . $accessToken ]); curl_setopt($ch, CURLOPT_POSTFIELDS, $body); curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); $result = curl_exec($ch); curl_close($ch); return $result; } /** * Append query details with vote information to the designated Google Sheet. * * @param string $query The text submitted by the user. * @param string $answer The response from OpenAI. * @param string $pinecone_context The context returned from Pinecone. * @param string $vote 'Thumbs Up' or 'Thumbs Down'. * @return mixed The result from the Sheets API call. */ function append_vote_to_google_sheet($query, $answer, $pinecone_context, $vote) { $timestamp = date('c'); $user_id = get_current_user_id(); $values = [[ $timestamp, $user_id, $query, $answer, $pinecone_context, $vote ]]; $body = json_encode([ 'values' => $values ]); $spreadsheetId = '1QpNFetkT-_HlxuxZQrWtzUIIVQTwiFtaMIJJAjvsUwk'; $range = 'Sheet1!A:F'; $url = 'https://sheets.googleapis.com/v4/spreadsheets/' . $spreadsheetId . '/values/' . $range . ':append?valueInputOption=USER_ENTERED'; $accessToken = get_google_access_token(); if (!$accessToken) { return false; } $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, $url); curl_setopt($ch, CURLOPT_POST, 1); curl_setopt($ch, CURLOPT_HTTPHEADER, [ 'Content-Type: application/json', 'Authorization: Bearer ' . $accessToken ]); curl_setopt($ch, CURLOPT_POSTFIELDS, $body); curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); $result = curl_exec($ch); curl_close($ch); return $result; }