- AI 설정과 스토리지 설정을 탭으로 구분 - GCS 버킷 이름, 서비스 계정 (JSON 직접입력/파일경로) 설정 가능 - GCS 연결 테스트 기능 추가 - GoogleCloudStorageService가 DB 설정 우선 사용 (fallback: 레거시 파일) - AiConfig 모델에 gcs provider 및 관련 메서드 추가 Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
242 lines
5.3 KiB
PHP
242 lines
5.3 KiB
PHP
<?php
|
|
|
|
namespace App\Models\System;
|
|
|
|
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
|
use Illuminate\Database\Eloquent\Model;
|
|
use Illuminate\Database\Eloquent\SoftDeletes;
|
|
|
|
/**
|
|
* AI API 설정 모델
|
|
*
|
|
* @property int $id
|
|
* @property string $name
|
|
* @property string $provider
|
|
* @property string $api_key
|
|
* @property string $model
|
|
* @property string|null $base_url
|
|
* @property string|null $description
|
|
* @property bool $is_active
|
|
* @property array|null $options
|
|
* @property \Carbon\Carbon $created_at
|
|
* @property \Carbon\Carbon $updated_at
|
|
* @property \Carbon\Carbon|null $deleted_at
|
|
*/
|
|
class AiConfig extends Model
|
|
{
|
|
use HasFactory, SoftDeletes;
|
|
|
|
protected $table = 'ai_configs';
|
|
|
|
protected $fillable = [
|
|
'name',
|
|
'provider',
|
|
'api_key',
|
|
'model',
|
|
'base_url',
|
|
'description',
|
|
'is_active',
|
|
'options',
|
|
];
|
|
|
|
protected $casts = [
|
|
'is_active' => 'boolean',
|
|
'options' => 'array',
|
|
];
|
|
|
|
/**
|
|
* Provider별 기본 Base URL
|
|
*/
|
|
public const DEFAULT_BASE_URLS = [
|
|
'gemini' => 'https://generativelanguage.googleapis.com/v1beta',
|
|
'claude' => 'https://api.anthropic.com/v1',
|
|
'openai' => 'https://api.openai.com/v1',
|
|
'gcs' => 'https://storage.googleapis.com',
|
|
];
|
|
|
|
/**
|
|
* Provider별 기본 모델
|
|
*/
|
|
public const DEFAULT_MODELS = [
|
|
'gemini' => 'gemini-2.0-flash',
|
|
'claude' => 'claude-sonnet-4-20250514',
|
|
'openai' => 'gpt-4o',
|
|
'gcs' => '-',
|
|
];
|
|
|
|
/**
|
|
* AI Provider 목록 (GCS 제외)
|
|
*/
|
|
public const AI_PROVIDERS = ['gemini', 'claude', 'openai'];
|
|
|
|
/**
|
|
* 스토리지 Provider 목록
|
|
*/
|
|
public const STORAGE_PROVIDERS = ['gcs'];
|
|
|
|
/**
|
|
* 활성화된 Gemini 설정 조회
|
|
*/
|
|
public static function getActiveGemini(): ?self
|
|
{
|
|
return self::where('provider', 'gemini')
|
|
->where('is_active', true)
|
|
->first();
|
|
}
|
|
|
|
/**
|
|
* 활성화된 Claude 설정 조회
|
|
*/
|
|
public static function getActiveClaude(): ?self
|
|
{
|
|
return self::where('provider', 'claude')
|
|
->where('is_active', true)
|
|
->first();
|
|
}
|
|
|
|
/**
|
|
* Provider별 활성 설정 조회
|
|
*/
|
|
public static function getActive(string $provider): ?self
|
|
{
|
|
return self::where('provider', $provider)
|
|
->where('is_active', true)
|
|
->first();
|
|
}
|
|
|
|
/**
|
|
* Base URL 가져오기 (설정 또는 기본값)
|
|
*/
|
|
public function getBaseUrlAttribute($value): string
|
|
{
|
|
return $value ?? self::DEFAULT_BASE_URLS[$this->provider] ?? '';
|
|
}
|
|
|
|
/**
|
|
* Provider 라벨
|
|
*/
|
|
public function getProviderLabelAttribute(): string
|
|
{
|
|
return match ($this->provider) {
|
|
'gemini' => 'Google Gemini',
|
|
'claude' => 'Anthropic Claude',
|
|
'openai' => 'OpenAI',
|
|
'gcs' => 'Google Cloud Storage',
|
|
default => $this->provider,
|
|
};
|
|
}
|
|
|
|
/**
|
|
* 활성화된 GCS 설정 조회
|
|
*/
|
|
public static function getActiveGcs(): ?self
|
|
{
|
|
return self::where('provider', 'gcs')
|
|
->where('is_active', true)
|
|
->first();
|
|
}
|
|
|
|
/**
|
|
* GCS 버킷 이름
|
|
*/
|
|
public function getBucketName(): ?string
|
|
{
|
|
return $this->options['bucket_name'] ?? null;
|
|
}
|
|
|
|
/**
|
|
* GCS 서비스 계정 JSON (직접 저장된 경우)
|
|
*/
|
|
public function getServiceAccountJson(): ?array
|
|
{
|
|
return $this->options['service_account_json'] ?? null;
|
|
}
|
|
|
|
/**
|
|
* GCS 설정인지 확인
|
|
*/
|
|
public function isGcs(): bool
|
|
{
|
|
return $this->provider === 'gcs';
|
|
}
|
|
|
|
/**
|
|
* AI 설정인지 확인
|
|
*/
|
|
public function isAi(): bool
|
|
{
|
|
return in_array($this->provider, self::AI_PROVIDERS);
|
|
}
|
|
|
|
/**
|
|
* 상태 라벨
|
|
*/
|
|
public function getStatusLabelAttribute(): string
|
|
{
|
|
return $this->is_active ? '활성' : '비활성';
|
|
}
|
|
|
|
/**
|
|
* 상태 색상 (Tailwind)
|
|
*/
|
|
public function getStatusColorAttribute(): string
|
|
{
|
|
return $this->is_active ? 'green' : 'gray';
|
|
}
|
|
|
|
/**
|
|
* 마스킹된 API 키 (앞 8자리만 표시)
|
|
*/
|
|
public function getMaskedApiKeyAttribute(): string
|
|
{
|
|
if (strlen($this->api_key) <= 8) {
|
|
return $this->api_key;
|
|
}
|
|
|
|
return substr($this->api_key, 0, 8) . str_repeat('*', 8) . '...';
|
|
}
|
|
|
|
/**
|
|
* Vertex AI 사용 여부
|
|
*/
|
|
public function isVertexAi(): bool
|
|
{
|
|
return ($this->options['auth_type'] ?? 'api_key') === 'vertex_ai';
|
|
}
|
|
|
|
/**
|
|
* Vertex AI 프로젝트 ID
|
|
*/
|
|
public function getProjectId(): ?string
|
|
{
|
|
return $this->options['project_id'] ?? null;
|
|
}
|
|
|
|
/**
|
|
* Vertex AI 리전
|
|
*/
|
|
public function getRegion(): string
|
|
{
|
|
return $this->options['region'] ?? 'us-central1';
|
|
}
|
|
|
|
/**
|
|
* 서비스 계정 파일 경로
|
|
*/
|
|
public function getServiceAccountPath(): ?string
|
|
{
|
|
return $this->options['service_account_path'] ?? null;
|
|
}
|
|
|
|
/**
|
|
* 인증 방식 라벨
|
|
*/
|
|
public function getAuthTypeLabelAttribute(): string
|
|
{
|
|
if ($this->isVertexAi()) {
|
|
return 'Vertex AI (서비스 계정)';
|
|
}
|
|
return 'API 키';
|
|
}
|
|
}
|