merge: sync upstream/master preserving local changes
This commit is contained in:
@@ -0,0 +1,78 @@
|
||||
<?php
|
||||
|
||||
namespace App\Models;
|
||||
|
||||
use Illuminate\Database\Eloquent\Model;
|
||||
|
||||
class MailTemplate extends Model
|
||||
{
|
||||
protected $table = 'v2_mail_templates';
|
||||
|
||||
protected $fillable = ['name', 'subject', 'content'];
|
||||
|
||||
/**
|
||||
* Template definitions: required/optional vars and default content.
|
||||
*/
|
||||
public const TEMPLATES = [
|
||||
'verify' => [
|
||||
'label' => '邮箱验证码',
|
||||
'required_vars' => ['code'],
|
||||
'optional_vars' => ['name', 'url'],
|
||||
],
|
||||
'notify' => [
|
||||
'label' => '站点通知',
|
||||
'required_vars' => ['content'],
|
||||
'optional_vars' => ['name', 'url'],
|
||||
],
|
||||
'remindExpire' => [
|
||||
'label' => '到期提醒',
|
||||
'required_vars' => [],
|
||||
'optional_vars' => ['name', 'url'],
|
||||
],
|
||||
'remindTraffic' => [
|
||||
'label' => '流量提醒',
|
||||
'required_vars' => [],
|
||||
'optional_vars' => ['name', 'url'],
|
||||
],
|
||||
'mailLogin' => [
|
||||
'label' => '邮件登录',
|
||||
'required_vars' => ['link'],
|
||||
'optional_vars' => ['name', 'url'],
|
||||
],
|
||||
];
|
||||
|
||||
/**
|
||||
* Get template metadata (vars, label) for a given template name.
|
||||
*/
|
||||
public static function getMeta(string $name): ?array
|
||||
{
|
||||
return self::TEMPLATES[$name] ?? null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get all template names.
|
||||
*/
|
||||
public static function getNames(): array
|
||||
{
|
||||
return array_keys(self::TEMPLATES);
|
||||
}
|
||||
|
||||
/**
|
||||
* Validate that required placeholders are present in the content.
|
||||
*/
|
||||
public static function validateContent(string $name, string $content): array
|
||||
{
|
||||
$meta = self::getMeta($name);
|
||||
if (!$meta) {
|
||||
return ["Unknown template: {$name}"];
|
||||
}
|
||||
|
||||
$errors = [];
|
||||
foreach ($meta['required_vars'] as $var) {
|
||||
if (strpos($content, '{{' . $var . '}}') === false) {
|
||||
$errors[] = "缺少必要占位符: {{{$var}}}";
|
||||
}
|
||||
}
|
||||
return $errors;
|
||||
}
|
||||
}
|
||||
@@ -15,4 +15,8 @@ class Payment extends Model
|
||||
'config' => 'array',
|
||||
'enable' => 'boolean'
|
||||
];
|
||||
|
||||
protected $hidden = [
|
||||
'config',
|
||||
];
|
||||
}
|
||||
|
||||
@@ -17,6 +17,8 @@ class ServerMachineLoadHistory extends Model
|
||||
'mem_used' => 'integer',
|
||||
'disk_total' => 'integer',
|
||||
'disk_used' => 'integer',
|
||||
'net_in_speed' => 'float',
|
||||
'net_out_speed' => 'float',
|
||||
'recorded_at' => 'integer',
|
||||
'created_at' => 'timestamp',
|
||||
'updated_at' => 'timestamp',
|
||||
|
||||
@@ -39,6 +39,9 @@ class Ticket extends Model
|
||||
self::STATUS_CLOSED => '关闭'
|
||||
];
|
||||
|
||||
const REPLY_STATUS_WAITING = 0;
|
||||
const REPLY_STATUS_REPLIED = 1;
|
||||
|
||||
public function user(): BelongsTo
|
||||
{
|
||||
return $this->belongsTo(User::class, 'user_id', 'id');
|
||||
|
||||
@@ -29,6 +29,7 @@ class TicketMessage extends Model
|
||||
];
|
||||
|
||||
protected $appends = ['is_from_user', 'is_from_admin'];
|
||||
protected $hidden = ['ticket'];
|
||||
|
||||
/**
|
||||
* 关联的工单
|
||||
@@ -43,7 +44,7 @@ class TicketMessage extends Model
|
||||
*/
|
||||
public function getIsFromUserAttribute(): bool
|
||||
{
|
||||
return $this->ticket->user_id === $this->user_id;
|
||||
return $this->ticket && $this->ticket->user_id === $this->user_id;
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -51,6 +52,6 @@ class TicketMessage extends Model
|
||||
*/
|
||||
public function getIsFromAdminAttribute(): bool
|
||||
{
|
||||
return $this->ticket->user_id !== $this->user_id;
|
||||
return $this->ticket && $this->ticket->user_id !== $this->user_id;
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user