sortBy('date') ->map(fn (BrentPrice $p) => $p->date->toDateString().': $'.$p->price_usd) ->implode("\n"); $model = config('services.gemini.model', 'gemini-2.0-flash'); $url = "https://generativelanguage.googleapis.com/v1beta/models/{$model}:generateContent"; try { $response = $this->apiLogger->send('gemini', 'POST', $url, fn () => Http::timeout(15) ->withQueryParameters(['key' => config('services.gemini.api_key')]) ->post($url, [ 'contents' => [[ 'parts' => [['text' => $this->prompt($priceList)]], ]], 'generationConfig' => [ 'responseMimeType' => 'application/json', 'responseSchema' => [ 'type' => 'OBJECT', 'properties' => [ 'direction' => [ 'type' => 'STRING', 'enum' => ['rising', 'falling', 'flat'], ], 'confidence' => ['type' => 'INTEGER'], 'reasoning' => ['type' => 'STRING'], ], 'required' => ['direction', 'confidence', 'reasoning'], ], ], ])); if (! $response->successful()) { Log::error('GeminiPredictionProvider: request failed', ['status' => $response->status()]); return null; } $text = $response->json('candidates.0.content.parts.0.text') ?? ''; $data = json_decode($text, true); if (! isset($data['direction'], $data['confidence'], $data['reasoning'])) { Log::error('GeminiPredictionProvider: unexpected response format', ['text' => $text]); return null; } $direction = TrendDirection::tryFrom($data['direction']); if ($direction === null) { Log::error('GeminiPredictionProvider: invalid direction', ['direction' => $data['direction']]); return null; } return new PricePrediction([ 'predicted_for' => now()->toDateString(), 'source' => PredictionSource::Llm, 'direction' => $direction, 'confidence' => min((int) $data['confidence'], self::LLM_MAX_CONFIDENCE), 'reasoning' => $data['reasoning'], 'generated_at' => now(), ]); } catch (Throwable $e) { Log::error('GeminiPredictionProvider: predict failed', ['error' => $e->getMessage()]); return null; } } private function prompt(string $priceList): string { return <<