初始化仓库及v1.0.0提交
This commit is contained in:
33
.gitignore
vendored
Normal file
33
.gitignore
vendored
Normal file
@@ -0,0 +1,33 @@
|
||||
# Dependencies
|
||||
node_modules/
|
||||
vendor/
|
||||
|
||||
# Environment variables
|
||||
.env
|
||||
.env.local
|
||||
|
||||
# Config files (managed by Web installer)
|
||||
config/db-config.json
|
||||
config/app-config.json
|
||||
|
||||
# Uploads (keep .gitkeep)
|
||||
uploads/*
|
||||
!uploads/.gitkeep
|
||||
|
||||
# IDE and Editor
|
||||
.vscode/
|
||||
.idea/
|
||||
*.swp
|
||||
*.swo
|
||||
*~
|
||||
|
||||
# Logs
|
||||
logs/
|
||||
*.log
|
||||
|
||||
# OS generated files
|
||||
.DS_Store
|
||||
Thumbs.db
|
||||
|
||||
# Plan workflow
|
||||
.cospec/
|
||||
122
INSTALL.md
Normal file
122
INSTALL.md
Normal file
@@ -0,0 +1,122 @@
|
||||
# 安装指南
|
||||
|
||||
## 环境要求
|
||||
|
||||
| 组件 | 最低版本 | 说明 |
|
||||
|------|----------|------|
|
||||
| PHP | 8.0+ | 需启用 PHP-FPM |
|
||||
| MySQL | 5.7+ | 或 MariaDB 10.3+ |
|
||||
| Nginx | 1.18+ | 或其他支持 PHP-FPM 的 Web 服务器 |
|
||||
| Composer | 2.0+ | PHP 依赖管理工具 |
|
||||
|
||||
### PHP 扩展要求
|
||||
|
||||
- PDO(含 PDO MySQL 驱动)
|
||||
- cURL
|
||||
- JSON
|
||||
- mbstring
|
||||
- OpenSSL
|
||||
|
||||
## 安装方式
|
||||
|
||||
### 方式一:宝塔面板安装(推荐)
|
||||
|
||||
宝塔面板提供图形化界面,最适合快速部署。
|
||||
|
||||
#### 1. 安装宝塔面板环境
|
||||
|
||||
在宝塔面板中安装以下组件:
|
||||
- Nginx(任意版本)
|
||||
- PHP 8.0(注意选择正确版本)
|
||||
- MySQL 5.7 或 8.0
|
||||
|
||||
#### 2. 创建网站
|
||||
|
||||
1. 登录宝塔面板
|
||||
2. 点击"网站" → "添加站点"
|
||||
3. 填写域名,PHP 版本选择 **PHP 8.0**
|
||||
4. 数据库选择"不创建"(安装向导中会创建)
|
||||
|
||||
#### 3. 上传代码
|
||||
|
||||
将项目代码上传到网站目录(例如 `/www/wwwroot/your-domain.com/`),或使用 Git 克隆。
|
||||
|
||||
#### 4. 安装 PHP 依赖
|
||||
|
||||
在宝塔终端中执行:
|
||||
```bash
|
||||
cd /www/wwwroot/your-domain.com
|
||||
composer install
|
||||
```
|
||||
|
||||
#### 5. 设置目录权限
|
||||
|
||||
```bash
|
||||
chmod 755 uploads/
|
||||
chmod 755 config/
|
||||
```
|
||||
|
||||
#### 6. 配置 Nginx
|
||||
|
||||
1. 在宝塔面板中找到你的站点 → 设置 → 配置文件
|
||||
2. 将网站根目录修改为 `/www/wwwroot/your-domain.com/public`
|
||||
3. 参考 `docs/baota-nginx-snippet.conf` 中的配置片段,添加到 server 块中
|
||||
4. 保存配置
|
||||
|
||||
#### 7. 运行安装向导
|
||||
|
||||
访问 `http://your-domain.com/install.php`,按照向导步骤:
|
||||
1. **环境检查** — 系统自动检查 PHP 版本和扩展
|
||||
2. **数据库配置** — 填写 MySQL 连接信息
|
||||
3. **应用配置** — JWT 密钥(可自动生成)
|
||||
4. **管理员账户** — 设置管理员用户名和密码
|
||||
5. **AI 供应商** — 配置至少一个 AI 服务供应商
|
||||
|
||||
### 方式二:手动安装
|
||||
|
||||
#### 1. 安装环境
|
||||
|
||||
```bash
|
||||
# Ubuntu/Debian
|
||||
sudo apt update
|
||||
sudo apt install php8.0-fpm php8.0-mysql php8.0-curl php8.0-mbstring php8.0-xml
|
||||
sudo apt install mysql-server nginx composer
|
||||
```
|
||||
|
||||
#### 2. 部署代码
|
||||
|
||||
```bash
|
||||
cd /var/www
|
||||
git clone <repository-url> ai-chat
|
||||
cd ai-chat
|
||||
composer install
|
||||
```
|
||||
|
||||
#### 3. 设置权限
|
||||
|
||||
```bash
|
||||
chown -R www-data:www-data /var/www/ai-chat
|
||||
chmod 755 uploads/ config/
|
||||
```
|
||||
|
||||
#### 4. 配置 Nginx
|
||||
|
||||
参考 `docs/nginx.conf` 配置 Nginx 站点。
|
||||
|
||||
#### 5. 运行安装向导
|
||||
|
||||
访问 `http://your-domain.com/install.php` 完成安装。
|
||||
|
||||
## 常见问题
|
||||
|
||||
### Q: 安装页面空白?
|
||||
检查 PHP 版本是否 >= 8.0,以及 PHP 扩展是否已安装。
|
||||
|
||||
### Q: 500 错误?
|
||||
检查 Nginx 错误日志:`tail -f /var/log/nginx/error.log`
|
||||
|
||||
### Q: 数据库连接失败?
|
||||
确认 MySQL 服务正在运行,且数据库账号密码正确。
|
||||
|
||||
### Q: SSE 流式响应不工作?
|
||||
确认 Nginx 配置中已禁用缓冲(参考 baota-nginx-snippet.conf)。
|
||||
671
LICENSE
Normal file
671
LICENSE
Normal file
@@ -0,0 +1,671 @@
|
||||
Copyright (C) AI Chat Contributors. All rights reserved.
|
||||
This software is licensed under the GNU Affero General Public License v3.0.
|
||||
|
||||
GNU AFFERO GENERAL PUBLIC LICENSE
|
||||
Version 3, 19 November 2007
|
||||
|
||||
Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
|
||||
Everyone is permitted to copy and distribute verbatim copies
|
||||
of this license document, but changing it is not allowed.
|
||||
|
||||
Preamble
|
||||
|
||||
The GNU Affero General Public License is a free, copyleft license
|
||||
for software and other kinds of works, which are specifically designed
|
||||
to ensure cooperation with the community in the case of network server
|
||||
software.
|
||||
|
||||
The licenses for most software and other practical works are designed
|
||||
to take away your freedom to share and change the works. By contrast,
|
||||
our General Public Licenses are intended to guarantee your freedom to
|
||||
share and change all versions of a program--to make sure it remains free
|
||||
software for all its users. When we speak of free software, we are
|
||||
referring to freedom, not price. Our General Public Licenses are
|
||||
designed to make sure that you have the freedom to distribute copies of
|
||||
free software (and charge for them if you wish), that you receive source
|
||||
code or can get it if you want it, that you can change the software or
|
||||
use pieces of it in new free programs, and that you know you can do
|
||||
these things.
|
||||
|
||||
Developers that use our General Public Licenses protect your rights
|
||||
with two steps: (1) assert copyright on the software, and (2) offer
|
||||
you this License which gives you legal permission to copy, distribute
|
||||
and/or modify the software.
|
||||
|
||||
A secondary benefit of defending all users' freedom is that
|
||||
improvements made in alternate versions of a program, if they
|
||||
receive widespread use, become available for other developers to
|
||||
reuse or incorporate. Many developers of free software are heartened
|
||||
and encouraged by the resulting cooperation. However, in the case of
|
||||
software used on network servers, this result may fail to come about.
|
||||
The GNU General Public License permits making a modified version and
|
||||
letting the public access it on a server without ever releasing its
|
||||
source code to the public.
|
||||
|
||||
The GNU Affero General Public License is designed specifically to
|
||||
ensure that, in such cases, the modified source code becomes available
|
||||
to the community. It requires the operator of a network server to
|
||||
provide the source code of the modified version running there to the
|
||||
users of that server. Therefore, public use of a modified version, on
|
||||
a publicly accessible server, gives the public access to the source
|
||||
code of the modified version.
|
||||
|
||||
An older license, called the Affero General Public License and
|
||||
published by Affero, was designed to accomplish similar goals. This is
|
||||
a different license, not a version of the Affero GPL, but Affero has
|
||||
released a new version of the Affero GPL which permits relicensing under
|
||||
this license.
|
||||
|
||||
The precise terms and conditions for copying, distribution and
|
||||
modification follow.
|
||||
|
||||
TERMS AND CONDITIONS
|
||||
|
||||
0. Definitions.
|
||||
|
||||
"This License" refers to version 3 of the GNU Affero General Public License.
|
||||
|
||||
"Copyright" also means copyright-like laws that apply to other kinds of
|
||||
works, in any medium.
|
||||
|
||||
"The Program" refers to any copyrightable work licensed under this
|
||||
License. Each licensee is addressed as "you". "Licensees" and
|
||||
"recipients" may be individuals or organizations.
|
||||
|
||||
To "modify" a work means to copy from or adapt all or part of the work
|
||||
in a fashion requiring copyright permission, other than the making of an
|
||||
exact copy. The resulting work is called a "modified version" of the
|
||||
earlier work or a work "based on" the earlier work.
|
||||
|
||||
A "covered work" means either the unmodified Program or a work based
|
||||
on the Program.
|
||||
|
||||
To "propagate" a work means to do anything with it that, without
|
||||
permission, would make you directly or secondarily liable for
|
||||
infringement under applicable copyright law, except executing it on a
|
||||
computer or modifying a private copy. Propagation includes copying,
|
||||
distribution (with or without modification), making available to the
|
||||
public, and in some countries other activities as well.
|
||||
|
||||
To "convey" a work means any kind of propagation that enables other
|
||||
parties to make or receive copies. Mere interaction with a user through
|
||||
a computer network, with no transfer of a copy, is not conveying.
|
||||
|
||||
An interactive user interface displays "Appropriate Legal Notices"
|
||||
to the extent that it includes a convenient and prominently visible
|
||||
feature that (1) displays an appropriate copyright notice, and (2)
|
||||
tells the user that there is no warranty for the work (except to the
|
||||
extent that warranties are provided), that licensees may convey the
|
||||
work under this License, and how to view a copy of this License. If
|
||||
the interface presents a list of user commands or options, such as a
|
||||
menu, a prominent item in the list meets this criterion.
|
||||
|
||||
1. Source Code.
|
||||
|
||||
The "source code" for a work means the preferred form of the work
|
||||
for making modifications to it. "Object code" means any non-source
|
||||
form of a work.
|
||||
|
||||
A "Standard Interface" means an interface that either is an official
|
||||
standard defined by a recognized standards body, or, in the case of
|
||||
interfaces specified for a particular programming language, one that
|
||||
is widely used among developers working in that language.
|
||||
|
||||
The "System Libraries" of an executable work include anything, other
|
||||
than the work as a whole, that (a) is included in the normal form of
|
||||
packaging a Major Component, but which is not part of that Major
|
||||
Component, and (b) serves only to enable use of the work with that
|
||||
Major Component, or to implement a Standard Interface for which an
|
||||
implementation is available to the public in source code form. A
|
||||
"Major Component", in this context, means a major essential component
|
||||
(kernel, window system, and so on) of the specific operating system
|
||||
(if any) on which the executable work runs, or a compiler used to
|
||||
produce the work, or an object code interpreter used to run it.
|
||||
|
||||
The "Corresponding Source" for a work in object code form means all
|
||||
the source code needed to generate, install, and (for an executable
|
||||
work) run the object code and to modify the work, including scripts to
|
||||
control those activities. However, it does not include the work's
|
||||
System Libraries, or general-purpose tools or generally available free
|
||||
programs which are used unmodified in performing those activities but
|
||||
which are not part of the work. For example, Corresponding Source
|
||||
includes interface definition files associated with source files for
|
||||
the work, and the source code for shared libraries and dynamically
|
||||
linked subprograms that the work is specifically designed to require,
|
||||
such as by intimate data communication or control flow between those
|
||||
subprograms and other parts of the work.
|
||||
|
||||
The Corresponding Source need not include anything that users
|
||||
can regenerate automatically from other parts of the Corresponding
|
||||
Source.
|
||||
|
||||
The Corresponding Source is intended to be the minimal source code
|
||||
that needs to be available to allow anyone to generate, install,
|
||||
run, maintain, and modify the work. It does not include general-purpose
|
||||
tools that are generally available and are not part of the work.
|
||||
|
||||
For example, Corresponding Source includes scripts used to control
|
||||
compilation and installation of the executable. It does not include
|
||||
the source code for the compiler itself, or the source code for any
|
||||
standard libraries that the work uses.
|
||||
|
||||
2. Basic Permissions.
|
||||
|
||||
All rights granted under this License are granted for the term of
|
||||
copyright on the Program, and are irrevocable provided the stated
|
||||
conditions are met. This License explicitly affirms your unlimited
|
||||
permission to run the unmodified Program. The output from running a
|
||||
covered work is covered by this License only if the output, given its
|
||||
content, constitutes a covered work. This License acknowledges your
|
||||
rights of fair use or other equivalent, as provided by copyright law.
|
||||
|
||||
You may make, run and propagate covered works that you do not
|
||||
convey, without conditions so long as your license otherwise remains
|
||||
in force. You may convey covered works to others for the sole purpose
|
||||
of having them make modifications exclusively for you, or provide you
|
||||
with facilities for running those works, provided that you comply with
|
||||
the terms of this License in conveying all material for which you do
|
||||
not control copyright. Those thus making or running the covered works
|
||||
for you must do so exclusively on your behalf, under your direction
|
||||
and control, on terms that prohibit them from making any copies of
|
||||
your copyrighted material outside their relationship with you.
|
||||
|
||||
Conveying under any other circumstances is permitted solely under
|
||||
the conditions stated below. Sublicensing is not allowed; section 10
|
||||
makes it unnecessary.
|
||||
|
||||
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
||||
|
||||
No covered work shall be deemed part of an effective technological
|
||||
measure under any applicable law fulfilling obligations under article
|
||||
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
||||
similar laws prohibiting or restricting circumvention of such
|
||||
measures.
|
||||
|
||||
When you convey a covered work, you waive any legal power to forbid
|
||||
circumvention of technological measures to the extent such circumvention
|
||||
is effected by exercising rights under this License with respect to
|
||||
the covered work, and you disclaim any intention to limit operation or
|
||||
modification of the work as a means of enforcing, against the work's
|
||||
users, your or third parties' legal rights to forbid circumvention of
|
||||
technological measures.
|
||||
|
||||
4. Conveying Verbatim Copies.
|
||||
|
||||
You may convey verbatim copies of the Program's source code as you
|
||||
receive it, in any medium, provided that you conspicuously and
|
||||
appropriately publish on each copy an appropriate copyright notice;
|
||||
keep intact all notices stating that this License and any
|
||||
non-permissive terms added in accord with section 7 apply to the code;
|
||||
keep intact all notices of the absence of any warranty; and give all
|
||||
recipients a copy of this License along with the Program.
|
||||
|
||||
You may charge any price or no price for each copy that you convey,
|
||||
and you may offer support or warranty protection for a fee.
|
||||
|
||||
5. Conveying Modified Source Versions.
|
||||
|
||||
You may convey a work based on the Program, or the modifications to
|
||||
produce it from the Program, in the form of source code under the
|
||||
terms of section 4, provided that you also meet all of these conditions:
|
||||
|
||||
a) The work must carry prominent notices stating that you modified
|
||||
it, and giving a relevant date.
|
||||
|
||||
b) The work must carry prominent notices stating that it is
|
||||
released under this License and any conditions added under section
|
||||
7. This requirement modifies the requirement in section 4 to
|
||||
"keep intact all notices".
|
||||
|
||||
c) You must license the entire work, as a whole, under this
|
||||
License to anyone who comes into possession of a copy. This
|
||||
License will therefore apply, along with any applicable section 7
|
||||
additional terms, to the whole of the work, and all its parts,
|
||||
regardless of how they are packaged. This License gives no
|
||||
permission to license the work in any other way, but it does not
|
||||
invalidate such permission if you have separately received it.
|
||||
|
||||
d) If the work has interactive user interfaces, each must display
|
||||
Appropriate Legal Notices; however, if the Program has interactive
|
||||
interfaces that do not display Appropriate Legal Notices, your
|
||||
work need not make them do so.
|
||||
|
||||
A compilation of a covered work with other separate and independent
|
||||
works, which are not by their nature extensions of the covered work,
|
||||
and which are not combined with it such as to form a larger program,
|
||||
in or on a volume of a storage or distribution medium, is called an
|
||||
"aggregate" if the compilation and its resulting copyright are not
|
||||
used to limit the access or legal rights of the compilation's users
|
||||
beyond what the individual works permit. Inclusion of a covered work
|
||||
in an aggregate does not cause this License to apply to the other
|
||||
parts of the aggregate.
|
||||
|
||||
6. Conveying Non-Source Forms.
|
||||
|
||||
You may convey a covered work in object code form under the terms
|
||||
of sections 4 and 5, provided that you also convey the
|
||||
machine-readable Corresponding Source under the terms of this License,
|
||||
in one of these ways:
|
||||
|
||||
a) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by the
|
||||
Corresponding Source fixed on a durable physical medium
|
||||
customarily used for software interchange.
|
||||
|
||||
b) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by a
|
||||
written offer, valid for at least three years and valid for as
|
||||
long as you offer spare parts or customer support for that product
|
||||
model, to give anyone who possesses the object code either (1) a
|
||||
copy of the Corresponding Source for all the software in the
|
||||
product that is covered by this License, on a durable physical
|
||||
medium customarily used for software interchange, for a price no
|
||||
more than your reasonable cost of physically performing this
|
||||
conveying of source, or (2) access to copy the Corresponding
|
||||
Source from a network server at no charge.
|
||||
|
||||
c) Convey individual copies of the object code with a copy of the
|
||||
written offer to provide the Corresponding Source. This
|
||||
alternative is allowed only occasionally and noncommercially, and
|
||||
only if you received the object code with such an offer, in accord
|
||||
with subsection 6b.
|
||||
|
||||
d) Convey the object code by offering access from a designated
|
||||
place (gratis or for a charge), and offer equivalent access to the
|
||||
Corresponding Source in the same way through the same place at no
|
||||
further charge. You need not require recipients to copy the
|
||||
Corresponding Source along with the object code. If the place to
|
||||
copy the object code is a network server, the Corresponding Source
|
||||
may be on a different server (operated by you or a third party)
|
||||
that supports equivalent copying facilities, provided you maintain
|
||||
clear directions next to the object code saying where to find the
|
||||
Corresponding Source. Regardless of what server hosts the
|
||||
Corresponding Source, you remain obligated to ensure that it is
|
||||
available for as long as needed to satisfy these requirements.
|
||||
|
||||
e) Convey the object code using peer-to-peer transmission, provided
|
||||
you inform other peers where the object code and Corresponding
|
||||
Source of the work are being offered to the general public at no
|
||||
charge under subsection 6d.
|
||||
|
||||
A separable portion of the object code, whose source code is excluded
|
||||
from the Corresponding Source as a System Library, need not be
|
||||
included in conveying the object code work.
|
||||
|
||||
A "User Product" is either (1) a "consumer product", which means any
|
||||
tangible personal property which is normally used for personal, family,
|
||||
or household purposes, or (2) anything designed or sold for incorporation
|
||||
into a dwelling. In determining whether a product is a consumer product,
|
||||
doubtful cases shall be resolved in favor of coverage. For a particular
|
||||
product received by a particular user, "normally used" refers to a
|
||||
typical or common use of that class of product, regardless of the status
|
||||
of the particular user or of the way in which the particular user
|
||||
actually uses, or expects or is expected to use, the product. A product
|
||||
is a consumer product regardless of whether the product has substantial
|
||||
commercial, industrial or non-consumer uses, unless such uses represent
|
||||
the only significant mode of use of the product.
|
||||
|
||||
"Installation Information" for a User Product means any methods,
|
||||
procedures, authorization keys, or other information required to install
|
||||
and execute modified versions of a covered work in that User Product from
|
||||
a modified version of its Corresponding Source. The information must
|
||||
suffice to ensure that the continued functioning of the modified object
|
||||
code is in no case prevented or interfered with solely because
|
||||
modification has been made.
|
||||
|
||||
If you convey an object code work under this section in, or with, or
|
||||
specifically for use in, a User Product, and the conveying occurs as
|
||||
part of a transaction in which the right of possession and use of the
|
||||
User Product is transferred to the recipient in perpetuity or for a
|
||||
fixed term (regardless of how the transaction is characterized), the
|
||||
Corresponding Source conveyed under this section must be accompanied
|
||||
by the Installation Information. But this requirement does not apply
|
||||
if neither you nor any third party retains the ability to install
|
||||
modified object code on the User Product (for example, the work has
|
||||
been installed in ROM).
|
||||
|
||||
The requirement to provide Installation Information does not include a
|
||||
requirement to continue to provide support service, warranty, or updates
|
||||
for a work that has been modified or installed by the recipient, or for
|
||||
the User Product in which it has been modified or installed. Access to a
|
||||
network may be denied when the modification itself materially and
|
||||
adversely affects the operation of the network or violates the rules and
|
||||
protocols for communication across the network.
|
||||
|
||||
Corresponding Source conveyed, and Installation Information provided,
|
||||
in accord with this section must be in a format that is publicly
|
||||
documented (and with an implementation available to the public in
|
||||
source code form), and must require no special password or key for
|
||||
unpacking, reading or copying.
|
||||
|
||||
7. Additional Terms.
|
||||
|
||||
"Additional permissions" are terms that supplement the terms of this
|
||||
License by making exceptions from one or more of its conditions.
|
||||
Additional permissions that are applicable to the entire Program shall
|
||||
be treated as though they were included in this License, to the extent
|
||||
that they are valid under applicable law. If additional permissions
|
||||
apply only to part of the Program, that part may be used separately
|
||||
under those permissions, but the entire Program remains governed by
|
||||
this License without regard to the additional permissions.
|
||||
|
||||
When you convey a copy of a covered work, you may at your option
|
||||
remove any additional permissions from that copy, or from any part of
|
||||
it. (Additional permissions may be written to require their own
|
||||
removal in certain cases when you modify the work.) You may place
|
||||
additional permissions on material, added by you to a covered work,
|
||||
for which you have or can give appropriate copyright permission.
|
||||
|
||||
Notwithstanding any other provision of this License, for material you
|
||||
add to a covered work, you may (if authorized by the copyright holders of
|
||||
that material) supplement the terms of this License with terms:
|
||||
|
||||
a) Disclaiming warranty or limiting liability differently from the
|
||||
terms of sections 15 and 16 of this License; or
|
||||
|
||||
b) Requiring preservation of specified reasonable legal notices or
|
||||
author attributions in that material or in the Appropriate Legal
|
||||
Notices displayed by works containing it; or
|
||||
|
||||
c) Prohibiting misrepresentation of the origin of that material, or
|
||||
requiring that modified versions of such material be marked in
|
||||
reasonable ways as different from the original version; or
|
||||
|
||||
d) Limiting the use for publicity purposes of names of licensors or
|
||||
authors of the material; or
|
||||
|
||||
e) Declining to grant rights under trademark law for use of some
|
||||
trade names, trademarks, or service marks; or
|
||||
|
||||
f) Requiring indemnification of licensors and authors of that
|
||||
material by anyone who conveys the material (or modified versions of
|
||||
it) with contractual assumptions of liability to the recipient, for
|
||||
any liability that these contractual assumptions directly impose on
|
||||
those licensors and authors.
|
||||
|
||||
All other non-permissive additional terms are considered "further
|
||||
restrictions" within the meaning of section 10. If the Program as you
|
||||
received it, or any part of it, contains a notice stating that it is
|
||||
governed by this License along with a term that is a further
|
||||
restriction, you may remove that term. If a license document contains
|
||||
a further restriction but permits relicensing or conveying under this
|
||||
License, you may add to a covered work material governed by the terms
|
||||
of that license document, provided that the further restriction does
|
||||
not survive such relicensing or conveying.
|
||||
|
||||
If you add terms to a covered work in accord with this section, you
|
||||
must place, in the relevant source files, a statement of the
|
||||
additional terms that apply to those files, or a notice indicating
|
||||
where to find the applicable terms.
|
||||
|
||||
Additional terms, permissive or non-permissive, may be stated in the
|
||||
form of a separately written license, or stated as exceptions;
|
||||
the above requirements apply either way.
|
||||
|
||||
8. Termination.
|
||||
|
||||
You may not propagate or modify a covered work except as expressly
|
||||
provided under this License. Any attempt otherwise to propagate or
|
||||
modify it is void, and will automatically terminate your rights under
|
||||
this License (including any patent licenses granted under the third
|
||||
paragraph of section 11).
|
||||
|
||||
However, if you cease all violation of this License, then your
|
||||
license from a particular copyright holder is reinstated (a)
|
||||
provisionally, unless and until the copyright holder explicitly and
|
||||
finally terminates your license, and (b) permanently, if the copyright
|
||||
holder fails to notify you of the violation by some reasonable means
|
||||
prior to 60 days after the cessation.
|
||||
|
||||
Moreover, your license from a particular copyright holder is
|
||||
reinstated permanently if the copyright holder notifies you of the
|
||||
violation by some reasonable means, this is the first time you have
|
||||
received notice of violation of this License (for any work) from that
|
||||
copyright holder, and you cure the violation prior to 30 days after
|
||||
your receipt of the notice.
|
||||
|
||||
Termination of your rights under this section does not terminate the
|
||||
licenses of persons who have received copies or rights from you under
|
||||
this License. If your rights have been terminated and not permanently
|
||||
reinstated, you do not qualify to receive new licenses for the same
|
||||
material under section 10.
|
||||
|
||||
9. Acceptance Not Required for Having Copies.
|
||||
|
||||
You are not required to accept this License in order to receive or
|
||||
run a copy of the Program. Ancillary propagation of a covered work
|
||||
occurring solely as a consequence of using peer-to-peer transmission
|
||||
to receive a copy likewise does not require acceptance. However,
|
||||
nothing other than this License grants you permission to propagate or
|
||||
modify any covered work. These actions infringe copyright if you do
|
||||
not accept this License. Therefore, by modifying or propagating a
|
||||
covered work, you indicate your acceptance of this License to do so.
|
||||
|
||||
10. Automatic Licensing of Downstream Recipients.
|
||||
|
||||
Each time you convey a covered work, the recipient automatically
|
||||
receives a license from the original licensors, to run, modify and
|
||||
propagate that work, subject to this License. You are not responsible
|
||||
for enforcing compliance by third parties with this License.
|
||||
|
||||
An "entity transaction" is a transaction transferring control of an
|
||||
organization, or substantially all assets of one, or subdividing an
|
||||
organization, or merging organizations. If propagation of a covered
|
||||
work results from an entity transaction, each party to that
|
||||
transaction who receives a copy of the work also receives whatever
|
||||
licenses to the work the party's predecessor in interest had or could
|
||||
give under the previous paragraph, plus a right to possession of the
|
||||
Corresponding Source of the work from the predecessor in interest, if
|
||||
the predecessor has it or can get it with reasonable efforts.
|
||||
|
||||
You may not impose any further restrictions on the exercise of the
|
||||
rights granted or affirmed under this License. For example, you may
|
||||
not impose a license fee, royalty, or other charge for exercise of
|
||||
rights granted under this License, and you may not initiate litigation
|
||||
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
||||
any patent claim is infringed by making, using, selling, offering for
|
||||
sale, or importing the Program or any portion of it.
|
||||
|
||||
11. Patents.
|
||||
|
||||
A "contributor" is a copyright holder who authorizes use under this
|
||||
License of the Program or a work on which the Program is based. The
|
||||
work thus licensed is called the contributor's "contributor version".
|
||||
|
||||
A contributor's "essential patent claims" are all patent claims
|
||||
owned or controlled by the contributor, whether already acquired or
|
||||
hereafter acquired, that would be infringed by some manner, permitted
|
||||
by this License, of making, using, or selling its contributor version,
|
||||
but do not include claims that would be infringed only as a
|
||||
consequence of further modification of the contributor version. For
|
||||
purposes of this definition, "control" includes the right to grant
|
||||
patent sublicenses in a manner consistent with the requirements of
|
||||
this License.
|
||||
|
||||
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
||||
patent license under the contributor's essential patent claims, to
|
||||
make, use, sell, offer for sale, import, and otherwise run, modify and
|
||||
propagate the contents of its contributor version.
|
||||
|
||||
In the following three paragraphs, a "patent license" is any express
|
||||
agreement or commitment, however denominated, not to enforce a patent
|
||||
(such as an express permission to practice a patent or covenant not to
|
||||
sue for patent infringement). To "grant" such a patent license to a
|
||||
party means to make such an agreement or commitment not to enforce a
|
||||
patent against the party.
|
||||
|
||||
If you convey a covered work, knowingly relying on a patent license,
|
||||
and the Corresponding Source for the work is not available for anyone
|
||||
to copy, free of charge and under the terms of this License, through a
|
||||
publicly available network server or other readily accessible means,
|
||||
then you must either (1) cause the Corresponding Source to be so
|
||||
available, or (2) arrange to deprive yourself of the benefit of the
|
||||
patent license for this particular work, or (3) arrange, in a manner
|
||||
consistent with the requirements of this License, to extend the patent
|
||||
license to downstream recipients. "Knowingly relying" means you have
|
||||
actual knowledge that, but for the patent license, your conveying the
|
||||
covered work in a country, or your recipient's use of the covered work
|
||||
in a country, would infringe one or more identifiable patents in that
|
||||
country that you have reason to believe are valid.
|
||||
|
||||
If, pursuant to or in connection with a single transaction or
|
||||
arrangement, you convey, or propagate by procuring conveyance of, a
|
||||
covered work, and grant a patent license to some of the parties
|
||||
receiving the covered work authorizing them to use, propagate, modify
|
||||
or convey a specific copy of the covered work, then the patent license
|
||||
you grant is automatically extended to all recipients of the covered
|
||||
work and works based on it.
|
||||
|
||||
A patent license is "discriminatory" if it does not include within
|
||||
the scope of its coverage, prohibit the exercise of, or is
|
||||
conditioned on the non-exercise of one or more of the rights that are
|
||||
specifically granted under this License. You may not convey a covered
|
||||
work if you are a party to an arrangement with a third party that is
|
||||
in the business of distributing software, under which you make payment
|
||||
to the third party based on the extent of your activity of conveying
|
||||
the work, and under which the third party grants, to any of the
|
||||
parties who would receive the covered work from you, a discriminatory
|
||||
patent license (a) in connection with copies of the covered work
|
||||
conveyed by you (or copies made from those copies), or (b) primarily
|
||||
for and in connection with specific products or compilations that
|
||||
contain the covered work, unless you entered into that arrangement,
|
||||
or that patent license was granted, prior to 28 March 2007.
|
||||
|
||||
Nothing in this License shall be construed as excluding or limiting
|
||||
any implied license or other defenses to infringement that may
|
||||
otherwise be available to you under applicable patent law.
|
||||
|
||||
12. No Surrender of Others' Freedom.
|
||||
|
||||
If conditions are imposed on you (whether by court order, agreement or
|
||||
otherwise) that contradict the conditions of this License, they do not
|
||||
excuse you from the conditions of this License. If you cannot convey a
|
||||
covered work so as to satisfy simultaneously your obligations under this
|
||||
License and any other pertinent obligations, then as a consequence you may
|
||||
not convey it at all. For example, if you agree to terms that obligate you
|
||||
to collect a royalty for further conveying from those to whom you convey
|
||||
the Program, the only way you could satisfy both those terms and this
|
||||
License would be to refrain entirely from conveying the Program.
|
||||
|
||||
13. Remote Network Interaction; Use with the GNU General Public License.
|
||||
|
||||
Notwithstanding any other provision of this License, if you modify the
|
||||
Program, your modified version must prominently offer all users
|
||||
interacting with it remotely through a computer network (if your version
|
||||
supports such interaction) an opportunity to receive the Corresponding
|
||||
Source of your version by providing access to the Corresponding Source
|
||||
from a network server at no charge, through some standard or customary
|
||||
means of facilitating copying of software. This Corresponding Source
|
||||
shall include the Corresponding Source for any work covered by version 3
|
||||
of the GNU General Public License that is incorporated pursuant to the
|
||||
following paragraph.
|
||||
|
||||
Notwithstanding any other provision of this License, you have
|
||||
permission to link or combine any covered work with a work licensed
|
||||
under version 3 of the GNU General Public License into a single
|
||||
combined work, and to convey the resulting work. The terms of this
|
||||
License will continue to apply to the part which is the covered work,
|
||||
but the work with which it is combined will remain governed by version
|
||||
3 of the GNU General Public License.
|
||||
|
||||
14. Revised Versions of this License.
|
||||
|
||||
The Free Software Foundation may publish revised and/or new versions of
|
||||
the GNU Affero General Public License from time to time. Such new versions
|
||||
will be similar in spirit to the present version, but may differ in detail to
|
||||
address new problems or concerns.
|
||||
|
||||
Each version is given a distinguishing version number. If the
|
||||
Program specifies that a certain numbered version of the GNU Affero General
|
||||
Public License "or any later version" applies to it, you have the
|
||||
option of following the terms and conditions either of that numbered
|
||||
version or of any later version published by the Free Software
|
||||
Foundation. If the Program does not specify a version number of the
|
||||
GNU Affero General Public License, you may choose any version ever published
|
||||
by the Free Software Foundation.
|
||||
|
||||
If the Program specifies that a proxy can decide which future
|
||||
versions of the GNU Affero General Public License can be used, that proxy's
|
||||
public statement of acceptance of a version permanently authorizes you
|
||||
to choose that version for the Program.
|
||||
|
||||
Later license versions may give you additional or different
|
||||
permissions. However, no additional obligations are imposed on any
|
||||
author or copyright holder as a result of your choosing to follow a
|
||||
later version.
|
||||
|
||||
15. Disclaimer of Warranty.
|
||||
|
||||
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
||||
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
||||
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
||||
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
||||
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
||||
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
||||
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
||||
|
||||
16. Limitation of Liability.
|
||||
|
||||
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
||||
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
||||
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
||||
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
||||
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
||||
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
||||
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
||||
SUCH DAMAGES.
|
||||
|
||||
17. Interpretation of Sections 15 and 16.
|
||||
|
||||
If the disclaimer of warranty and limitation of liability provided
|
||||
above cannot be given legal effect according to their terms,
|
||||
reviewing courts shall apply local law that most closely approximates
|
||||
an absolute waiver of all civil liability in connection with the
|
||||
Program, unless a warranty or assumption of liability accompanies a
|
||||
copy of the Program in return for a fee.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
How to Apply These Terms to Your New Programs
|
||||
|
||||
If you develop a new program, and you want it to be of the greatest
|
||||
possible use to the public, the best way to achieve this is to make it
|
||||
free software which everyone can redistribute and change under these terms.
|
||||
|
||||
To do so, attach the following notices to the program. It is safest
|
||||
to attach them to the start of each source file to most effectively
|
||||
state the exclusion of warranty; and each file should have at least the
|
||||
"copyright" line and a pointer to where the full notice is found.
|
||||
|
||||
<one line to give the program's name and a brief idea of what it does.>
|
||||
Copyright (C) <year> <name of author>
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU Affero General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU Affero General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU Affero General Public License
|
||||
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
Also add information on how to contact you by electronic and paper mail.
|
||||
|
||||
If your software can interact with users remotely through a computer
|
||||
network, you should also make sure that it provides a way for users to
|
||||
get its source. For example, if your program is a web application, you
|
||||
could provide a link from your application to the source code on a
|
||||
network server.
|
||||
|
||||
The GNU Affero General Public License does not permit incorporating your
|
||||
program into proprietary software. If your program is a subroutine
|
||||
library, you may consider it more useful to permit linking proprietary
|
||||
applications with the library. If this is what you want to do, use the
|
||||
GNU Lesser General Public License instead of this License. But first,
|
||||
please read <https://www.gnu.org/licenses/why-not-lgpl.html>.
|
||||
114
README.md
Normal file
114
README.md
Normal file
@@ -0,0 +1,114 @@
|
||||
# AI Chat - 智能对话助手
|
||||
|
||||
> **版本:v1.0.0** | **协议:AGPLv3** | **PHP 8.0 + MySQL + Nginx**
|
||||
|
||||
一个基于 PHP 8.0 的 AI 聊天 Web 应用,支持多种 AI 服务供应商,提供流式对话体验。
|
||||
|
||||
## ✨ 功能特性
|
||||
|
||||
- 🔐 用户认证系统(JWT)
|
||||
- 🤖 多 AI 服务供应商支持(OpenAI、Claude、DeepSeek 等 12+ 供应商)
|
||||
- 💭 思考/非思考模式切换
|
||||
- 📡 SSE 流式响应(打字机效果)
|
||||
- 💬 多会话管理
|
||||
- 📁 文件上传(图片 + 代码文件)
|
||||
- 🎨 代码高亮和 Markdown 渲染
|
||||
- 🧠 自定义提示词和可配置人格(预设 + 自定义)
|
||||
- 💾 双重历史记录存储(服务器 + 本地缓存)
|
||||
- 🌐 Web 安装界面(零配置部署)
|
||||
- ⚙️ API 配置管理界面
|
||||
|
||||
## 🛠️ 技术栈
|
||||
|
||||
| 组件 | 技术 |
|
||||
|------|------|
|
||||
| 后端 | PHP 8.0(原生,无框架) |
|
||||
| 数据库 | MySQL 5.7+ / 8.0(PDO 驱动) |
|
||||
| 前端 | PHP 模板 + 原生 JavaScript + AJAX |
|
||||
| Web 服务器 | Nginx + PHP-FPM |
|
||||
| 认证 | JWT(firebase/php-jwt) |
|
||||
| 依赖管理 | Composer(PSR-4 自动加载) |
|
||||
| 代码高亮 | highlight.js(CDN) |
|
||||
| Markdown 渲染 | marked.js(CDN) |
|
||||
|
||||
## 📁 项目结构
|
||||
|
||||
```
|
||||
ai-chat/
|
||||
├── public/ # Web 根目录(Nginx 指向此目录)
|
||||
│ ├── api.php # API 统一入口
|
||||
│ ├── login.php # 登录页面
|
||||
│ ├── chat.php # 聊天主页面
|
||||
│ ├── config.php # 配置管理页面
|
||||
│ └── install.php # 安装向导页面
|
||||
├── app/ # 应用代码(PSR-4 自动加载)
|
||||
│ ├── Config/ # 配置管理(Database、AppConfig)
|
||||
│ ├── Controllers/ # 控制器(Auth、Chat、Session 等)
|
||||
│ ├── Middleware/ # 中间件(Auth、Admin)
|
||||
│ ├── Models/ # 数据模型(User、Session、Message 等)
|
||||
│ ├── Services/ # 服务层(AIService、Installer、Providers)
|
||||
│ └── Views/ # 视图模板(login、chat、config、layout)
|
||||
├── assets/ # 前端静态资源
|
||||
│ ├── css/ # 样式文件
|
||||
│ ├── js/ # JavaScript 文件
|
||||
│ └── img/ # 图片资源
|
||||
├── config/ # JSON 配置文件(安装后生成)
|
||||
├── uploads/ # 用户上传文件
|
||||
├── docs/ # 文档
|
||||
├── vendor/ # Composer 依赖(安装后生成)
|
||||
├── composer.json # PHP 依赖配置
|
||||
└── .gitignore
|
||||
```
|
||||
|
||||
## 🚀 快速开始
|
||||
|
||||
### 环境要求
|
||||
|
||||
- PHP >= 8.0
|
||||
- MySQL >= 5.7
|
||||
- Nginx(或 Apache)
|
||||
- Composer
|
||||
- PHP 扩展:PDO、cURL、json、mbstring
|
||||
|
||||
### 安装步骤
|
||||
|
||||
1. **克隆项目**
|
||||
```bash
|
||||
git clone <repository-url>
|
||||
cd ai-chat
|
||||
```
|
||||
|
||||
2. **安装 PHP 依赖**
|
||||
```bash
|
||||
composer install
|
||||
```
|
||||
|
||||
3. **配置 Nginx**
|
||||
- 将网站根目录指向 `public/` 目录
|
||||
- 参考 `docs/nginx.conf` 或 `docs/baota-nginx-snippet.conf` 配置 Nginx
|
||||
|
||||
4. **运行安装向导**
|
||||
- 访问 `http://your-domain.com/install.php`
|
||||
- 按照向导步骤完成安装
|
||||
|
||||
详细安装说明请查看 [INSTALL.md](INSTALL.md)。
|
||||
|
||||
## 📖 文档
|
||||
|
||||
- [安装指南](INSTALL.md)
|
||||
- [部署文档](docs/DEPLOY.md)
|
||||
- [架构说明](docs/ARCHITECTURE.md)
|
||||
- [API 文档](docs/API.md)
|
||||
- [人格系统](docs/PERSONALITY.md)
|
||||
- [Nginx 配置](docs/nginx.conf)
|
||||
- [宝塔面板配置](docs/baota-nginx-snippet.conf)
|
||||
|
||||
## 📄 开源协议
|
||||
|
||||
本项目基于 [GNU Affero General Public License v3.0 (AGPLv3)](LICENSE) 开源。
|
||||
|
||||
## 📌 版本历史
|
||||
|
||||
| 版本 | 日期 | 说明 |
|
||||
|------|------|------|
|
||||
| v1.0.0 | 2026.5.5 | 初始版本,PHP 8.0 全栈实现 |
|
||||
0
app/Config/.gitkeep
Normal file
0
app/Config/.gitkeep
Normal file
57
app/Config/AppConfig.php
Normal file
57
app/Config/AppConfig.php
Normal file
@@ -0,0 +1,57 @@
|
||||
<?php
|
||||
|
||||
namespace App\Config;
|
||||
|
||||
class AppConfig
|
||||
{
|
||||
private static ?array $cache = null;
|
||||
private static string $configPath = '';
|
||||
|
||||
private static function getConfigPath(): string
|
||||
{
|
||||
if (self::$configPath === '') {
|
||||
self::$configPath = dirname(__DIR__, 2) . '/config/app-config.json';
|
||||
}
|
||||
|
||||
return self::$configPath;
|
||||
}
|
||||
|
||||
private static function load(): array
|
||||
{
|
||||
if (self::$cache !== null) {
|
||||
return self::$cache;
|
||||
}
|
||||
|
||||
$path = self::getConfigPath();
|
||||
|
||||
if (!file_exists($path)) {
|
||||
self::$cache = [];
|
||||
return self::$cache;
|
||||
}
|
||||
|
||||
$content = file_get_contents($path);
|
||||
self::$cache = json_decode($content, true) ?? [];
|
||||
|
||||
return self::$cache;
|
||||
}
|
||||
|
||||
public static function get(string $key, $default = null)
|
||||
{
|
||||
$config = self::load();
|
||||
return $config[$key] ?? $default;
|
||||
}
|
||||
|
||||
public static function set(string $key, $value): void
|
||||
{
|
||||
$config = self::load();
|
||||
$config[$key] = $value;
|
||||
self::$cache = $config;
|
||||
|
||||
$dir = dirname(self::getConfigPath());
|
||||
if (!is_dir($dir)) {
|
||||
mkdir($dir, 0755, true);
|
||||
}
|
||||
|
||||
file_put_contents(self::getConfigPath(), json_encode($config, JSON_PRETTY_PRINT | JSON_UNESCAPED_UNICODE));
|
||||
}
|
||||
}
|
||||
39
app/Config/Database.php
Normal file
39
app/Config/Database.php
Normal file
@@ -0,0 +1,39 @@
|
||||
<?php
|
||||
|
||||
namespace App\Config;
|
||||
|
||||
use PDO;
|
||||
use PDOException;
|
||||
|
||||
class Database
|
||||
{
|
||||
private static ?PDO $instance = null;
|
||||
|
||||
private function __construct()
|
||||
{
|
||||
$configPath = dirname(__DIR__, 2) . '/config/db-config.json';
|
||||
|
||||
if (!file_exists($configPath)) {
|
||||
throw new PDOException('数据库配置文件不存在');
|
||||
}
|
||||
|
||||
$config = json_decode(file_get_contents($configPath), true);
|
||||
|
||||
$dsn = "mysql:host={$config['host']};port={$config['port']};dbname={$config['dbname']};charset=utf8mb4";
|
||||
|
||||
self::$instance = new PDO($dsn, $config['user'], $config['password'], [
|
||||
PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
|
||||
PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC,
|
||||
PDO::MYSQL_ATTR_INIT_COMMAND => "SET NAMES utf8mb4",
|
||||
]);
|
||||
}
|
||||
|
||||
public static function getInstance(): PDO
|
||||
{
|
||||
if (self::$instance === null) {
|
||||
new self();
|
||||
}
|
||||
|
||||
return self::$instance;
|
||||
}
|
||||
}
|
||||
0
app/Controllers/.gitkeep
Normal file
0
app/Controllers/.gitkeep
Normal file
69
app/Controllers/AuthController.php
Normal file
69
app/Controllers/AuthController.php
Normal file
@@ -0,0 +1,69 @@
|
||||
<?php
|
||||
|
||||
namespace App\Controllers;
|
||||
|
||||
use App\Models\User;
|
||||
use App\Config\AppConfig;
|
||||
use Firebase\JWT\JWT;
|
||||
use Firebase\JWT\Key;
|
||||
|
||||
class AuthController
|
||||
{
|
||||
public static function login(): void
|
||||
{
|
||||
$input = json_decode(file_get_contents('php://input'), true);
|
||||
$username = $input['username'] ?? '';
|
||||
$password = $input['password'] ?? '';
|
||||
|
||||
if (!$username || !$password) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '用户名和密码不能为空']);
|
||||
return;
|
||||
}
|
||||
|
||||
$user = User::findByUsername($username);
|
||||
if (!$user || !User::verifyPassword($username, $password)) {
|
||||
http_response_code(401);
|
||||
echo json_encode(['success' => false, 'message' => '用户名或密码错误']);
|
||||
return;
|
||||
}
|
||||
|
||||
$jwtSecret = AppConfig::get('jwtSecret');
|
||||
$jwtExpiry = AppConfig::get('jwtExpiry', 86400);
|
||||
|
||||
$payload = [
|
||||
'userId' => $user['id'],
|
||||
'username' => $user['username'],
|
||||
'role' => $user['role'],
|
||||
'iat' => time(),
|
||||
'exp' => time() + $jwtExpiry
|
||||
];
|
||||
|
||||
$token = JWT::encode($payload, $jwtSecret, 'HS256');
|
||||
|
||||
echo json_encode([
|
||||
'success' => true,
|
||||
'data' => [
|
||||
'token' => $token,
|
||||
'user' => [
|
||||
'id' => $user['id'],
|
||||
'username' => $user['username'],
|
||||
'role' => $user['role']
|
||||
]
|
||||
]
|
||||
]);
|
||||
}
|
||||
|
||||
public static function me(): void
|
||||
{
|
||||
$user = $GLOBALS['auth_user'];
|
||||
echo json_encode([
|
||||
'success' => true,
|
||||
'data' => [
|
||||
'id' => $user['userId'],
|
||||
'username' => $user['username'],
|
||||
'role' => $user['role']
|
||||
]
|
||||
]);
|
||||
}
|
||||
}
|
||||
117
app/Controllers/ChatController.php
Normal file
117
app/Controllers/ChatController.php
Normal file
@@ -0,0 +1,117 @@
|
||||
<?php
|
||||
|
||||
namespace App\Controllers;
|
||||
|
||||
use App\Models\Config;
|
||||
use App\Services\AIService;
|
||||
|
||||
class ChatController
|
||||
{
|
||||
public static function completions(): void
|
||||
{
|
||||
$input = json_decode(file_get_contents('php://input'), true);
|
||||
|
||||
$provider = $input['provider'] ?? '';
|
||||
$model = $input['model'] ?? '';
|
||||
$messages = $input['messages'] ?? [];
|
||||
$stream = $input['stream'] ?? true;
|
||||
$systemPrompt = $input['systemPrompt'] ?? '';
|
||||
$thinkingMode = $input['thinkingMode'] ?? false;
|
||||
|
||||
if (!$provider || !$model || !$messages) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '缺少必要参数(provider、model、messages)']);
|
||||
return;
|
||||
}
|
||||
|
||||
if (!is_array($messages) || empty($messages)) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => 'messages 必须是非空数组']);
|
||||
return;
|
||||
}
|
||||
|
||||
// 获取供应商配置
|
||||
$configRow = Config::getByKey('providers');
|
||||
if (!$configRow) {
|
||||
http_response_code(500);
|
||||
echo json_encode(['success' => false, 'message' => '未找到供应商配置']);
|
||||
return;
|
||||
}
|
||||
|
||||
$providers = json_decode($configRow['config_value'], true);
|
||||
if (!is_array($providers)) {
|
||||
http_response_code(500);
|
||||
echo json_encode(['success' => false, 'message' => '供应商配置格式错误']);
|
||||
return;
|
||||
}
|
||||
|
||||
$providerKey = $provider;
|
||||
if (!isset($providers[$providerKey])) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '供应商不存在: ' . $providerKey]);
|
||||
return;
|
||||
}
|
||||
|
||||
$providerConfig = $providers[$providerKey];
|
||||
if (empty($providerConfig['enabled'])) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '供应商已禁用: ' . $providerKey]);
|
||||
return;
|
||||
}
|
||||
|
||||
$models = $providerConfig['models'] ?? [];
|
||||
if (!empty($models) && !in_array($model, $models)) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '模型不在供应商支持列表中: ' . $model]);
|
||||
return;
|
||||
}
|
||||
|
||||
// 非流式模式暂不支持
|
||||
if (!$stream) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '当前仅支持流式响应']);
|
||||
return;
|
||||
}
|
||||
|
||||
// 设置 SSE 响应头
|
||||
header('Content-Type: text/event-stream');
|
||||
header('Cache-Control: no-cache');
|
||||
header('Connection: keep-alive');
|
||||
header('X-Accel-Buffering: no');
|
||||
|
||||
// 关闭输出缓冲
|
||||
while (ob_get_level()) {
|
||||
ob_end_flush();
|
||||
}
|
||||
|
||||
$options = [
|
||||
'provider' => $providerConfig,
|
||||
'systemPrompt' => $systemPrompt,
|
||||
'thinkingMode' => (bool) $thinkingMode
|
||||
];
|
||||
|
||||
try {
|
||||
AIService::streamChat($providerKey, $model, $messages, $options, function ($chunk, $type = 'content') {
|
||||
if ($type === 'thinking') {
|
||||
self::sendSSE(['thinking' => $chunk]);
|
||||
} else {
|
||||
self::sendSSE(['content' => $chunk]);
|
||||
}
|
||||
});
|
||||
|
||||
self::sendSSE('[DONE]');
|
||||
} catch (\Throwable $e) {
|
||||
self::sendSSE(['type' => 'error', 'message' => $e->getMessage()]);
|
||||
}
|
||||
}
|
||||
|
||||
private static function sendSSE($data): void
|
||||
{
|
||||
if (is_string($data)) {
|
||||
echo "data: " . $data . "\n\n";
|
||||
} else {
|
||||
echo "data: " . json_encode($data, JSON_UNESCAPED_UNICODE) . "\n\n";
|
||||
}
|
||||
flush();
|
||||
}
|
||||
}
|
||||
106
app/Controllers/ConfigController.php
Normal file
106
app/Controllers/ConfigController.php
Normal file
@@ -0,0 +1,106 @@
|
||||
<?php
|
||||
|
||||
namespace App\Controllers;
|
||||
|
||||
use App\Models\Config;
|
||||
use App\Models\Personality;
|
||||
|
||||
class ConfigController
|
||||
{
|
||||
public static function getConfig(): void
|
||||
{
|
||||
$configRow = Config::getByKey('providers');
|
||||
$providers = $configRow ? json_decode($configRow['config_value'], true) : [];
|
||||
echo json_encode(['success' => true, 'data' => ['providers' => $providers]]);
|
||||
}
|
||||
|
||||
public static function updateConfig(): void
|
||||
{
|
||||
$input = json_decode(file_get_contents('php://input'), true);
|
||||
$providers = $input['providers'] ?? [];
|
||||
|
||||
if (!is_array($providers)) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => 'providers 必须是数组']);
|
||||
return;
|
||||
}
|
||||
|
||||
Config::setByKey('providers', json_encode($providers, JSON_UNESCAPED_UNICODE));
|
||||
echo json_encode(['success' => true, 'message' => '配置更新成功']);
|
||||
}
|
||||
|
||||
public static function listPersonalities(): void
|
||||
{
|
||||
$personalities = Personality::findAll();
|
||||
echo json_encode(['success' => true, 'data' => $personalities]);
|
||||
}
|
||||
|
||||
public static function createPersonality(): void
|
||||
{
|
||||
$input = json_decode(file_get_contents('php://input'), true);
|
||||
$name = $input['name'] ?? '';
|
||||
$prompt = $input['prompt'] ?? '';
|
||||
|
||||
if (!$name || !$prompt) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '名称和提示词不能为空']);
|
||||
return;
|
||||
}
|
||||
|
||||
$data = [
|
||||
'name' => $name,
|
||||
'prompt' => $prompt,
|
||||
'description' => $input['description'] ?? null,
|
||||
'icon' => $input['icon'] ?? null,
|
||||
'is_preset' => 0,
|
||||
'created_by' => $GLOBALS['auth_user']['userId']
|
||||
];
|
||||
|
||||
$personality = Personality::create($data);
|
||||
echo json_encode(['success' => true, 'data' => $personality]);
|
||||
}
|
||||
|
||||
public static function updatePersonality($id): void
|
||||
{
|
||||
$personality = Personality::findById($id);
|
||||
if (!$personality) {
|
||||
http_response_code(404);
|
||||
echo json_encode(['success' => false, 'message' => '人格不存在']);
|
||||
return;
|
||||
}
|
||||
if ($personality['is_preset']) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '预设人格不可编辑']);
|
||||
return;
|
||||
}
|
||||
|
||||
$input = json_decode(file_get_contents('php://input'), true);
|
||||
$data = array_filter([
|
||||
'name' => $input['name'] ?? null,
|
||||
'prompt' => $input['prompt'] ?? null,
|
||||
'description' => $input['description'] ?? null,
|
||||
'icon' => $input['icon'] ?? null,
|
||||
], fn($v) => $v !== null);
|
||||
|
||||
$updated = Personality::update($id, $data);
|
||||
echo json_encode(['success' => true, 'data' => $updated]);
|
||||
}
|
||||
|
||||
public static function deletePersonality($id): void
|
||||
{
|
||||
$personality = Personality::findById($id);
|
||||
if (!$personality) {
|
||||
http_response_code(404);
|
||||
echo json_encode(['success' => false, 'message' => '人格不存在']);
|
||||
return;
|
||||
}
|
||||
if ($personality['is_preset']) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '预设人格不可删除']);
|
||||
return;
|
||||
}
|
||||
|
||||
Personality::delete($id);
|
||||
echo json_encode(['success' => true, 'message' => '删除成功']);
|
||||
}
|
||||
}
|
||||
118
app/Controllers/InstallController.php
Normal file
118
app/Controllers/InstallController.php
Normal file
@@ -0,0 +1,118 @@
|
||||
<?php
|
||||
|
||||
namespace App\Controllers;
|
||||
|
||||
use App\Services\Installer;
|
||||
use App\Config\AppConfig;
|
||||
use App\Config\Database;
|
||||
use App\Models\User;
|
||||
use App\Models\Config;
|
||||
|
||||
class InstallController
|
||||
{
|
||||
public static function status(): void
|
||||
{
|
||||
$installed = Installer::isInstalled();
|
||||
echo json_encode(['success' => true, 'data' => ['installed' => $installed]]);
|
||||
}
|
||||
|
||||
public static function testDb(): void
|
||||
{
|
||||
$input = json_decode(file_get_contents('php://input'), true);
|
||||
$host = $input['host'] ?? '';
|
||||
$port = $input['port'] ?? 3306;
|
||||
$user = $input['user'] ?? '';
|
||||
$password = $input['password'] ?? '';
|
||||
$database = $input['database'] ?? '';
|
||||
|
||||
if (!$host || !$user || !$database) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '缺少必要参数']);
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
$dsn = "mysql:host={$host};port={$port};dbname={$database};charset=utf8mb4";
|
||||
$pdo = new \PDO($dsn, $user, $password, [
|
||||
\PDO::ATTR_ERRMODE => \PDO::ERRMODE_EXCEPTION
|
||||
]);
|
||||
echo json_encode(['success' => true, 'message' => '数据库连接成功']);
|
||||
} catch (\PDOException $e) {
|
||||
http_response_code(500);
|
||||
echo json_encode(['success' => false, 'message' => '数据库连接失败: ' . $e->getMessage()]);
|
||||
}
|
||||
}
|
||||
|
||||
public static function setup(): void
|
||||
{
|
||||
if (Installer::isInstalled()) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '系统已安装,不能重复安装']);
|
||||
return;
|
||||
}
|
||||
|
||||
$input = json_decode(file_get_contents('php://input'), true);
|
||||
$username = $input['username'] ?? '';
|
||||
$password = $input['password'] ?? '';
|
||||
$dbConfig = $input['dbConfig'] ?? [];
|
||||
$providers = $input['providers'] ?? [];
|
||||
|
||||
if (!$username || !$password) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '用户名和密码不能为空']);
|
||||
return;
|
||||
}
|
||||
if (strlen($password) < 6) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '密码长度不能少于6位']);
|
||||
return;
|
||||
}
|
||||
if (!$dbConfig || !$dbConfig['host'] || !$dbConfig['user'] || !$dbConfig['database']) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '请填写数据库配置']);
|
||||
return;
|
||||
}
|
||||
if (!is_array($providers) || count($providers) === 0) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '请至少配置一个AI服务提供商']);
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
$dsn = "mysql:host={$dbConfig['host']};port=" . ($dbConfig['port'] ?? 3306) . ";dbname={$dbConfig['database']};charset=utf8mb4";
|
||||
$pdo = new \PDO($dsn, $dbConfig['user'], $dbConfig['password'], [\PDO::ATTR_ERRMODE => \PDO::ERRMODE_EXCEPTION]);
|
||||
} catch (\PDOException $e) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '数据库连接失败: ' . $e->getMessage()]);
|
||||
return;
|
||||
}
|
||||
|
||||
$configDir = __DIR__ . '/../../config';
|
||||
file_put_contents($configDir . '/db-config.json', json_encode($dbConfig, JSON_PRETTY_PRINT | JSON_UNESCAPED_UNICODE));
|
||||
|
||||
try {
|
||||
Installer::runMigration();
|
||||
} catch (\Exception $e) {
|
||||
http_response_code(500);
|
||||
echo json_encode(['success' => false, 'message' => '数据库迁移失败: ' . $e->getMessage()]);
|
||||
return;
|
||||
}
|
||||
|
||||
Installer::seedDefaults();
|
||||
|
||||
$admin = User::create(['username' => $username, 'password' => $password, 'role' => 'admin']);
|
||||
|
||||
$jwtSecret = bin2hex(random_bytes(32));
|
||||
AppConfig::set('jwtSecret', $jwtSecret);
|
||||
AppConfig::set('jwtExpiry', 86400);
|
||||
AppConfig::set('corsOrigin', '');
|
||||
|
||||
Config::setByKey('providers', json_encode($providers));
|
||||
|
||||
echo json_encode([
|
||||
'success' => true,
|
||||
'message' => '安装成功',
|
||||
'data' => ['user' => $admin]
|
||||
]);
|
||||
}
|
||||
}
|
||||
57
app/Controllers/MessageController.php
Normal file
57
app/Controllers/MessageController.php
Normal file
@@ -0,0 +1,57 @@
|
||||
<?php
|
||||
|
||||
namespace App\Controllers;
|
||||
|
||||
use App\Models\Session;
|
||||
use App\Models\Message;
|
||||
|
||||
class MessageController
|
||||
{
|
||||
public static function index(int $sessionId): void
|
||||
{
|
||||
$user = $GLOBALS['auth_user'];
|
||||
$session = Session::findById($sessionId);
|
||||
if (!$session || $session['user_id'] != $user['userId']) {
|
||||
http_response_code(404);
|
||||
echo json_encode(['success' => false, 'message' => '会话不存在']);
|
||||
return;
|
||||
}
|
||||
$messages = Message::findBySessionId($sessionId);
|
||||
echo json_encode(['success' => true, 'data' => $messages]);
|
||||
}
|
||||
|
||||
public static function create(int $sessionId): void
|
||||
{
|
||||
$user = $GLOBALS['auth_user'];
|
||||
$session = Session::findById($sessionId);
|
||||
if (!$session || $session['user_id'] != $user['userId']) {
|
||||
http_response_code(404);
|
||||
echo json_encode(['success' => false, 'message' => '会话不存在']);
|
||||
return;
|
||||
}
|
||||
|
||||
$input = json_decode(file_get_contents('php://input'), true);
|
||||
|
||||
if (empty($input['role']) || !isset($input['content']) || $input['content'] === '') {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => 'role 和 content 为必填字段']);
|
||||
return;
|
||||
}
|
||||
|
||||
$data = [
|
||||
'session_id' => $sessionId,
|
||||
'role' => $input['role'],
|
||||
'content' => $input['content'],
|
||||
];
|
||||
|
||||
if (isset($input['file_info'])) {
|
||||
$data['file_info'] = $input['file_info'];
|
||||
}
|
||||
if (isset($input['thinking_content'])) {
|
||||
$data['thinking_content'] = $input['thinking_content'];
|
||||
}
|
||||
|
||||
$message = Message::create($data);
|
||||
echo json_encode(['success' => true, 'data' => $message]);
|
||||
}
|
||||
}
|
||||
88
app/Controllers/SessionController.php
Normal file
88
app/Controllers/SessionController.php
Normal file
@@ -0,0 +1,88 @@
|
||||
<?php
|
||||
|
||||
namespace App\Controllers;
|
||||
|
||||
use App\Models\Session;
|
||||
|
||||
class SessionController
|
||||
{
|
||||
public static function index(): void
|
||||
{
|
||||
$user = $GLOBALS['auth_user'];
|
||||
$sessions = Session::findByUserId($user['userId']);
|
||||
echo json_encode(['success' => true, 'data' => $sessions]);
|
||||
}
|
||||
|
||||
public static function create(): void
|
||||
{
|
||||
$input = json_decode(file_get_contents('php://input'), true);
|
||||
$user = $GLOBALS['auth_user'];
|
||||
|
||||
$data = [
|
||||
'user_id' => $user['userId'],
|
||||
'name' => $input['name'] ?? '新会话',
|
||||
'provider' => $input['provider'] ?? 'newapi',
|
||||
'model' => $input['model'] ?? 'gpt-3.5-turbo',
|
||||
'system_prompt' => $input['system_prompt'] ?? '',
|
||||
'thinking_mode' => $input['thinking_mode'] ?? false,
|
||||
];
|
||||
|
||||
if (isset($input['personality_id'])) {
|
||||
$data['personality_id'] = $input['personality_id'];
|
||||
}
|
||||
|
||||
$session = Session::create($data);
|
||||
echo json_encode(['success' => true, 'data' => $session]);
|
||||
}
|
||||
|
||||
public static function update(int $id): void
|
||||
{
|
||||
$session = Session::findById($id);
|
||||
if (!$session) {
|
||||
http_response_code(404);
|
||||
echo json_encode(['success' => false, 'message' => '会话不存在']);
|
||||
return;
|
||||
}
|
||||
|
||||
$user = $GLOBALS['auth_user'];
|
||||
if ($session['user_id'] != $user['userId']) {
|
||||
http_response_code(404);
|
||||
echo json_encode(['success' => false, 'message' => '会话不存在']);
|
||||
return;
|
||||
}
|
||||
|
||||
$input = json_decode(file_get_contents('php://input'), true);
|
||||
|
||||
$allowedFields = ['name', 'provider', 'model', 'system_prompt', 'personality_id', 'thinking_mode'];
|
||||
$data = [];
|
||||
foreach ($allowedFields as $field) {
|
||||
if (isset($input[$field])) {
|
||||
$data[$field] = $input[$field];
|
||||
}
|
||||
}
|
||||
|
||||
Session::update($id, $data);
|
||||
$updatedSession = Session::findById($id);
|
||||
echo json_encode(['success' => true, 'data' => $updatedSession]);
|
||||
}
|
||||
|
||||
public static function delete(int $id): void
|
||||
{
|
||||
$session = Session::findById($id);
|
||||
if (!$session) {
|
||||
http_response_code(404);
|
||||
echo json_encode(['success' => false, 'message' => '会话不存在']);
|
||||
return;
|
||||
}
|
||||
|
||||
$user = $GLOBALS['auth_user'];
|
||||
if ($session['user_id'] != $user['userId']) {
|
||||
http_response_code(404);
|
||||
echo json_encode(['success' => false, 'message' => '会话不存在']);
|
||||
return;
|
||||
}
|
||||
|
||||
Session::delete($id);
|
||||
echo json_encode(['success' => true, 'message' => '删除成功']);
|
||||
}
|
||||
}
|
||||
56
app/Controllers/UploadController.php
Normal file
56
app/Controllers/UploadController.php
Normal file
@@ -0,0 +1,56 @@
|
||||
<?php
|
||||
|
||||
namespace App\Controllers;
|
||||
|
||||
class UploadController
|
||||
{
|
||||
public static function upload(): void
|
||||
{
|
||||
if (!isset($_FILES['file']) || $_FILES['file']['error'] !== UPLOAD_ERR_OK) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '请选择要上传的文件']);
|
||||
return;
|
||||
}
|
||||
|
||||
$file = $_FILES['file'];
|
||||
$allowedTypes = [
|
||||
'jpg', 'jpeg', 'png', 'gif', 'webp',
|
||||
'js', 'ts', 'py', 'java', 'cpp', 'c', 'html', 'css', 'json', 'xml',
|
||||
'txt', 'md', 'go', 'rs', 'php', 'rb', 'sql', 'yaml', 'yml', 'sh', 'bat'
|
||||
];
|
||||
|
||||
$ext = strtolower(pathinfo($file['name'], PATHINFO_EXTENSION));
|
||||
|
||||
if (!in_array($ext, $allowedTypes)) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '不支持的文件类型: .' . $ext]);
|
||||
return;
|
||||
}
|
||||
|
||||
if ($file['size'] > 10 * 1024 * 1024) {
|
||||
http_response_code(400);
|
||||
echo json_encode(['success' => false, 'message' => '文件大小不能超过10MB']);
|
||||
return;
|
||||
}
|
||||
|
||||
$filename = uniqid() . '.' . $ext;
|
||||
$uploadDir = __DIR__ . '/../../uploads/';
|
||||
$filepath = $uploadDir . $filename;
|
||||
|
||||
if (!move_uploaded_file($file['tmp_name'], $filepath)) {
|
||||
http_response_code(500);
|
||||
echo json_encode(['success' => false, 'message' => '文件上传失败']);
|
||||
return;
|
||||
}
|
||||
|
||||
echo json_encode([
|
||||
'success' => true,
|
||||
'data' => [
|
||||
'url' => '/uploads/' . $filename,
|
||||
'name' => $file['name'],
|
||||
'size' => $file['size'],
|
||||
'type' => $ext
|
||||
]
|
||||
]);
|
||||
}
|
||||
}
|
||||
0
app/Middleware/.gitkeep
Normal file
0
app/Middleware/.gitkeep
Normal file
17
app/Middleware/AdminMiddleware.php
Normal file
17
app/Middleware/AdminMiddleware.php
Normal file
@@ -0,0 +1,17 @@
|
||||
<?php
|
||||
|
||||
namespace App\Middleware;
|
||||
|
||||
class AdminMiddleware
|
||||
{
|
||||
public static function handle(): void
|
||||
{
|
||||
$user = $GLOBALS['auth_user'] ?? null;
|
||||
|
||||
if (!$user || ($user['role'] ?? '') !== 'admin') {
|
||||
http_response_code(403);
|
||||
echo json_encode(['success' => false, 'message' => '需要管理员权限']);
|
||||
exit;
|
||||
}
|
||||
}
|
||||
}
|
||||
37
app/Middleware/AuthMiddleware.php
Normal file
37
app/Middleware/AuthMiddleware.php
Normal file
@@ -0,0 +1,37 @@
|
||||
<?php
|
||||
|
||||
namespace App\Middleware;
|
||||
|
||||
use App\Config\AppConfig;
|
||||
use Firebase\JWT\JWT;
|
||||
use Firebase\JWT\Key;
|
||||
|
||||
class AuthMiddleware
|
||||
{
|
||||
public static function handle(): void
|
||||
{
|
||||
$authHeader = $_SERVER['HTTP_AUTHORIZATION'] ?? '';
|
||||
|
||||
if (!$authHeader || !preg_match('/Bearer\s+(.*)$/i', $authHeader, $matches)) {
|
||||
http_response_code(401);
|
||||
echo json_encode(['success' => false, 'message' => '请先登录']);
|
||||
exit;
|
||||
}
|
||||
|
||||
$token = $matches[1];
|
||||
|
||||
try {
|
||||
$jwtSecret = AppConfig::get('jwtSecret');
|
||||
$decoded = JWT::decode($token, new Key($jwtSecret, 'HS256'));
|
||||
$GLOBALS['auth_user'] = [
|
||||
'userId' => $decoded->userId,
|
||||
'username' => $decoded->username,
|
||||
'role' => $decoded->role
|
||||
];
|
||||
} catch (\Exception $e) {
|
||||
http_response_code(401);
|
||||
echo json_encode(['success' => false, 'message' => '请先登录']);
|
||||
exit;
|
||||
}
|
||||
}
|
||||
}
|
||||
0
app/Models/.gitkeep
Normal file
0
app/Models/.gitkeep
Normal file
27
app/Models/Config.php
Normal file
27
app/Models/Config.php
Normal file
@@ -0,0 +1,27 @@
|
||||
<?php
|
||||
|
||||
namespace App\Models;
|
||||
|
||||
use App\Config\Database;
|
||||
|
||||
class Config
|
||||
{
|
||||
public static function getByKey(string $key): ?array
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->prepare('SELECT * FROM config WHERE config_key = :config_key');
|
||||
$stmt->execute(['config_key' => $key]);
|
||||
$result = $stmt->fetch();
|
||||
return $result ?: null;
|
||||
}
|
||||
|
||||
public static function setByKey(string $key, string $value): bool
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->prepare('INSERT INTO config (config_key, config_value) VALUES (:config_key, :config_value) ON DUPLICATE KEY UPDATE config_value = VALUES(config_value)');
|
||||
return $stmt->execute([
|
||||
'config_key' => $key,
|
||||
'config_value' => $value,
|
||||
]);
|
||||
}
|
||||
}
|
||||
33
app/Models/Message.php
Normal file
33
app/Models/Message.php
Normal file
@@ -0,0 +1,33 @@
|
||||
<?php
|
||||
|
||||
namespace App\Models;
|
||||
|
||||
use App\Config\Database;
|
||||
|
||||
class Message
|
||||
{
|
||||
public static function findBySessionId(int $sessionId): array
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->prepare('SELECT * FROM messages WHERE session_id = :session_id ORDER BY created_at ASC');
|
||||
$stmt->execute(['session_id' => $sessionId]);
|
||||
return $stmt->fetchAll();
|
||||
}
|
||||
|
||||
public static function create(array $data): array
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->prepare('INSERT INTO messages (session_id, role, content, file_info, thinking_content) VALUES (:session_id, :role, :content, :file_info, :thinking_content)');
|
||||
$stmt->execute([
|
||||
'session_id' => $data['session_id'],
|
||||
'role' => $data['role'],
|
||||
'content' => $data['content'],
|
||||
'file_info' => isset($data['file_info']) ? json_encode($data['file_info']) : null,
|
||||
'thinking_content' => $data['thinking_content'] ?? null,
|
||||
]);
|
||||
|
||||
$stmt = $db->prepare('SELECT * FROM messages WHERE id = :id');
|
||||
$stmt->execute(['id' => (int) $db->lastInsertId()]);
|
||||
return $stmt->fetch();
|
||||
}
|
||||
}
|
||||
65
app/Models/Personality.php
Normal file
65
app/Models/Personality.php
Normal file
@@ -0,0 +1,65 @@
|
||||
<?php
|
||||
|
||||
namespace App\Models;
|
||||
|
||||
use App\Config\Database;
|
||||
|
||||
class Personality
|
||||
{
|
||||
public static function findAll(): array
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->query('SELECT * FROM personalities ORDER BY id ASC');
|
||||
return $stmt->fetchAll();
|
||||
}
|
||||
|
||||
public static function findById(int $id): ?array
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->prepare('SELECT * FROM personalities WHERE id = :id');
|
||||
$stmt->execute(['id' => $id]);
|
||||
$result = $stmt->fetch();
|
||||
return $result ?: null;
|
||||
}
|
||||
|
||||
public static function create(array $data): array
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->prepare('INSERT INTO personalities (name, prompt, description, icon, is_preset, created_by) VALUES (:name, :prompt, :description, :icon, :is_preset, :created_by)');
|
||||
$stmt->execute([
|
||||
'name' => $data['name'],
|
||||
'prompt' => $data['prompt'],
|
||||
'description' => $data['description'] ?? null,
|
||||
'icon' => $data['icon'] ?? null,
|
||||
'is_preset' => $data['is_preset'] ?? 0,
|
||||
'created_by' => $data['created_by'] ?? null,
|
||||
]);
|
||||
|
||||
return self::findById((int) $db->lastInsertId());
|
||||
}
|
||||
|
||||
public static function update(int $id, array $data): bool
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
|
||||
$fields = [];
|
||||
$params = ['id' => $id];
|
||||
|
||||
foreach ($data as $key => $value) {
|
||||
$fields[] = "{$key} = :{$key}";
|
||||
$params[$key] = $value;
|
||||
}
|
||||
|
||||
$sql = 'UPDATE personalities SET ' . implode(', ', $fields) . ' WHERE id = :id';
|
||||
$stmt = $db->prepare($sql);
|
||||
|
||||
return $stmt->execute($params);
|
||||
}
|
||||
|
||||
public static function delete(int $id): bool
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->prepare('DELETE FROM personalities WHERE id = :id');
|
||||
return $stmt->execute(['id' => $id]);
|
||||
}
|
||||
}
|
||||
78
app/Models/Session.php
Normal file
78
app/Models/Session.php
Normal file
@@ -0,0 +1,78 @@
|
||||
<?php
|
||||
|
||||
namespace App\Models;
|
||||
|
||||
use App\Config\Database;
|
||||
|
||||
class Session
|
||||
{
|
||||
public static function findByUserId(int $userId): array
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->prepare('SELECT * FROM sessions WHERE user_id = :user_id ORDER BY updated_at DESC');
|
||||
$stmt->execute(['user_id' => $userId]);
|
||||
return $stmt->fetchAll();
|
||||
}
|
||||
|
||||
public static function findById(int $id): ?array
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->prepare('SELECT * FROM sessions WHERE id = :id');
|
||||
$stmt->execute(['id' => $id]);
|
||||
$result = $stmt->fetch();
|
||||
return $result ?: null;
|
||||
}
|
||||
|
||||
public static function create(array $data): array
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->prepare('INSERT INTO sessions (user_id, name, provider, model, system_prompt, personality_id, thinking_mode) VALUES (:user_id, :name, :provider, :model, :system_prompt, :personality_id, :thinking_mode)');
|
||||
$stmt->execute([
|
||||
'user_id' => $data['user_id'],
|
||||
'name' => $data['name'] ?? '新会话',
|
||||
'provider' => $data['provider'] ?? 'newapi',
|
||||
'model' => $data['model'] ?? 'gpt-3.5-turbo',
|
||||
'system_prompt' => $data['system_prompt'] ?? '',
|
||||
'personality_id' => $data['personality_id'] ?? null,
|
||||
'thinking_mode' => $data['thinking_mode'] ?? 0,
|
||||
]);
|
||||
|
||||
return self::findById((int) $db->lastInsertId());
|
||||
}
|
||||
|
||||
public static function update(int $id, array $data): bool
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
|
||||
$fields = [];
|
||||
$params = ['id' => $id];
|
||||
|
||||
foreach ($data as $key => $value) {
|
||||
$fields[] = "{$key} = :{$key}";
|
||||
$params[$key] = $value;
|
||||
}
|
||||
|
||||
$fields[] = 'updated_at = CURRENT_TIMESTAMP';
|
||||
|
||||
$sql = 'UPDATE sessions SET ' . implode(', ', $fields) . ' WHERE id = :id';
|
||||
$stmt = $db->prepare($sql);
|
||||
|
||||
return $stmt->execute($params);
|
||||
}
|
||||
|
||||
public static function delete(int $id): bool
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
|
||||
$db->beginTransaction();
|
||||
try {
|
||||
$db->prepare('DELETE FROM messages WHERE session_id = :session_id')->execute(['session_id' => $id]);
|
||||
$db->prepare('DELETE FROM sessions WHERE id = :id')->execute(['id' => $id]);
|
||||
$db->commit();
|
||||
return true;
|
||||
} catch (\Throwable $e) {
|
||||
$db->rollBack();
|
||||
throw $e;
|
||||
}
|
||||
}
|
||||
}
|
||||
55
app/Models/User.php
Normal file
55
app/Models/User.php
Normal file
@@ -0,0 +1,55 @@
|
||||
<?php
|
||||
|
||||
namespace App\Models;
|
||||
|
||||
use App\Config\Database;
|
||||
|
||||
class User
|
||||
{
|
||||
public static function findByUsername(string $username): ?array
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->prepare('SELECT * FROM users WHERE username = :username');
|
||||
$stmt->execute(['username' => $username]);
|
||||
$result = $stmt->fetch();
|
||||
return $result ?: null;
|
||||
}
|
||||
|
||||
public static function findById(int $id): ?array
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->prepare('SELECT * FROM users WHERE id = :id');
|
||||
$stmt->execute(['id' => $id]);
|
||||
$result = $stmt->fetch();
|
||||
return $result ?: null;
|
||||
}
|
||||
|
||||
public static function create(array $data): array
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->prepare('INSERT INTO users (username, password_hash, role) VALUES (:username, :password_hash, :role)');
|
||||
$stmt->execute([
|
||||
'username' => $data['username'],
|
||||
'password_hash' => password_hash($data['password'], PASSWORD_DEFAULT),
|
||||
'role' => $data['role'] ?? 'user',
|
||||
]);
|
||||
|
||||
$user = self::findById((int) $db->lastInsertId());
|
||||
unset($user['password_hash']);
|
||||
return $user;
|
||||
}
|
||||
|
||||
public static function verifyPassword(string $username, string $password): bool
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->prepare('SELECT password_hash FROM users WHERE username = :username');
|
||||
$stmt->execute(['username' => $username]);
|
||||
$result = $stmt->fetch();
|
||||
|
||||
if (!$result) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return password_verify($password, $result['password_hash']);
|
||||
}
|
||||
}
|
||||
26
app/Services/AIService.php
Normal file
26
app/Services/AIService.php
Normal file
@@ -0,0 +1,26 @@
|
||||
<?php
|
||||
|
||||
namespace App\Services;
|
||||
|
||||
use App\Services\Providers\OpenAIProvider;
|
||||
use App\Services\Providers\ClaudeProvider;
|
||||
use App\Services\Providers\NewAPIProvider;
|
||||
|
||||
class AIService
|
||||
{
|
||||
public static function streamChat(string $providerName, string $model, array $messages, array $options, callable $onChunk): void
|
||||
{
|
||||
$provider = self::getProvider($providerName);
|
||||
$provider::stream($model, $messages, $options, $onChunk);
|
||||
}
|
||||
|
||||
public static function getProvider(string $name): string
|
||||
{
|
||||
return match ($name) {
|
||||
'openai' => OpenAIProvider::class,
|
||||
'claude' => ClaudeProvider::class,
|
||||
'newapi' => NewAPIProvider::class,
|
||||
default => throw new \RuntimeException('不支持的供应商: ' . $name)
|
||||
};
|
||||
}
|
||||
}
|
||||
132
app/Services/Installer.php
Normal file
132
app/Services/Installer.php
Normal file
@@ -0,0 +1,132 @@
|
||||
<?php
|
||||
|
||||
namespace App\Services;
|
||||
|
||||
use App\Config\Database;
|
||||
use App\Models\Personality;
|
||||
|
||||
class Installer
|
||||
{
|
||||
public static function isInstalled(): bool
|
||||
{
|
||||
$configPath = dirname(__DIR__, 2) . '/config/db-config.json';
|
||||
|
||||
if (!file_exists($configPath)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
try {
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->query("SHOW TABLES LIKE 'users'");
|
||||
return $stmt->fetch() !== false;
|
||||
} catch (\Throwable $e) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
public static function runMigration(): void
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
|
||||
$db->exec("CREATE TABLE IF NOT EXISTS users (
|
||||
id INT AUTO_INCREMENT PRIMARY KEY,
|
||||
username VARCHAR(50) UNIQUE NOT NULL,
|
||||
password_hash VARCHAR(255) NOT NULL,
|
||||
role ENUM('admin','user') DEFAULT 'user',
|
||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4");
|
||||
|
||||
$db->exec("CREATE TABLE IF NOT EXISTS sessions (
|
||||
id INT AUTO_INCREMENT PRIMARY KEY,
|
||||
user_id INT NOT NULL,
|
||||
name VARCHAR(100),
|
||||
provider VARCHAR(50),
|
||||
model VARCHAR(50),
|
||||
system_prompt TEXT,
|
||||
personality_id INT NULL,
|
||||
thinking_mode TINYINT(1) DEFAULT 0,
|
||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||
updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
|
||||
INDEX idx_user_id (user_id)
|
||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4");
|
||||
|
||||
$db->exec("CREATE TABLE IF NOT EXISTS messages (
|
||||
id INT AUTO_INCREMENT PRIMARY KEY,
|
||||
session_id INT NOT NULL,
|
||||
role ENUM('user','assistant','system') NOT NULL,
|
||||
content LONGTEXT,
|
||||
file_info JSON,
|
||||
thinking_content LONGTEXT,
|
||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||
INDEX idx_session_id (session_id)
|
||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4");
|
||||
|
||||
$db->exec("CREATE TABLE IF NOT EXISTS config (
|
||||
id INT AUTO_INCREMENT PRIMARY KEY,
|
||||
config_key VARCHAR(50) UNIQUE NOT NULL,
|
||||
config_value LONGTEXT
|
||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4");
|
||||
|
||||
$db->exec("CREATE TABLE IF NOT EXISTS personalities (
|
||||
id INT AUTO_INCREMENT PRIMARY KEY,
|
||||
name VARCHAR(50) NOT NULL,
|
||||
prompt TEXT NOT NULL,
|
||||
description TEXT,
|
||||
icon VARCHAR(50),
|
||||
is_preset TINYINT(1) DEFAULT 0,
|
||||
created_by INT,
|
||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4");
|
||||
}
|
||||
|
||||
public static function seedDefaults(): void
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
|
||||
$presets = [
|
||||
[
|
||||
'name' => '智能助手',
|
||||
'prompt' => '你是一个全能的智能助手,善于回答各种问题,提供准确、有帮助的回答。',
|
||||
'icon' => '🤖',
|
||||
],
|
||||
[
|
||||
'name' => '代码专家',
|
||||
'prompt' => '你是一个专业的编程专家,精通多种编程语言和技术框架,擅长代码编写、调试和优化。',
|
||||
'icon' => '💻',
|
||||
],
|
||||
[
|
||||
'name' => '翻译官',
|
||||
'prompt' => '你是一个专业的翻译官,精通中文、英文、日文等多种语言,提供准确、流畅的翻译服务。',
|
||||
'icon' => '🌐',
|
||||
],
|
||||
[
|
||||
'name' => '写作助手',
|
||||
'prompt' => '你是一个专业的写作助手,擅长各类文体的写作,包括文章、报告、邮件、创意写作等。',
|
||||
'icon' => '✍️',
|
||||
],
|
||||
[
|
||||
'name' => '数学家',
|
||||
'prompt' => '你是一个数学专家,精通各领域的数学知识,善于解答数学问题并提供详细的解题步骤。',
|
||||
'icon' => '🔢',
|
||||
],
|
||||
];
|
||||
|
||||
$stmt = $db->prepare('INSERT INTO personalities (name, prompt, description, icon, is_preset) VALUES (:name, :prompt, :description, :icon, :is_preset)');
|
||||
|
||||
foreach ($presets as $preset) {
|
||||
$stmt->execute([
|
||||
'name' => $preset['name'],
|
||||
'prompt' => $preset['prompt'],
|
||||
'description' => null,
|
||||
'icon' => $preset['icon'],
|
||||
'is_preset' => 1,
|
||||
]);
|
||||
}
|
||||
|
||||
$configStmt = $db->prepare('INSERT INTO config (config_key, config_value) VALUES (:config_key, :config_value) ON DUPLICATE KEY UPDATE config_value = VALUES(config_value)');
|
||||
$configStmt->execute([
|
||||
'config_key' => 'providers',
|
||||
'config_value' => '[]',
|
||||
]);
|
||||
}
|
||||
}
|
||||
0
app/Services/Providers/.gitkeep
Normal file
0
app/Services/Providers/.gitkeep
Normal file
118
app/Services/Providers/ClaudeProvider.php
Normal file
118
app/Services/Providers/ClaudeProvider.php
Normal file
@@ -0,0 +1,118 @@
|
||||
<?php
|
||||
|
||||
namespace App\Services\Providers;
|
||||
|
||||
class ClaudeProvider
|
||||
{
|
||||
public static function stream(string $model, array $messages, array $options, callable $onChunk): void
|
||||
{
|
||||
$provider = $options['provider'];
|
||||
$apiUrl = rtrim($provider['apiUrl'] ?? 'https://api.anthropic.com', '/');
|
||||
$apiKey = $provider['apiKey'] ?? '';
|
||||
$systemPrompt = $options['systemPrompt'] ?? '';
|
||||
$thinkingMode = $options['thinkingMode'] ?? false;
|
||||
|
||||
$url = $apiUrl . '/v1/messages';
|
||||
|
||||
// Claude 的 system 消息通过单独参数传递,从 messages 中排除
|
||||
$claudeMessages = array_values(array_filter($messages, function ($msg) {
|
||||
return $msg['role'] !== 'system';
|
||||
}));
|
||||
|
||||
$bodyData = [
|
||||
'model' => $model,
|
||||
'max_tokens' => $thinkingMode ? 16000 : 4096,
|
||||
'messages' => $claudeMessages,
|
||||
'stream' => true
|
||||
];
|
||||
|
||||
if (!empty($systemPrompt)) {
|
||||
$bodyData['system'] = $systemPrompt;
|
||||
}
|
||||
|
||||
if ($thinkingMode) {
|
||||
$bodyData['thinking'] = [
|
||||
'type' => 'enabled',
|
||||
'budget_tokens' => 10000
|
||||
];
|
||||
}
|
||||
|
||||
$body = json_encode($bodyData);
|
||||
|
||||
$headers = [
|
||||
'x-api-key: ' . $apiKey,
|
||||
'anthropic-version: 2023-06-01',
|
||||
'Content-Type: application/json',
|
||||
'Accept: text/event-stream'
|
||||
];
|
||||
|
||||
$ch = curl_init($url);
|
||||
curl_setopt_array($ch, [
|
||||
CURLOPT_POST => true,
|
||||
CURLOPT_POSTFIELDS => $body,
|
||||
CURLOPT_HTTPHEADER => $headers,
|
||||
CURLOPT_TIMEOUT => 120,
|
||||
CURLOPT_RETURNTRANSFER => false,
|
||||
CURLOPT_SSL_VERIFYPEER => true,
|
||||
CURLOPT_WRITEFUNCTION => function ($ch, $data) use ($onChunk) {
|
||||
$lines = explode("\n", $data);
|
||||
$currentEvent = '';
|
||||
|
||||
foreach ($lines as $line) {
|
||||
$line = trim($line);
|
||||
if ($line === '') {
|
||||
$currentEvent = '';
|
||||
continue;
|
||||
}
|
||||
if (str_starts_with($line, 'event: ')) {
|
||||
$currentEvent = substr($line, 7);
|
||||
continue;
|
||||
}
|
||||
if (str_starts_with($line, 'data: ')) {
|
||||
$payload = substr($line, 6);
|
||||
$json = json_decode($payload, true);
|
||||
|
||||
if (!$json) {
|
||||
continue;
|
||||
}
|
||||
|
||||
switch ($currentEvent) {
|
||||
case 'content_block_delta':
|
||||
if (isset($json['delta'])) {
|
||||
$delta = $json['delta'];
|
||||
if (isset($delta['type']) && $delta['type'] === 'thinking_delta' && isset($delta['thinking'])) {
|
||||
$onChunk($delta['thinking'], 'thinking');
|
||||
} elseif (isset($delta['text'])) {
|
||||
$onChunk($delta['text'], 'content');
|
||||
}
|
||||
}
|
||||
break;
|
||||
case 'message_stop':
|
||||
return strlen($data);
|
||||
case 'error':
|
||||
if (isset($json['error']['message'])) {
|
||||
throw new \RuntimeException('Claude API错误: ' . $json['error']['message']);
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
return strlen($data);
|
||||
}
|
||||
]);
|
||||
|
||||
$result = curl_exec($ch);
|
||||
if ($result === false) {
|
||||
$error = curl_error($ch);
|
||||
curl_close($ch);
|
||||
throw new \RuntimeException('API请求失败: ' . $error);
|
||||
}
|
||||
|
||||
$httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
|
||||
curl_close($ch);
|
||||
|
||||
if ($httpCode >= 400) {
|
||||
throw new \RuntimeException('API返回错误,状态码: ' . $httpCode);
|
||||
}
|
||||
}
|
||||
}
|
||||
11
app/Services/Providers/NewAPIProvider.php
Normal file
11
app/Services/Providers/NewAPIProvider.php
Normal file
@@ -0,0 +1,11 @@
|
||||
<?php
|
||||
|
||||
namespace App\Services\Providers;
|
||||
|
||||
class NewAPIProvider
|
||||
{
|
||||
public static function stream(string $model, array $messages, array $options, callable $onChunk): void
|
||||
{
|
||||
OpenAIProvider::stream($model, $messages, $options, $onChunk);
|
||||
}
|
||||
}
|
||||
77
app/Services/Providers/OpenAIProvider.php
Normal file
77
app/Services/Providers/OpenAIProvider.php
Normal file
@@ -0,0 +1,77 @@
|
||||
<?php
|
||||
|
||||
namespace App\Services\Providers;
|
||||
|
||||
class OpenAIProvider
|
||||
{
|
||||
public static function stream(string $model, array $messages, array $options, callable $onChunk): void
|
||||
{
|
||||
$provider = $options['provider'];
|
||||
$apiUrl = rtrim($provider['apiUrl'] ?? 'https://api.openai.com', '/');
|
||||
$apiKey = $provider['apiKey'] ?? '';
|
||||
$systemPrompt = $options['systemPrompt'] ?? '';
|
||||
|
||||
$url = $apiUrl . '/v1/chat/completions';
|
||||
|
||||
if (!empty($systemPrompt)) {
|
||||
array_unshift($messages, ['role' => 'system', 'content' => $systemPrompt]);
|
||||
}
|
||||
|
||||
$body = json_encode([
|
||||
'model' => $model,
|
||||
'messages' => $messages,
|
||||
'stream' => true
|
||||
]);
|
||||
|
||||
$ch = curl_init($url);
|
||||
curl_setopt_array($ch, [
|
||||
CURLOPT_POST => true,
|
||||
CURLOPT_POSTFIELDS => $body,
|
||||
CURLOPT_HTTPHEADER => [
|
||||
'Authorization: Bearer ' . $apiKey,
|
||||
'Content-Type: application/json',
|
||||
'Accept: text/event-stream'
|
||||
],
|
||||
CURLOPT_TIMEOUT => 120,
|
||||
CURLOPT_RETURNTRANSFER => false,
|
||||
CURLOPT_SSL_VERIFYPEER => true,
|
||||
CURLOPT_WRITEFUNCTION => function ($ch, $data) use ($onChunk) {
|
||||
$lines = explode("\n", $data);
|
||||
foreach ($lines as $line) {
|
||||
$line = trim($line);
|
||||
if ($line === '') {
|
||||
continue;
|
||||
}
|
||||
if (str_starts_with($line, 'data: ')) {
|
||||
$payload = substr($line, 6);
|
||||
if ($payload === '[DONE]') {
|
||||
return strlen($data);
|
||||
}
|
||||
$json = json_decode($payload, true);
|
||||
if ($json && isset($json['choices'][0]['delta']['content'])) {
|
||||
$content = $json['choices'][0]['delta']['content'];
|
||||
if ($content !== '') {
|
||||
$onChunk($content);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
return strlen($data);
|
||||
}
|
||||
]);
|
||||
|
||||
$result = curl_exec($ch);
|
||||
if ($result === false) {
|
||||
$error = curl_error($ch);
|
||||
curl_close($ch);
|
||||
throw new \RuntimeException('API请求失败: ' . $error);
|
||||
}
|
||||
|
||||
$httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
|
||||
curl_close($ch);
|
||||
|
||||
if ($httpCode >= 400) {
|
||||
throw new \RuntimeException('API返回错误,状态码: ' . $httpCode);
|
||||
}
|
||||
}
|
||||
}
|
||||
164
app/Views/chat.php
Normal file
164
app/Views/chat.php
Normal file
@@ -0,0 +1,164 @@
|
||||
<div class="chat-layout" id="chatLayout">
|
||||
<!-- 左侧边栏 -->
|
||||
<div class="sidebar" id="sidebar">
|
||||
<div class="sidebar-header">
|
||||
<button class="btn btn-primary btn-sm" onclick="SessionManager.createSession()">+ 新建会话</button>
|
||||
</div>
|
||||
<div class="session-list" id="sessionList">
|
||||
<!-- 由 SessionManager.renderSessionList() 动态渲染 -->
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 主聊天区域 -->
|
||||
<div class="chat-main">
|
||||
<!-- 顶部工具栏 -->
|
||||
<div class="toolbar">
|
||||
<button class="toggle-sidebar" onclick="toggleSidebar()" title="切换侧边栏">☰</button>
|
||||
|
||||
<select id="providerSelect" onchange="onProviderChange()">
|
||||
<option value="">选择供应商</option>
|
||||
<!-- 由 loadProviders() 动态填充 -->
|
||||
</select>
|
||||
|
||||
<select id="modelSelect">
|
||||
<option value="">选择模型</option>
|
||||
<!-- 由 onProviderChange() 动态填充 -->
|
||||
</select>
|
||||
|
||||
<label class="toggle-switch" title="思考模式">
|
||||
<input type="checkbox" id="thinkingMode">
|
||||
<span class="slider"></span>
|
||||
</label>
|
||||
<span style="font-size:12px;color:var(--text-secondary)">思考</span>
|
||||
|
||||
<select id="personalitySelect">
|
||||
<option value="">默认人格</option>
|
||||
<!-- 由 loadPersonalities() 动态填充 -->
|
||||
</select>
|
||||
|
||||
<a href="/config.php" class="btn btn-secondary btn-sm" style="margin-left:auto;">⚙️ 设置</a>
|
||||
</div>
|
||||
|
||||
<!-- 消息列表 -->
|
||||
<div class="messages-container" id="messagesContainer">
|
||||
<div class="empty-state">
|
||||
<h3>开始新的对话</h3>
|
||||
<p>输入消息开始聊天</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 底部输入区域 -->
|
||||
<div class="input-area">
|
||||
<div class="file-preview" id="filePreview"></div>
|
||||
<div class="input-wrapper">
|
||||
<button id="uploadBtn" title="上传文件" style="background:none;border:none;color:var(--text-secondary);cursor:pointer;font-size:18px;">📎</button>
|
||||
<textarea id="messageInput" rows="1" placeholder="输入消息,Ctrl+Enter 发送..."></textarea>
|
||||
<button id="sendBtn">发送</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<script>
|
||||
// 切换侧边栏
|
||||
function toggleSidebar() {
|
||||
document.getElementById('sidebar').classList.toggle('collapsed');
|
||||
}
|
||||
|
||||
// 供应商/模型数据缓存
|
||||
let providersData = [];
|
||||
let personalitiesData = [];
|
||||
|
||||
// 加载供应商配置
|
||||
async function loadProviders() {
|
||||
try {
|
||||
const res = await api.get('/config');
|
||||
providersData = res.data.providers || [];
|
||||
const select = document.getElementById('providerSelect');
|
||||
select.innerHTML = '<option value="">选择供应商</option>';
|
||||
providersData.forEach((p, i) => {
|
||||
if (p.enabled) {
|
||||
const option = document.createElement('option');
|
||||
option.value = i;
|
||||
option.textContent = p.name;
|
||||
select.appendChild(option);
|
||||
}
|
||||
});
|
||||
// 如果只有一个供应商,自动选择
|
||||
if (providersData.filter(p => p.enabled).length === 1) {
|
||||
select.value = providersData.findIndex(p => p.enabled);
|
||||
onProviderChange();
|
||||
}
|
||||
} catch (err) {
|
||||
console.error('加载供应商配置失败:', err);
|
||||
}
|
||||
}
|
||||
|
||||
// 供应商切换时更新模型列表
|
||||
function onProviderChange() {
|
||||
const index = document.getElementById('providerSelect').value;
|
||||
const modelSelect = document.getElementById('modelSelect');
|
||||
modelSelect.innerHTML = '<option value="">选择模型</option>';
|
||||
|
||||
if (index !== '' && providersData[index]) {
|
||||
const provider = providersData[index];
|
||||
(provider.models || []).forEach(m => {
|
||||
const option = document.createElement('option');
|
||||
option.value = m;
|
||||
option.textContent = m;
|
||||
modelSelect.appendChild(option);
|
||||
});
|
||||
// 如果只有一个模型,自动选择
|
||||
if (provider.models && provider.models.length === 1) {
|
||||
modelSelect.value = provider.models[0];
|
||||
}
|
||||
// 如果有默认模型,自动选择
|
||||
if (provider.defaultModel) {
|
||||
modelSelect.value = provider.defaultModel;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 加载人格列表
|
||||
async function loadPersonalities() {
|
||||
try {
|
||||
const res = await api.get('/personalities');
|
||||
personalitiesData = res.data || [];
|
||||
const select = document.getElementById('personalitySelect');
|
||||
select.innerHTML = '<option value="">默认人格</option>';
|
||||
personalitiesData.forEach(p => {
|
||||
const option = document.createElement('option');
|
||||
option.value = p.id;
|
||||
option.textContent = (p.icon || '🤖') + ' ' + p.name;
|
||||
select.appendChild(option);
|
||||
});
|
||||
} catch (err) {
|
||||
console.error('加载人格列表失败:', err);
|
||||
}
|
||||
}
|
||||
|
||||
// 页面初始化
|
||||
(async function() {
|
||||
// 检查认证
|
||||
const token = Storage.getToken();
|
||||
if (!token) {
|
||||
window.location.href = '/login.php';
|
||||
return;
|
||||
}
|
||||
|
||||
// 初始化聊天管理器
|
||||
ChatManager.init();
|
||||
|
||||
// 加载数据
|
||||
await Promise.all([
|
||||
loadProviders(),
|
||||
loadPersonalities(),
|
||||
SessionManager.loadSessions()
|
||||
]);
|
||||
|
||||
// 如果有会话,自动选择第一个
|
||||
if (SessionManager.sessions.length > 0) {
|
||||
await SessionManager.switchSession(SessionManager.sessions[0].id);
|
||||
}
|
||||
})();
|
||||
</script>
|
||||
249
app/Views/config.php
Normal file
249
app/Views/config.php
Normal file
@@ -0,0 +1,249 @@
|
||||
<div class="config-container">
|
||||
<div class="config-header">
|
||||
<h1>⚙️ 系统配置</h1>
|
||||
<a href="/chat.php" class="btn btn-secondary">← 返回聊天</a>
|
||||
</div>
|
||||
|
||||
<!-- 供应商管理 -->
|
||||
<div class="config-section">
|
||||
<h2>AI 供应商管理</h2>
|
||||
<div id="providerList">
|
||||
<!-- 由 JS 动态渲染 -->
|
||||
</div>
|
||||
<button class="btn btn-secondary" onclick="ConfigPage.addProvider()" style="margin-top:12px;">+ 添加供应商</button>
|
||||
</div>
|
||||
|
||||
<!-- 人格管理 -->
|
||||
<div class="config-section">
|
||||
<h2>人格管理</h2>
|
||||
<div id="personalityList">
|
||||
<!-- 由 JS 动态渲染 -->
|
||||
</div>
|
||||
<h3 style="margin-top:20px;">添加自定义人格</h3>
|
||||
<div class="form-group">
|
||||
<label>名称</label>
|
||||
<input type="text" id="newPersonalityName" placeholder="人格名称">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>图标(Emoji)</label>
|
||||
<input type="text" id="newPersonalityIcon" placeholder="如:🤖" maxlength="2">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>提示词</label>
|
||||
<textarea id="newPersonalityPrompt" rows="3" placeholder="人格的系统提示词"></textarea>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>描述</label>
|
||||
<input type="text" id="newPersonalityDesc" placeholder="简短描述">
|
||||
</div>
|
||||
<button class="btn btn-primary" onclick="ConfigPage.createPersonality()">添加人格</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div id="configMessage"></div>
|
||||
|
||||
<script>
|
||||
const ConfigPage = {
|
||||
providers: [],
|
||||
personalities: [],
|
||||
|
||||
async init() {
|
||||
// 检查认证和管理员权限
|
||||
const token = Storage.getToken();
|
||||
if (!token) {
|
||||
window.location.href = '/login.php';
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
// 验证管理员权限
|
||||
const userRes = await api.get('/auth/me');
|
||||
if (userRes.data.role !== 'admin') {
|
||||
window.location.href = '/chat.php';
|
||||
return;
|
||||
}
|
||||
} catch (err) {
|
||||
window.location.href = '/login.php';
|
||||
return;
|
||||
}
|
||||
|
||||
await this.loadProviders();
|
||||
await this.loadPersonalities();
|
||||
},
|
||||
|
||||
async loadProviders() {
|
||||
try {
|
||||
const res = await api.get('/config');
|
||||
this.providers = res.data.providers || [];
|
||||
this.renderProviders();
|
||||
} catch (err) {
|
||||
this.showMessage('加载供应商配置失败: ' + err.message, 'error');
|
||||
}
|
||||
},
|
||||
|
||||
renderProviders() {
|
||||
const list = document.getElementById('providerList');
|
||||
if (this.providers.length === 0) {
|
||||
list.innerHTML = '<p style="color:var(--text-secondary)">暂无供应商配置</p>';
|
||||
return;
|
||||
}
|
||||
|
||||
list.innerHTML = this.providers.map((p, i) => `
|
||||
<div class="provider-item" style="background:var(--bg-secondary);padding:16px;border-radius:var(--radius);margin-bottom:12px;">
|
||||
<div style="display:flex;justify-content:space-between;align-items:center;margin-bottom:12px;">
|
||||
<strong>${this.escapeHtml(p.name)}</strong>
|
||||
<div>
|
||||
<label class="toggle-switch">
|
||||
<input type="checkbox" ${p.enabled ? 'checked' : ''} onchange="ConfigPage.toggleProvider(${i}, this.checked)">
|
||||
<span class="slider"></span>
|
||||
</label>
|
||||
<button class="btn btn-danger btn-sm" onclick="ConfigPage.deleteProvider(${i})" style="margin-left:8px;">删除</button>
|
||||
</div>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>API URL</label>
|
||||
<input type="text" value="${this.escapeHtml(p.apiUrl || '')}" onchange="ConfigPage.updateProvider(${i}, 'apiUrl', this.value)">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>API Key</label>
|
||||
<input type="password" value="${this.escapeHtml(p.apiKey || '')}" onchange="ConfigPage.updateProvider(${i}, 'apiKey', this.value)" placeholder="API 密钥">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>可用模型(逗号分隔)</label>
|
||||
<input type="text" value="${this.escapeHtml((p.models || []).join(', '))}" onchange="ConfigPage.updateProviderModels(${i}, this.value)">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>供应商类型</label>
|
||||
<select onchange="ConfigPage.updateProvider(${i}, 'type', this.value)">
|
||||
<option value="newapi" ${p.type === 'newapi' ? 'selected' : ''}>OpenAI 兼容</option>
|
||||
<option value="openai" ${p.type === 'openai' ? 'selected' : ''}>OpenAI 官方</option>
|
||||
<option value="claude" ${p.type === 'claude' ? 'selected' : ''}>Claude (Anthropic)</option>
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
`).join('');
|
||||
},
|
||||
|
||||
addProvider() {
|
||||
this.providers.push({
|
||||
name: '新供应商',
|
||||
apiUrl: '',
|
||||
apiKey: '',
|
||||
models: [],
|
||||
type: 'newapi',
|
||||
enabled: true
|
||||
});
|
||||
this.renderProviders();
|
||||
this.saveProviders();
|
||||
},
|
||||
|
||||
updateProvider(index, field, value) {
|
||||
this.providers[index][field] = value;
|
||||
this.saveProviders();
|
||||
},
|
||||
|
||||
updateProviderModels(index, value) {
|
||||
this.providers[index].models = value.split(',').map(m => m.trim()).filter(m => m);
|
||||
this.saveProviders();
|
||||
},
|
||||
|
||||
toggleProvider(index, enabled) {
|
||||
this.providers[index].enabled = enabled;
|
||||
this.saveProviders();
|
||||
},
|
||||
|
||||
deleteProvider(index) {
|
||||
if (!confirm('确定要删除供应商 "' + this.providers[index].name + '" 吗?')) return;
|
||||
this.providers.splice(index, 1);
|
||||
this.renderProviders();
|
||||
this.saveProviders();
|
||||
},
|
||||
|
||||
async saveProviders() {
|
||||
try {
|
||||
await api.put('/config', { providers: this.providers });
|
||||
this.showMessage('供应商配置已保存', 'success');
|
||||
} catch (err) {
|
||||
this.showMessage('保存失败: ' + err.message, 'error');
|
||||
}
|
||||
},
|
||||
|
||||
async loadPersonalities() {
|
||||
try {
|
||||
const res = await api.get('/personalities');
|
||||
this.personalities = res.data || [];
|
||||
this.renderPersonalities();
|
||||
} catch (err) {
|
||||
this.showMessage('加载人格列表失败: ' + err.message, 'error');
|
||||
}
|
||||
},
|
||||
|
||||
renderPersonalities() {
|
||||
const list = document.getElementById('personalityList');
|
||||
if (this.personalities.length === 0) {
|
||||
list.innerHTML = '<p style="color:var(--text-secondary)">暂无人格</p>';
|
||||
return;
|
||||
}
|
||||
|
||||
list.innerHTML = this.personalities.map(p => `
|
||||
<div style="display:flex;justify-content:space-between;align-items:center;padding:12px;background:var(--bg-secondary);border-radius:var(--radius);margin-bottom:8px;">
|
||||
<div>
|
||||
<span>${p.icon || '🤖'} ${this.escapeHtml(p.name)}</span>
|
||||
${p.is_preset ? '<span style="color:var(--warning);font-size:12px;margin-left:8px;">预设</span>' : '<span style="color:var(--success);font-size:12px;margin-left:8px;">自定义</span>'}
|
||||
</div>
|
||||
${!p.is_preset ? '<button class="btn btn-danger btn-sm" onclick="ConfigPage.deletePersonality(' + p.id + ')">删除</button>' : ''}
|
||||
</div>
|
||||
`).join('');
|
||||
},
|
||||
|
||||
async createPersonality() {
|
||||
const name = document.getElementById('newPersonalityName').value.trim();
|
||||
const prompt = document.getElementById('newPersonalityPrompt').value.trim();
|
||||
const icon = document.getElementById('newPersonalityIcon').value.trim();
|
||||
const description = document.getElementById('newPersonalityDesc').value.trim();
|
||||
|
||||
if (!name || !prompt) {
|
||||
this.showMessage('名称和提示词不能为空', 'error');
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
await api.post('/personalities', { name, prompt, icon, description });
|
||||
this.showMessage('人格创建成功', 'success');
|
||||
// 清空表单
|
||||
document.getElementById('newPersonalityName').value = '';
|
||||
document.getElementById('newPersonalityPrompt').value = '';
|
||||
document.getElementById('newPersonalityIcon').value = '';
|
||||
document.getElementById('newPersonalityDesc').value = '';
|
||||
await this.loadPersonalities();
|
||||
} catch (err) {
|
||||
this.showMessage('创建失败: ' + err.message, 'error');
|
||||
}
|
||||
},
|
||||
|
||||
async deletePersonality(id) {
|
||||
if (!confirm('确定要删除这个人格吗?')) return;
|
||||
try {
|
||||
await api.delete('/personalities/' + id);
|
||||
this.showMessage('人格已删除', 'success');
|
||||
await this.loadPersonalities();
|
||||
} catch (err) {
|
||||
this.showMessage('删除失败: ' + err.message, 'error');
|
||||
}
|
||||
},
|
||||
|
||||
showMessage(text, type) {
|
||||
const el = document.getElementById('configMessage');
|
||||
el.innerHTML = `<div class="alert alert-${type === 'error' ? 'error' : 'success'}" style="position:fixed;bottom:20px;right:20px;z-index:1000;min-width:200px;">${this.escapeHtml(text)}</div>`;
|
||||
setTimeout(() => { el.innerHTML = ''; }, 3000);
|
||||
},
|
||||
|
||||
escapeHtml(text) {
|
||||
const div = document.createElement('div');
|
||||
div.textContent = text;
|
||||
return div.innerHTML;
|
||||
}
|
||||
};
|
||||
|
||||
ConfigPage.init();
|
||||
</script>
|
||||
0
app/Views/layout/.gitkeep
Normal file
0
app/Views/layout/.gitkeep
Normal file
10
app/Views/layout/footer.php
Normal file
10
app/Views/layout/footer.php
Normal file
@@ -0,0 +1,10 @@
|
||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/highlight.min.js"></script>
|
||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/marked.js/12.0.0/marked.min.js"></script>
|
||||
<script src="/assets/js/storage.js"></script>
|
||||
<script src="/assets/js/api.js"></script>
|
||||
<script src="/assets/js/markdown.js"></script>
|
||||
<script src="/assets/js/session.js"></script>
|
||||
<script src="/assets/js/upload.js"></script>
|
||||
<script src="/assets/js/chat.js"></script>
|
||||
</body>
|
||||
</html>
|
||||
12
app/Views/layout/header.php
Normal file
12
app/Views/layout/header.php
Normal file
@@ -0,0 +1,12 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="zh-CN">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<title>AI Chat</title>
|
||||
<link rel="stylesheet" href="/assets/css/style.css">
|
||||
<link rel="stylesheet" href="/assets/css/chat.css">
|
||||
<link rel="stylesheet" href="/assets/css/markdown.css">
|
||||
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/styles/github-dark.min.css">
|
||||
</head>
|
||||
<body>
|
||||
69
app/Views/login.php
Normal file
69
app/Views/login.php
Normal file
@@ -0,0 +1,69 @@
|
||||
<div class="login-container">
|
||||
<div class="login-card">
|
||||
<h1>🤖 AI Chat</h1>
|
||||
<div id="loginError" class="alert alert-error" style="display:none;"></div>
|
||||
<form id="loginForm">
|
||||
<div class="form-group">
|
||||
<label for="username">用户名</label>
|
||||
<input type="text" id="username" name="username" required autocomplete="username" placeholder="请输入用户名">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label for="password">密码</label>
|
||||
<input type="password" id="password" name="password" required autocomplete="current-password" placeholder="请输入密码">
|
||||
</div>
|
||||
<button type="submit" class="btn btn-primary btn-block" id="loginBtn">登录</button>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
<script>
|
||||
// 页面加载时检查是否已登录
|
||||
(function() {
|
||||
const token = localStorage.getItem('token');
|
||||
if (token) {
|
||||
window.location.href = '/chat.php';
|
||||
}
|
||||
})();
|
||||
|
||||
// 登录表单提交
|
||||
document.getElementById('loginForm').addEventListener('submit', async function(e) {
|
||||
e.preventDefault();
|
||||
const btn = document.getElementById('loginBtn');
|
||||
const errorEl = document.getElementById('loginError');
|
||||
const username = document.getElementById('username').value.trim();
|
||||
const password = document.getElementById('password').value;
|
||||
|
||||
if (!username || !password) {
|
||||
errorEl.textContent = '请输入用户名和密码';
|
||||
errorEl.style.display = 'block';
|
||||
return;
|
||||
}
|
||||
|
||||
btn.disabled = true;
|
||||
btn.textContent = '登录中...';
|
||||
errorEl.style.display = 'none';
|
||||
|
||||
try {
|
||||
const response = await fetch('/api/auth/login', {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify({ username, password })
|
||||
});
|
||||
|
||||
const data = await response.json();
|
||||
|
||||
if (data.success) {
|
||||
localStorage.setItem('token', data.data.token);
|
||||
window.location.href = '/chat.php';
|
||||
} else {
|
||||
errorEl.textContent = data.message || '登录失败';
|
||||
errorEl.style.display = 'block';
|
||||
}
|
||||
} catch (err) {
|
||||
errorEl.textContent = '网络错误,请稍后重试';
|
||||
errorEl.style.display = 'block';
|
||||
} finally {
|
||||
btn.disabled = false;
|
||||
btn.textContent = '登录';
|
||||
}
|
||||
});
|
||||
</script>
|
||||
0
assets/css/.gitkeep
Normal file
0
assets/css/.gitkeep
Normal file
345
assets/css/chat.css
Normal file
345
assets/css/chat.css
Normal file
@@ -0,0 +1,345 @@
|
||||
/* 聊天布局 */
|
||||
.chat-layout {
|
||||
display: flex;
|
||||
height: 100vh;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
/* 侧边栏 */
|
||||
.sidebar {
|
||||
width: var(--sidebar-width);
|
||||
background: var(--bg-secondary);
|
||||
border-right: 1px solid var(--border-color);
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
transition: transform 0.3s;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.sidebar.collapsed {
|
||||
transform: translateX(-100%);
|
||||
width: 0;
|
||||
border: none;
|
||||
}
|
||||
|
||||
.sidebar-header {
|
||||
padding: 16px;
|
||||
border-bottom: 1px solid var(--border-color);
|
||||
display: flex;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.sidebar-header .btn {
|
||||
flex: 1;
|
||||
}
|
||||
|
||||
.session-list {
|
||||
flex: 1;
|
||||
overflow-y: auto;
|
||||
padding: 8px;
|
||||
}
|
||||
|
||||
.session-item {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
padding: 12px;
|
||||
border-radius: var(--radius);
|
||||
cursor: pointer;
|
||||
transition: background 0.2s;
|
||||
margin-bottom: 4px;
|
||||
}
|
||||
|
||||
.session-item:hover {
|
||||
background: var(--bg-card);
|
||||
}
|
||||
|
||||
.session-item.active {
|
||||
background: var(--primary);
|
||||
color: white;
|
||||
}
|
||||
|
||||
.session-item .session-name {
|
||||
flex: 1;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
font-size: 14px;
|
||||
}
|
||||
|
||||
.session-item .session-delete {
|
||||
opacity: 0;
|
||||
background: none;
|
||||
border: none;
|
||||
color: inherit;
|
||||
cursor: pointer;
|
||||
padding: 4px;
|
||||
font-size: 16px;
|
||||
}
|
||||
|
||||
.session-item:hover .session-delete {
|
||||
opacity: 1;
|
||||
}
|
||||
|
||||
/* 主区域 */
|
||||
.chat-main {
|
||||
flex: 1;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
min-width: 0;
|
||||
}
|
||||
|
||||
/* 工具栏 */
|
||||
.toolbar {
|
||||
height: var(--toolbar-height);
|
||||
background: var(--bg-secondary);
|
||||
border-bottom: 1px solid var(--border-color);
|
||||
display: flex;
|
||||
align-items: center;
|
||||
padding: 0 16px;
|
||||
gap: 12px;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.toolbar select {
|
||||
padding: 6px 10px;
|
||||
background: var(--bg-input);
|
||||
border: 1px solid var(--border-color);
|
||||
border-radius: var(--radius);
|
||||
color: var(--text-primary);
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.toolbar .toggle-sidebar {
|
||||
background: none;
|
||||
border: none;
|
||||
color: var(--text-primary);
|
||||
cursor: pointer;
|
||||
font-size: 20px;
|
||||
padding: 4px;
|
||||
}
|
||||
|
||||
/* 思考模式开关 */
|
||||
.toggle-switch {
|
||||
position: relative;
|
||||
display: inline-block;
|
||||
width: 40px;
|
||||
height: 20px;
|
||||
}
|
||||
|
||||
.toggle-switch input {
|
||||
opacity: 0;
|
||||
width: 0;
|
||||
height: 0;
|
||||
}
|
||||
|
||||
.toggle-switch .slider {
|
||||
position: absolute;
|
||||
cursor: pointer;
|
||||
top: 0; left: 0; right: 0; bottom: 0;
|
||||
background-color: var(--border-color);
|
||||
border-radius: 20px;
|
||||
transition: 0.3s;
|
||||
}
|
||||
|
||||
.toggle-switch .slider:before {
|
||||
position: absolute;
|
||||
content: "";
|
||||
height: 16px;
|
||||
width: 16px;
|
||||
left: 2px;
|
||||
bottom: 2px;
|
||||
background-color: white;
|
||||
border-radius: 50%;
|
||||
transition: 0.3s;
|
||||
}
|
||||
|
||||
.toggle-switch input:checked + .slider {
|
||||
background-color: var(--primary);
|
||||
}
|
||||
|
||||
.toggle-switch input:checked + .slider:before {
|
||||
transform: translateX(20px);
|
||||
}
|
||||
|
||||
/* 消息区域 */
|
||||
.messages-container {
|
||||
flex: 1;
|
||||
overflow-y: auto;
|
||||
padding: 20px;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 16px;
|
||||
}
|
||||
|
||||
.message {
|
||||
max-width: 80%;
|
||||
padding: 12px 16px;
|
||||
border-radius: var(--radius);
|
||||
line-height: 1.6;
|
||||
font-size: 14px;
|
||||
word-wrap: break-word;
|
||||
}
|
||||
|
||||
.message.user {
|
||||
align-self: flex-end;
|
||||
background: var(--primary);
|
||||
color: white;
|
||||
border-bottom-right-radius: 2px;
|
||||
}
|
||||
|
||||
.message.assistant {
|
||||
align-self: flex-start;
|
||||
background: var(--bg-card);
|
||||
border-bottom-left-radius: 2px;
|
||||
}
|
||||
|
||||
.message .message-role {
|
||||
font-size: 12px;
|
||||
color: var(--text-secondary);
|
||||
margin-bottom: 4px;
|
||||
}
|
||||
|
||||
/* 思考内容折叠 */
|
||||
.thinking-toggle {
|
||||
font-size: 12px;
|
||||
color: var(--primary);
|
||||
cursor: pointer;
|
||||
margin-top: 8px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 4px;
|
||||
}
|
||||
|
||||
.thinking-content {
|
||||
display: none;
|
||||
margin-top: 8px;
|
||||
padding: 12px;
|
||||
background: rgba(0,0,0,0.2);
|
||||
border-radius: var(--radius);
|
||||
font-size: 13px;
|
||||
color: var(--text-secondary);
|
||||
border-left: 3px solid var(--primary);
|
||||
}
|
||||
|
||||
.thinking-content.expanded {
|
||||
display: block;
|
||||
}
|
||||
|
||||
/* 打字机光标 */
|
||||
.typing-cursor::after {
|
||||
content: '▊';
|
||||
animation: blink 1s step-end infinite;
|
||||
color: var(--primary);
|
||||
}
|
||||
|
||||
@keyframes blink {
|
||||
50% { opacity: 0; }
|
||||
}
|
||||
|
||||
/* 输入区域 */
|
||||
.input-area {
|
||||
padding: 16px;
|
||||
background: var(--bg-secondary);
|
||||
border-top: 1px solid var(--border-color);
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.input-wrapper {
|
||||
display: flex;
|
||||
align-items: flex-end;
|
||||
gap: 8px;
|
||||
background: var(--bg-input);
|
||||
border: 1px solid var(--border-color);
|
||||
border-radius: var(--radius);
|
||||
padding: 8px 12px;
|
||||
}
|
||||
|
||||
.input-wrapper textarea {
|
||||
flex: 1;
|
||||
background: none;
|
||||
border: none;
|
||||
color: var(--text-primary);
|
||||
font-size: 14px;
|
||||
line-height: 1.5;
|
||||
resize: none;
|
||||
max-height: 120px;
|
||||
min-height: 24px;
|
||||
font-family: var(--font-family);
|
||||
}
|
||||
|
||||
.input-wrapper textarea:focus {
|
||||
outline: none;
|
||||
}
|
||||
|
||||
.input-wrapper button {
|
||||
background: var(--primary);
|
||||
border: none;
|
||||
color: white;
|
||||
padding: 8px 12px;
|
||||
border-radius: var(--radius);
|
||||
cursor: pointer;
|
||||
font-size: 14px;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.input-wrapper button:hover {
|
||||
background: var(--primary-hover);
|
||||
}
|
||||
|
||||
.input-wrapper button:disabled {
|
||||
opacity: 0.5;
|
||||
cursor: not-allowed;
|
||||
}
|
||||
|
||||
/* 文件预览 */
|
||||
.file-preview {
|
||||
display: flex;
|
||||
gap: 8px;
|
||||
padding: 8px 0;
|
||||
flex-wrap: wrap;
|
||||
}
|
||||
|
||||
.file-preview-item {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 6px;
|
||||
padding: 4px 8px;
|
||||
background: var(--bg-card);
|
||||
border-radius: 4px;
|
||||
font-size: 12px;
|
||||
}
|
||||
|
||||
.file-preview-item .remove-file {
|
||||
cursor: pointer;
|
||||
color: var(--danger);
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
/* 空状态 */
|
||||
.empty-state {
|
||||
text-align: center;
|
||||
color: var(--text-secondary);
|
||||
padding: 60px 20px;
|
||||
}
|
||||
|
||||
.empty-state h3 {
|
||||
margin-bottom: 8px;
|
||||
}
|
||||
|
||||
/* 响应式 */
|
||||
@media (max-width: 768px) {
|
||||
.sidebar {
|
||||
position: fixed;
|
||||
z-index: 100;
|
||||
height: 100vh;
|
||||
}
|
||||
.message {
|
||||
max-width: 95%;
|
||||
}
|
||||
.toolbar {
|
||||
flex-wrap: wrap;
|
||||
height: auto;
|
||||
padding: 8px 12px;
|
||||
}
|
||||
}
|
||||
127
assets/css/markdown.css
Normal file
127
assets/css/markdown.css
Normal file
@@ -0,0 +1,127 @@
|
||||
/* 消息中的 Markdown 内容 */
|
||||
.message.assistant .markdown-body {
|
||||
font-size: 14px;
|
||||
line-height: 1.7;
|
||||
}
|
||||
|
||||
.markdown-body p {
|
||||
margin-bottom: 12px;
|
||||
}
|
||||
|
||||
.markdown-body p:last-child {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
|
||||
.markdown-body h1, .markdown-body h2, .markdown-body h3,
|
||||
.markdown-body h4, .markdown-body h5, .markdown-body h6 {
|
||||
margin-top: 16px;
|
||||
margin-bottom: 8px;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.markdown-body h1 { font-size: 1.4em; }
|
||||
.markdown-body h2 { font-size: 1.3em; }
|
||||
.markdown-body h3 { font-size: 1.2em; }
|
||||
|
||||
.markdown-body ul, .markdown-body ol {
|
||||
margin: 8px 0;
|
||||
padding-left: 24px;
|
||||
}
|
||||
|
||||
.markdown-body li {
|
||||
margin-bottom: 4px;
|
||||
}
|
||||
|
||||
.markdown-body code {
|
||||
background: rgba(255,255,255,0.1);
|
||||
padding: 2px 6px;
|
||||
border-radius: 3px;
|
||||
font-size: 0.9em;
|
||||
font-family: 'Consolas', 'Monaco', monospace;
|
||||
}
|
||||
|
||||
.markdown-body pre {
|
||||
background: #1e1e2e;
|
||||
border-radius: var(--radius);
|
||||
padding: 0;
|
||||
margin: 12px 0;
|
||||
overflow: hidden;
|
||||
position: relative;
|
||||
}
|
||||
|
||||
.markdown-body pre code {
|
||||
display: block;
|
||||
padding: 16px;
|
||||
background: none;
|
||||
overflow-x: auto;
|
||||
font-size: 13px;
|
||||
line-height: 1.5;
|
||||
}
|
||||
|
||||
.markdown-body pre .copy-btn {
|
||||
position: absolute;
|
||||
top: 8px;
|
||||
right: 8px;
|
||||
background: rgba(255,255,255,0.1);
|
||||
border: none;
|
||||
color: var(--text-secondary);
|
||||
padding: 4px 8px;
|
||||
border-radius: 4px;
|
||||
cursor: pointer;
|
||||
font-size: 12px;
|
||||
transition: background 0.2s;
|
||||
}
|
||||
|
||||
.markdown-body pre .copy-btn:hover {
|
||||
background: rgba(255,255,255,0.2);
|
||||
}
|
||||
|
||||
.markdown-body blockquote {
|
||||
border-left: 4px solid var(--primary);
|
||||
margin: 12px 0;
|
||||
padding: 8px 16px;
|
||||
background: rgba(74, 144, 217, 0.05);
|
||||
color: var(--text-secondary);
|
||||
}
|
||||
|
||||
.markdown-body table {
|
||||
width: 100%;
|
||||
border-collapse: collapse;
|
||||
margin: 12px 0;
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.markdown-body th, .markdown-body td {
|
||||
border: 1px solid var(--border-color);
|
||||
padding: 8px 12px;
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
.markdown-body th {
|
||||
background: var(--bg-secondary);
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.markdown-body tr:nth-child(even) {
|
||||
background: rgba(255,255,255,0.02);
|
||||
}
|
||||
|
||||
.markdown-body a {
|
||||
color: var(--primary);
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
.markdown-body a:hover {
|
||||
text-decoration: underline;
|
||||
}
|
||||
|
||||
.markdown-body hr {
|
||||
border: none;
|
||||
border-top: 1px solid var(--border-color);
|
||||
margin: 16px 0;
|
||||
}
|
||||
|
||||
.markdown-body img {
|
||||
max-width: 100%;
|
||||
border-radius: var(--radius);
|
||||
}
|
||||
230
assets/css/style.css
Normal file
230
assets/css/style.css
Normal file
@@ -0,0 +1,230 @@
|
||||
/* CSS 变量 */
|
||||
:root {
|
||||
--primary: #4a90d9;
|
||||
--primary-hover: #357abd;
|
||||
--bg-primary: #1a1a2e;
|
||||
--bg-secondary: #16213e;
|
||||
--bg-card: #1e2a45;
|
||||
--bg-input: #0f1b33;
|
||||
--text-primary: #e0e0e0;
|
||||
--text-secondary: #a0a0a0;
|
||||
--border-color: #2a3a5c;
|
||||
--success: #4caf50;
|
||||
--danger: #f44336;
|
||||
--warning: #ff9800;
|
||||
--sidebar-width: 260px;
|
||||
--toolbar-height: 50px;
|
||||
--input-height: 120px;
|
||||
--radius: 8px;
|
||||
--font-family: system-ui, -apple-system, sans-serif;
|
||||
}
|
||||
|
||||
/* Reset */
|
||||
*, *::before, *::after { margin: 0; padding: 0; box-sizing: border-box; }
|
||||
|
||||
body {
|
||||
font-family: var(--font-family);
|
||||
background: var(--bg-primary);
|
||||
color: var(--text-primary);
|
||||
line-height: 1.6;
|
||||
min-height: 100vh;
|
||||
}
|
||||
|
||||
/* 登录页样式 */
|
||||
.login-container {
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
min-height: 100vh;
|
||||
padding: 20px;
|
||||
}
|
||||
|
||||
.login-card {
|
||||
background: var(--bg-card);
|
||||
border-radius: var(--radius);
|
||||
padding: 40px;
|
||||
width: 100%;
|
||||
max-width: 400px;
|
||||
box-shadow: 0 4px 20px rgba(0,0,0,0.3);
|
||||
}
|
||||
|
||||
.login-card h1 {
|
||||
text-align: center;
|
||||
margin-bottom: 30px;
|
||||
font-size: 24px;
|
||||
color: var(--primary);
|
||||
}
|
||||
|
||||
/* 安装向导样式 */
|
||||
.install-container {
|
||||
max-width: 700px;
|
||||
margin: 40px auto;
|
||||
padding: 0 20px;
|
||||
}
|
||||
|
||||
.step-indicator {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
margin-bottom: 30px;
|
||||
padding: 0;
|
||||
list-style: none;
|
||||
}
|
||||
|
||||
.step-indicator .step {
|
||||
flex: 1;
|
||||
text-align: center;
|
||||
padding: 10px;
|
||||
font-size: 14px;
|
||||
color: var(--text-secondary);
|
||||
border-bottom: 3px solid var(--border-color);
|
||||
transition: all 0.3s;
|
||||
}
|
||||
|
||||
.step-indicator .step.active {
|
||||
color: var(--primary);
|
||||
border-bottom-color: var(--primary);
|
||||
}
|
||||
|
||||
.step-indicator .step.completed {
|
||||
color: var(--success);
|
||||
border-bottom-color: var(--success);
|
||||
}
|
||||
|
||||
/* 通用表单样式 */
|
||||
.form-group {
|
||||
margin-bottom: 16px;
|
||||
}
|
||||
|
||||
.form-group label {
|
||||
display: block;
|
||||
margin-bottom: 6px;
|
||||
font-size: 14px;
|
||||
color: var(--text-secondary);
|
||||
}
|
||||
|
||||
.form-group input,
|
||||
.form-group select,
|
||||
.form-group textarea {
|
||||
width: 100%;
|
||||
padding: 10px 12px;
|
||||
background: var(--bg-input);
|
||||
border: 1px solid var(--border-color);
|
||||
border-radius: var(--radius);
|
||||
color: var(--text-primary);
|
||||
font-size: 14px;
|
||||
transition: border-color 0.3s;
|
||||
}
|
||||
|
||||
.form-group input:focus,
|
||||
.form-group select:focus,
|
||||
.form-group textarea:focus {
|
||||
outline: none;
|
||||
border-color: var(--primary);
|
||||
}
|
||||
|
||||
/* 按钮 */
|
||||
.btn {
|
||||
display: inline-block;
|
||||
padding: 10px 20px;
|
||||
border: none;
|
||||
border-radius: var(--radius);
|
||||
font-size: 14px;
|
||||
cursor: pointer;
|
||||
transition: all 0.3s;
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
.btn-primary {
|
||||
background: var(--primary);
|
||||
color: white;
|
||||
}
|
||||
|
||||
.btn-primary:hover {
|
||||
background: var(--primary-hover);
|
||||
}
|
||||
|
||||
.btn-secondary {
|
||||
background: var(--bg-secondary);
|
||||
color: var(--text-primary);
|
||||
border: 1px solid var(--border-color);
|
||||
}
|
||||
|
||||
.btn-danger {
|
||||
background: var(--danger);
|
||||
color: white;
|
||||
}
|
||||
|
||||
.btn-sm {
|
||||
padding: 6px 12px;
|
||||
font-size: 12px;
|
||||
}
|
||||
|
||||
.btn-block {
|
||||
display: block;
|
||||
width: 100%;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.btn-group {
|
||||
display: flex;
|
||||
gap: 10px;
|
||||
margin-top: 20px;
|
||||
justify-content: flex-end;
|
||||
}
|
||||
|
||||
/* 消息提示 */
|
||||
.alert {
|
||||
padding: 12px 16px;
|
||||
border-radius: var(--radius);
|
||||
margin-bottom: 16px;
|
||||
font-size: 14px;
|
||||
}
|
||||
|
||||
.alert-error {
|
||||
background: rgba(244, 67, 54, 0.1);
|
||||
border: 1px solid var(--danger);
|
||||
color: var(--danger);
|
||||
}
|
||||
|
||||
.alert-success {
|
||||
background: rgba(76, 175, 80, 0.1);
|
||||
border: 1px solid var(--success);
|
||||
color: var(--success);
|
||||
}
|
||||
|
||||
/* 配置页样式 */
|
||||
.config-container {
|
||||
max-width: 900px;
|
||||
margin: 40px auto;
|
||||
padding: 0 20px;
|
||||
}
|
||||
|
||||
.config-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.config-section {
|
||||
background: var(--bg-card);
|
||||
border-radius: var(--radius);
|
||||
padding: 24px;
|
||||
margin-bottom: 24px;
|
||||
}
|
||||
|
||||
.config-section h2 {
|
||||
margin-bottom: 16px;
|
||||
font-size: 18px;
|
||||
}
|
||||
|
||||
/* 响应式 */
|
||||
@media (max-width: 768px) {
|
||||
.login-card {
|
||||
padding: 24px;
|
||||
}
|
||||
.install-container,
|
||||
.config-container {
|
||||
margin: 20px auto;
|
||||
}
|
||||
}
|
||||
0
assets/img/.gitkeep
Normal file
0
assets/img/.gitkeep
Normal file
0
assets/js/.gitkeep
Normal file
0
assets/js/.gitkeep
Normal file
74
assets/js/api.js
Normal file
74
assets/js/api.js
Normal file
@@ -0,0 +1,74 @@
|
||||
const api = {
|
||||
async request(url, options = {}) {
|
||||
const token = Storage.getToken();
|
||||
const headers = {
|
||||
'Content-Type': 'application/json',
|
||||
...(token ? { 'Authorization': 'Bearer ' + token } : {}),
|
||||
...options.headers
|
||||
};
|
||||
|
||||
const response = await fetch('/api' + url, {
|
||||
...options,
|
||||
headers
|
||||
});
|
||||
|
||||
// 401 自动跳转登录
|
||||
if (response.status === 401) {
|
||||
Storage.clearToken();
|
||||
window.location.href = '/login.php';
|
||||
throw new Error('请先登录');
|
||||
}
|
||||
|
||||
const data = await response.json();
|
||||
if (!response.ok) {
|
||||
throw new Error(data.message || '请求失败');
|
||||
}
|
||||
return data;
|
||||
},
|
||||
|
||||
get(url) {
|
||||
return this.request(url, { method: 'GET' });
|
||||
},
|
||||
|
||||
post(url, data) {
|
||||
return this.request(url, {
|
||||
method: 'POST',
|
||||
body: JSON.stringify(data)
|
||||
});
|
||||
},
|
||||
|
||||
put(url, data) {
|
||||
return this.request(url, {
|
||||
method: 'PUT',
|
||||
body: JSON.stringify(data)
|
||||
});
|
||||
},
|
||||
|
||||
delete(url) {
|
||||
return this.request(url, { method: 'DELETE' });
|
||||
},
|
||||
|
||||
// 文件上传(不用 JSON Content-Type)
|
||||
async upload(url, formData) {
|
||||
const token = Storage.getToken();
|
||||
const headers = token ? { 'Authorization': 'Bearer ' + token } : {};
|
||||
|
||||
const response = await fetch('/api' + url, {
|
||||
method: 'POST',
|
||||
headers,
|
||||
body: formData
|
||||
});
|
||||
|
||||
if (response.status === 401) {
|
||||
Storage.clearToken();
|
||||
window.location.href = '/login.php';
|
||||
throw new Error('请先登录');
|
||||
}
|
||||
|
||||
const data = await response.json();
|
||||
if (!response.ok) {
|
||||
throw new Error(data.message || '上传失败');
|
||||
}
|
||||
return data;
|
||||
}
|
||||
};
|
||||
308
assets/js/chat.js
Normal file
308
assets/js/chat.js
Normal file
@@ -0,0 +1,308 @@
|
||||
const ChatManager = {
|
||||
isStreaming: false,
|
||||
messages: [],
|
||||
|
||||
init() {
|
||||
const input = document.getElementById('messageInput');
|
||||
if (input) {
|
||||
input.addEventListener('keydown', (e) => {
|
||||
if (e.key === 'Enter' && e.ctrlKey) {
|
||||
e.preventDefault();
|
||||
this.sendMessage();
|
||||
}
|
||||
});
|
||||
// 自适应高度
|
||||
input.addEventListener('input', () => {
|
||||
input.style.height = 'auto';
|
||||
input.style.height = Math.min(input.scrollHeight, 120) + 'px';
|
||||
});
|
||||
}
|
||||
|
||||
const sendBtn = document.getElementById('sendBtn');
|
||||
if (sendBtn) {
|
||||
sendBtn.addEventListener('click', () => this.sendMessage());
|
||||
}
|
||||
|
||||
UploadManager.init();
|
||||
},
|
||||
|
||||
async loadMessages(sessionId) {
|
||||
try {
|
||||
const res = await api.get('/sessions/' + sessionId + '/messages');
|
||||
this.messages = res.data || [];
|
||||
this.renderMessages();
|
||||
Storage.setCachedMessages(sessionId, this.messages);
|
||||
} catch (err) {
|
||||
// 尝试从缓存加载
|
||||
const cached = Storage.getCachedMessages(sessionId);
|
||||
if (cached) {
|
||||
this.messages = cached;
|
||||
this.renderMessages();
|
||||
}
|
||||
}
|
||||
},
|
||||
|
||||
renderMessages() {
|
||||
const container = document.getElementById('messagesContainer');
|
||||
if (!container) return;
|
||||
|
||||
if (this.messages.length === 0) {
|
||||
container.innerHTML = '<div class="empty-state"><h3>开始新的对话</h3><p>输入消息开始聊天</p></div>';
|
||||
return;
|
||||
}
|
||||
|
||||
container.innerHTML = this.messages.map(msg => {
|
||||
if (msg.role === 'user') {
|
||||
return `<div class="message user">${this.escapeHtml(msg.content)}</div>`;
|
||||
} else {
|
||||
let html = `<div class="message assistant">`;
|
||||
if (msg.thinking_content) {
|
||||
html += `<div class="thinking-toggle" onclick="this.nextElementSibling.classList.toggle('expanded')">💭 思考过程 ▾</div>`;
|
||||
html += `<div class="thinking-content">${MarkdownRenderer.render(msg.thinking_content)}</div>`;
|
||||
}
|
||||
html += `<div class="markdown-body">${MarkdownRenderer.render(msg.content)}</div>`;
|
||||
html += `</div>`;
|
||||
return html;
|
||||
}
|
||||
}).join('');
|
||||
|
||||
container.scrollTop = container.scrollHeight;
|
||||
},
|
||||
|
||||
async sendMessage() {
|
||||
if (this.isStreaming) return;
|
||||
|
||||
const input = document.getElementById('messageInput');
|
||||
const content = input.value.trim();
|
||||
if (!content) return;
|
||||
|
||||
if (!SessionManager.currentSessionId) {
|
||||
await SessionManager.createSession();
|
||||
}
|
||||
|
||||
// 清空输入
|
||||
input.value = '';
|
||||
input.style.height = 'auto';
|
||||
|
||||
// 构建消息
|
||||
const userMessage = { role: 'user', content };
|
||||
if (UploadManager.getFiles().length > 0) {
|
||||
userMessage.file_info = UploadManager.getFiles();
|
||||
}
|
||||
this.messages.push(userMessage);
|
||||
this.renderMessages();
|
||||
|
||||
// 保存用户消息到数据库
|
||||
api.post('/sessions/' + SessionManager.currentSessionId + '/messages', userMessage).catch(console.error);
|
||||
|
||||
// 清除文件
|
||||
UploadManager.clearFiles();
|
||||
|
||||
// 获取当前配置
|
||||
const provider = document.getElementById('providerSelect')?.value || 'newapi';
|
||||
const model = document.getElementById('modelSelect')?.value || 'gpt-3.5-turbo';
|
||||
const thinkingMode = document.getElementById('thinkingMode')?.checked || false;
|
||||
|
||||
// SSE 流式请求
|
||||
this.isStreaming = true;
|
||||
this.updateSendButton();
|
||||
|
||||
try {
|
||||
await this.streamChat(provider, model, thinkingMode);
|
||||
} catch (err) {
|
||||
this.addErrorMessage(err.message);
|
||||
} finally {
|
||||
this.isStreaming = false;
|
||||
this.updateSendButton();
|
||||
}
|
||||
},
|
||||
|
||||
async streamChat(provider, model, thinkingMode) {
|
||||
const token = Storage.getToken();
|
||||
|
||||
// 构建消息历史(只取 role 和 content)
|
||||
const messages = this.messages.map(m => ({
|
||||
role: m.role,
|
||||
content: m.content
|
||||
}));
|
||||
|
||||
// 添加 AI 消息占位
|
||||
const assistantEl = this.addAssistantPlaceholder();
|
||||
let fullContent = '';
|
||||
let thinkingContent = '';
|
||||
|
||||
const response = await fetch('/api/chat/completions', {
|
||||
method: 'POST',
|
||||
headers: {
|
||||
'Content-Type': 'application/json',
|
||||
'Authorization': 'Bearer ' + token
|
||||
},
|
||||
body: JSON.stringify({
|
||||
provider, model, messages, stream: true, thinkingMode
|
||||
})
|
||||
});
|
||||
|
||||
if (!response.ok) {
|
||||
const err = await response.json();
|
||||
throw new Error(err.message || 'AI 请求失败');
|
||||
}
|
||||
|
||||
const reader = response.body.getReader();
|
||||
const decoder = new TextDecoder();
|
||||
let buffer = '';
|
||||
|
||||
while (true) {
|
||||
const { done, value } = await reader.read();
|
||||
if (done) break;
|
||||
|
||||
buffer += decoder.decode(value, { stream: true });
|
||||
const lines = buffer.split('\n');
|
||||
buffer = lines.pop() || '';
|
||||
|
||||
for (const line of lines) {
|
||||
if (!line.startsWith('data: ')) continue;
|
||||
const data = line.slice(6).trim();
|
||||
|
||||
if (data === '[DONE]') {
|
||||
continue;
|
||||
}
|
||||
|
||||
try {
|
||||
const parsed = JSON.parse(data);
|
||||
|
||||
if (parsed.type === 'error') {
|
||||
throw new Error(parsed.message);
|
||||
}
|
||||
|
||||
if (parsed.thinking) {
|
||||
thinkingContent += parsed.thinking;
|
||||
this.updateThinkingContent(assistantEl, thinkingContent);
|
||||
}
|
||||
|
||||
if (parsed.content) {
|
||||
fullContent += parsed.content;
|
||||
this.updateAssistantContent(assistantEl, fullContent);
|
||||
}
|
||||
} catch (e) {
|
||||
if (e.message && !e.message.includes('JSON')) throw e;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 流结束,保存 AI 消息
|
||||
const assistantMessage = {
|
||||
role: 'assistant',
|
||||
content: fullContent,
|
||||
thinking_content: thinkingContent || null
|
||||
};
|
||||
this.messages.push(assistantMessage);
|
||||
|
||||
// 保存到数据库
|
||||
api.post('/sessions/' + SessionManager.currentSessionId + '/messages', assistantMessage).catch(console.error);
|
||||
|
||||
// 缓存到 localStorage
|
||||
Storage.setCachedMessages(SessionManager.currentSessionId, this.messages);
|
||||
|
||||
// 移除打字机光标
|
||||
this.removeTypingCursor(assistantEl);
|
||||
},
|
||||
|
||||
addAssistantPlaceholder() {
|
||||
const container = document.getElementById('messagesContainer');
|
||||
const empty = container.querySelector('.empty-state');
|
||||
if (empty) empty.remove();
|
||||
|
||||
const el = document.createElement('div');
|
||||
el.className = 'message assistant';
|
||||
el.innerHTML = '<span class="typing-cursor"></span>';
|
||||
container.appendChild(el);
|
||||
container.scrollTop = container.scrollHeight;
|
||||
return el;
|
||||
},
|
||||
|
||||
updateAssistantContent(el, content) {
|
||||
const cursor = el.querySelector('.typing-cursor');
|
||||
const body = el.querySelector('.markdown-body');
|
||||
|
||||
if (body) {
|
||||
body.innerHTML = MarkdownRenderer.render(content);
|
||||
if (cursor) body.appendChild(cursor);
|
||||
} else {
|
||||
const thinking = el.querySelector('.thinking-content, .thinking-toggle');
|
||||
const md = document.createElement('div');
|
||||
md.className = 'markdown-body';
|
||||
md.innerHTML = MarkdownRenderer.render(content);
|
||||
if (cursor) md.appendChild(cursor);
|
||||
|
||||
if (thinking) {
|
||||
el.appendChild(md);
|
||||
} else {
|
||||
el.innerHTML = '';
|
||||
el.appendChild(md);
|
||||
}
|
||||
}
|
||||
|
||||
el.parentElement.scrollTop = el.parentElement.scrollHeight;
|
||||
},
|
||||
|
||||
updateThinkingContent(el, content) {
|
||||
let toggle = el.querySelector('.thinking-toggle');
|
||||
let container = el.querySelector('.thinking-content');
|
||||
|
||||
if (!toggle) {
|
||||
toggle = document.createElement('div');
|
||||
toggle.className = 'thinking-toggle expanded';
|
||||
toggle.textContent = '💭 思考过程 ▾';
|
||||
toggle.onclick = function() {
|
||||
container.classList.toggle('expanded');
|
||||
this.textContent = container.classList.contains('expanded') ? '💭 思考过程 ▴' : '💭 思考过程 ▾';
|
||||
};
|
||||
el.insertBefore(toggle, el.firstChild);
|
||||
}
|
||||
|
||||
if (!container) {
|
||||
container = document.createElement('div');
|
||||
container.className = 'thinking-content expanded';
|
||||
el.insertBefore(container, toggle.nextSibling);
|
||||
}
|
||||
|
||||
container.innerHTML = MarkdownRenderer.render(content);
|
||||
el.parentElement.scrollTop = el.parentElement.scrollHeight;
|
||||
},
|
||||
|
||||
removeTypingCursor(el) {
|
||||
const cursor = el.querySelector('.typing-cursor');
|
||||
if (cursor) cursor.remove();
|
||||
},
|
||||
|
||||
addErrorMessage(message) {
|
||||
const container = document.getElementById('messagesContainer');
|
||||
const el = document.createElement('div');
|
||||
el.className = 'message assistant';
|
||||
el.innerHTML = `<div class="alert alert-error">❌ ${this.escapeHtml(message)} <button class="btn btn-sm btn-secondary" onclick="ChatManager.retryLast()">重试</button></div>`;
|
||||
container.appendChild(el);
|
||||
container.scrollTop = container.scrollHeight;
|
||||
},
|
||||
|
||||
clearMessages() {
|
||||
const container = document.getElementById('messagesContainer');
|
||||
if (container) {
|
||||
container.innerHTML = '<div class="empty-state"><h3>开始新的对话</h3><p>输入消息开始聊天</p></div>';
|
||||
}
|
||||
this.messages = [];
|
||||
},
|
||||
|
||||
updateSendButton() {
|
||||
const btn = document.getElementById('sendBtn');
|
||||
if (btn) {
|
||||
btn.disabled = this.isStreaming;
|
||||
btn.textContent = this.isStreaming ? '回复中...' : '发送';
|
||||
}
|
||||
},
|
||||
|
||||
escapeHtml(text) {
|
||||
const div = document.createElement('div');
|
||||
div.textContent = text;
|
||||
return div.innerHTML;
|
||||
}
|
||||
};
|
||||
34
assets/js/markdown.js
Normal file
34
assets/js/markdown.js
Normal file
@@ -0,0 +1,34 @@
|
||||
const MarkdownRenderer = {
|
||||
init() {
|
||||
// 配置 marked.js
|
||||
marked.setOptions({
|
||||
highlight: function(code, lang) {
|
||||
if (lang && hljs.getLanguage(lang)) {
|
||||
return hljs.highlight(code, { language: lang }).value;
|
||||
}
|
||||
return hljs.highlightAuto(code).value;
|
||||
},
|
||||
breaks: true,
|
||||
gfm: true
|
||||
});
|
||||
},
|
||||
|
||||
render(text) {
|
||||
if (!text) return '';
|
||||
let html = marked.parse(text);
|
||||
// 为代码块添加复制按钮
|
||||
html = html.replace(/<pre><code/g, '<pre><button class="copy-btn" onclick="MarkdownRenderer.copyCode(this)">复制</button><code');
|
||||
return html;
|
||||
},
|
||||
|
||||
copyCode(btn) {
|
||||
const code = btn.nextElementSibling;
|
||||
navigator.clipboard.writeText(code.textContent).then(() => {
|
||||
btn.textContent = '已复制';
|
||||
setTimeout(() => { btn.textContent = '复制'; }, 2000);
|
||||
});
|
||||
}
|
||||
};
|
||||
|
||||
// 初始化
|
||||
MarkdownRenderer.init();
|
||||
72
assets/js/session.js
Normal file
72
assets/js/session.js
Normal file
@@ -0,0 +1,72 @@
|
||||
const SessionManager = {
|
||||
sessions: [],
|
||||
currentSessionId: null,
|
||||
|
||||
async loadSessions() {
|
||||
try {
|
||||
const res = await api.get('/sessions');
|
||||
this.sessions = res.data || [];
|
||||
this.renderSessionList();
|
||||
} catch (err) {
|
||||
console.error('加载会话失败:', err);
|
||||
}
|
||||
},
|
||||
|
||||
renderSessionList() {
|
||||
const list = document.getElementById('sessionList');
|
||||
if (!list) return;
|
||||
|
||||
list.innerHTML = this.sessions.map(s => `
|
||||
<div class="session-item ${s.id === this.currentSessionId ? 'active' : ''}"
|
||||
onclick="SessionManager.switchSession(${s.id})">
|
||||
<span class="session-name">${this.escapeHtml(s.name)}</span>
|
||||
<button class="session-delete" onclick="event.stopPropagation(); SessionManager.deleteSession(${s.id})" title="删除">×</button>
|
||||
</div>
|
||||
`).join('');
|
||||
},
|
||||
|
||||
async createSession() {
|
||||
try {
|
||||
const res = await api.post('/sessions', {});
|
||||
this.sessions.unshift(res.data);
|
||||
this.currentSessionId = res.data.id;
|
||||
this.renderSessionList();
|
||||
ChatManager.clearMessages();
|
||||
return res.data;
|
||||
} catch (err) {
|
||||
console.error('创建会话失败:', err);
|
||||
}
|
||||
},
|
||||
|
||||
async switchSession(id) {
|
||||
this.currentSessionId = id;
|
||||
this.renderSessionList();
|
||||
await ChatManager.loadMessages(id);
|
||||
},
|
||||
|
||||
async deleteSession(id) {
|
||||
if (!confirm('确定要删除这个会话吗?')) return;
|
||||
try {
|
||||
await api.delete('/sessions/' + id);
|
||||
this.sessions = this.sessions.filter(s => s.id !== id);
|
||||
Storage.clearCachedMessages(id);
|
||||
|
||||
if (this.currentSessionId === id) {
|
||||
this.currentSessionId = null;
|
||||
ChatManager.clearMessages();
|
||||
if (this.sessions.length > 0) {
|
||||
await this.switchSession(this.sessions[0].id);
|
||||
}
|
||||
}
|
||||
this.renderSessionList();
|
||||
} catch (err) {
|
||||
alert('删除会话失败: ' + err.message);
|
||||
}
|
||||
},
|
||||
|
||||
escapeHtml(text) {
|
||||
const div = document.createElement('div');
|
||||
div.textContent = text;
|
||||
return div.innerHTML;
|
||||
}
|
||||
};
|
||||
22
assets/js/storage.js
Normal file
22
assets/js/storage.js
Normal file
@@ -0,0 +1,22 @@
|
||||
const Storage = {
|
||||
getToken() {
|
||||
return localStorage.getItem('token');
|
||||
},
|
||||
setToken(token) {
|
||||
localStorage.setItem('token', token);
|
||||
},
|
||||
clearToken() {
|
||||
localStorage.removeItem('token');
|
||||
},
|
||||
// 消息缓存
|
||||
getCachedMessages(sessionId) {
|
||||
const data = localStorage.getItem('messages_' + sessionId);
|
||||
return data ? JSON.parse(data) : null;
|
||||
},
|
||||
setCachedMessages(sessionId, messages) {
|
||||
localStorage.setItem('messages_' + sessionId, JSON.stringify(messages));
|
||||
},
|
||||
clearCachedMessages(sessionId) {
|
||||
localStorage.removeItem('messages_' + sessionId);
|
||||
}
|
||||
};
|
||||
67
assets/js/upload.js
Normal file
67
assets/js/upload.js
Normal file
@@ -0,0 +1,67 @@
|
||||
const UploadManager = {
|
||||
files: [], // 待发送的文件列表
|
||||
allowedTypes: '.jpg,.jpeg,.png,.gif,.webp,.js,.ts,.py,.java,.cpp,.c,.html,.css,.json,.xml,.txt,.md,.go,.rs,.php,.rb,.sql,.yaml,.yml,.sh,.bat',
|
||||
|
||||
init() {
|
||||
const btn = document.getElementById('uploadBtn');
|
||||
if (btn) {
|
||||
btn.addEventListener('click', () => this.openFileSelector());
|
||||
}
|
||||
},
|
||||
|
||||
openFileSelector() {
|
||||
const input = document.createElement('input');
|
||||
input.type = 'file';
|
||||
input.accept = this.allowedTypes;
|
||||
input.multiple = true;
|
||||
input.onchange = (e) => this.handleFiles(e.target.files);
|
||||
input.click();
|
||||
},
|
||||
|
||||
async handleFiles(fileList) {
|
||||
for (const file of fileList) {
|
||||
const formData = new FormData();
|
||||
formData.append('file', file);
|
||||
|
||||
try {
|
||||
const res = await api.upload('/upload', formData);
|
||||
this.files.push(res.data);
|
||||
this.renderPreview();
|
||||
} catch (err) {
|
||||
alert('文件上传失败: ' + err.message);
|
||||
}
|
||||
}
|
||||
},
|
||||
|
||||
renderPreview() {
|
||||
const preview = document.getElementById('filePreview');
|
||||
if (!preview) return;
|
||||
|
||||
preview.innerHTML = this.files.map((f, i) => `
|
||||
<div class="file-preview-item">
|
||||
<span>📎 ${this.escapeHtml(f.name)}</span>
|
||||
<span class="remove-file" onclick="UploadManager.removeFile(${i})">×</span>
|
||||
</div>
|
||||
`).join('');
|
||||
},
|
||||
|
||||
removeFile(index) {
|
||||
this.files.splice(index, 1);
|
||||
this.renderPreview();
|
||||
},
|
||||
|
||||
getFiles() {
|
||||
return this.files;
|
||||
},
|
||||
|
||||
clearFiles() {
|
||||
this.files = [];
|
||||
this.renderPreview();
|
||||
},
|
||||
|
||||
escapeHtml(text) {
|
||||
const div = document.createElement('div');
|
||||
div.textContent = text;
|
||||
return div.innerHTML;
|
||||
}
|
||||
};
|
||||
20
composer.json
Normal file
20
composer.json
Normal file
@@ -0,0 +1,20 @@
|
||||
{
|
||||
"name": "ai-chat/app",
|
||||
"description": "AI Chat Web Application - PHP Implementation",
|
||||
"type": "project",
|
||||
"require": {
|
||||
"php": ">=8.0",
|
||||
"firebase/php-jwt": "^6.0",
|
||||
"erusev/parsedown": "^1.7"
|
||||
},
|
||||
"autoload": {
|
||||
"psr-4": {
|
||||
"App\\": "app/"
|
||||
}
|
||||
},
|
||||
"config": {
|
||||
"platform": {
|
||||
"php": "8.0"
|
||||
}
|
||||
}
|
||||
}
|
||||
0
config/.gitkeep
Normal file
0
config/.gitkeep
Normal file
208
docs/API.md
Normal file
208
docs/API.md
Normal file
@@ -0,0 +1,208 @@
|
||||
# API 文档
|
||||
|
||||
## 基础信息
|
||||
|
||||
- 基础 URL:`/api`
|
||||
- 响应格式:JSON
|
||||
- 认证方式:JWT Bearer Token(Authorization 头)
|
||||
|
||||
### 统一响应格式
|
||||
|
||||
```json
|
||||
{
|
||||
"success": true,
|
||||
"data": {},
|
||||
"message": "操作成功"
|
||||
}
|
||||
```
|
||||
|
||||
## 认证 API
|
||||
|
||||
### POST /auth/login
|
||||
用户登录,获取 JWT 令牌。
|
||||
|
||||
**请求体:**
|
||||
```json
|
||||
{
|
||||
"username": "admin",
|
||||
"password": "123456"
|
||||
}
|
||||
```
|
||||
|
||||
**成功响应:**
|
||||
```json
|
||||
{
|
||||
"success": true,
|
||||
"data": {
|
||||
"token": "eyJ0eXAiOiJKV1QiLCJhbGc...",
|
||||
"user": {
|
||||
"id": 1,
|
||||
"username": "admin",
|
||||
"role": "admin"
|
||||
}
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### GET /auth/me
|
||||
获取当前用户信息。(需认证)
|
||||
|
||||
**响应:**
|
||||
```json
|
||||
{
|
||||
"success": true,
|
||||
"data": {
|
||||
"id": 1,
|
||||
"username": "admin",
|
||||
"role": "admin"
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
## 会话 API
|
||||
|
||||
### GET /sessions
|
||||
获取当前用户的会话列表。(需认证)
|
||||
|
||||
**响应:**
|
||||
```json
|
||||
{
|
||||
"success": true,
|
||||
"data": [
|
||||
{
|
||||
"id": 1,
|
||||
"user_id": 1,
|
||||
"name": "新会话",
|
||||
"provider": "newapi",
|
||||
"model": "gpt-3.5-turbo",
|
||||
"system_prompt": "",
|
||||
"personality_id": null,
|
||||
"thinking_mode": 0,
|
||||
"created_at": "2024-01-01 00:00:00",
|
||||
"updated_at": "2024-01-01 00:00:00"
|
||||
}
|
||||
]
|
||||
}
|
||||
```
|
||||
|
||||
### POST /sessions
|
||||
创建新会话。(需认证)
|
||||
|
||||
**请求体:**
|
||||
```json
|
||||
{
|
||||
"name": "新对话",
|
||||
"provider": "newapi",
|
||||
"model": "gpt-3.5-turbo",
|
||||
"system_prompt": "",
|
||||
"thinking_mode": false
|
||||
}
|
||||
```
|
||||
|
||||
### PUT /sessions/{id}
|
||||
更新会话。(需认证)
|
||||
|
||||
### DELETE /sessions/{id}
|
||||
删除会话。(需认证)
|
||||
|
||||
## 消息 API
|
||||
|
||||
### GET /sessions/{id}/messages
|
||||
获取指定会话的消息列表。(需认证)
|
||||
|
||||
### POST /sessions/{id}/messages
|
||||
保存消息到指定会话。(需认证)
|
||||
|
||||
**请求体:**
|
||||
```json
|
||||
{
|
||||
"role": "user",
|
||||
"content": "Hello!",
|
||||
"file_info": null,
|
||||
"thinking_content": null
|
||||
}
|
||||
```
|
||||
|
||||
## AI 对话 API
|
||||
|
||||
### POST /chat/completions
|
||||
AI 对话请求(SSE 流式响应)。(需认证)
|
||||
|
||||
**请求体:**
|
||||
```json
|
||||
{
|
||||
"provider": "newapi",
|
||||
"model": "gpt-3.5-turbo",
|
||||
"messages": [
|
||||
{"role": "user", "content": "Hello!"}
|
||||
],
|
||||
"stream": true,
|
||||
"systemPrompt": "",
|
||||
"thinkingMode": false
|
||||
}
|
||||
```
|
||||
|
||||
**SSE 响应格式:**
|
||||
```
|
||||
data: {"content":"Hello"}
|
||||
|
||||
data: {"content":"!"}
|
||||
|
||||
data: {"thinking":"Let me think..."}
|
||||
|
||||
data: [DONE]
|
||||
```
|
||||
|
||||
## 文件上传 API
|
||||
|
||||
### POST /upload
|
||||
上传文件。(需认证,multipart/form-data)
|
||||
|
||||
**支持格式:** jpg, jpeg, png, gif, webp, js, ts, py, java, cpp, c, html, css, json, xml, txt, md, go, rs, php, rb, sql, yaml, yml, sh, bat
|
||||
**大小限制:** 10MB
|
||||
|
||||
**响应:**
|
||||
```json
|
||||
{
|
||||
"success": true,
|
||||
"data": {
|
||||
"url": "/uploads/xxx.jpg",
|
||||
"name": "原始文件名.jpg",
|
||||
"size": 1024,
|
||||
"type": "jpg"
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
## 配置 API
|
||||
|
||||
### GET /config
|
||||
获取系统配置。(需认证)
|
||||
|
||||
### PUT /config
|
||||
更新系统配置。(需认证 + 管理员)
|
||||
|
||||
## 人格 API
|
||||
|
||||
### GET /personalities
|
||||
获取人格列表。(需认证)
|
||||
|
||||
### POST /personalities
|
||||
创建人格。(需认证 + 管理员)
|
||||
|
||||
### PUT /personalities/{id}
|
||||
更新人格。(需认证 + 管理员)
|
||||
|
||||
### DELETE /personalities/{id}
|
||||
删除人格。(需认证 + 管理员)
|
||||
|
||||
## 安装 API
|
||||
|
||||
### GET /install/status
|
||||
检查安装状态。(无需认证)
|
||||
|
||||
### POST /install/test-db
|
||||
测试数据库连接。(无需认证)
|
||||
|
||||
### POST /install/setup
|
||||
执行安装。(无需认证)
|
||||
138
docs/ARCHITECTURE.md
Normal file
138
docs/ARCHITECTURE.md
Normal file
@@ -0,0 +1,138 @@
|
||||
# 架构说明
|
||||
|
||||
## 技术架构
|
||||
|
||||
本项目采用 PHP 8.0 原生 MVC 架构,不使用任何 PHP 框架。
|
||||
|
||||
```
|
||||
┌──────────────────────────────────────────────┐
|
||||
│ Nginx │
|
||||
│ (Web 服务器 + 反向代理) │
|
||||
└──────────┬───────────────────────────────────┘
|
||||
│
|
||||
▼
|
||||
┌──────────────────────────────────────────────┐
|
||||
│ public/ │
|
||||
│ (Web 根目录, 入口文件) │
|
||||
│ api.php, login.php, chat.php, config.php │
|
||||
└──────────┬───────────────────────────────────┘
|
||||
│
|
||||
┌─────┴─────┐
|
||||
│ │
|
||||
▼ ▼
|
||||
┌─────────┐ ┌──────────────────────────────┐
|
||||
│ PHP 页面 │ │ API (api.php) │
|
||||
│ (视图) │ │ 路由分发 + 中间件处理 │
|
||||
└────┬────┘ └──────┬───────────────────────┘
|
||||
│ │
|
||||
▼ ▼
|
||||
┌──────────────────────────────────────────────┐
|
||||
│ Controllers │
|
||||
│ AuthController, ChatController, │
|
||||
│ SessionController, MessageController, │
|
||||
│ ConfigController, InstallController, │
|
||||
│ UploadController │
|
||||
└──────────┬───────────────────────────────────┘
|
||||
│
|
||||
┌─────┼─────────────┐
|
||||
│ │ │
|
||||
▼ ▼ ▼
|
||||
┌───────┐ ┌───────┐ ┌──────────────────────┐
|
||||
│Models │ │Services│ │ Middleware │
|
||||
│(数据) │ │(业务) │ │ Auth/Admin │
|
||||
└───┬───┘ └───┬───┘ └──────────────────────┘
|
||||
│ │
|
||||
▼ ▼
|
||||
┌────────┐ ┌─────────────────┐
|
||||
│ MySQL │ │ AI Providers │
|
||||
│(PDO) │ │ OpenAI/Claude/ │
|
||||
│ │ │ NewAPI │
|
||||
└────────┘ └─────────────────┘
|
||||
```
|
||||
|
||||
## 目录结构
|
||||
|
||||
```
|
||||
ai-chat/
|
||||
├── public/ # Web 根目录
|
||||
│ ├── api.php # API 统一入口(路由分发)
|
||||
│ ├── login.php # 登录页面
|
||||
│ ├── chat.php # 聊天主页面
|
||||
│ ├── config.php # 配置管理页面
|
||||
│ └── install.php # 安装向导页面
|
||||
├── app/ # 应用代码(命名空间 App\)
|
||||
│ ├── Config/ # 配置类
|
||||
│ │ ├── Database.php # 数据库连接(单例 PDO)
|
||||
│ │ └── AppConfig.php # 应用配置读写
|
||||
│ ├── Controllers/ # 控制器
|
||||
│ │ ├── AuthController.php
|
||||
│ │ ├── ChatController.php
|
||||
│ │ ├── SessionController.php
|
||||
│ │ ├── MessageController.php
|
||||
│ │ ├── ConfigController.php
|
||||
│ │ ├── InstallController.php
|
||||
│ │ └── UploadController.php
|
||||
│ ├── Middleware/ # 中间件
|
||||
│ │ ├── AuthMiddleware.php # JWT 认证
|
||||
│ │ └── AdminMiddleware.php # 管理员权限
|
||||
│ ├── Models/ # 数据模型
|
||||
│ │ ├── User.php
|
||||
│ │ ├── Session.php
|
||||
│ │ ├── Message.php
|
||||
│ │ ├── Config.php
|
||||
│ │ └── Personality.php
|
||||
│ ├── Services/ # 服务层
|
||||
│ │ ├── AIService.php # AI 服务路由
|
||||
│ │ ├── Installer.php # 安装迁移服务
|
||||
│ │ └── Providers/ # AI 供应商
|
||||
│ │ ├── OpenAIProvider.php
|
||||
│ │ ├── ClaudeProvider.php
|
||||
│ │ └── NewAPIProvider.php
|
||||
│ └── Views/ # 视图模板
|
||||
│ ├── login.php
|
||||
│ ├── chat.php
|
||||
│ ├── config.php
|
||||
│ └── layout/
|
||||
│ ├── header.php
|
||||
│ └── footer.php
|
||||
├── assets/ # 前端静态资源
|
||||
│ ├── css/ # 样式
|
||||
│ │ ├── style.css # 全局样式
|
||||
│ │ ├── chat.css # 聊天界面样式
|
||||
│ │ └── markdown.css # Markdown 渲染样式
|
||||
│ ├── js/ # JavaScript
|
||||
│ │ ├── api.js # API 调用封装
|
||||
│ │ ├── chat.js # 聊天核心功能
|
||||
│ │ ├── session.js # 会话管理
|
||||
│ │ ├── upload.js # 文件上传
|
||||
│ │ ├── markdown.js # Markdown 渲染
|
||||
│ │ └── storage.js # 本地存储
|
||||
│ └── img/ # 图片
|
||||
├── config/ # JSON 配置文件
|
||||
├── uploads/ # 用户上传文件
|
||||
├── docs/ # 文档
|
||||
└── composer.json # Composer 配置
|
||||
```
|
||||
|
||||
## 请求流程
|
||||
|
||||
### API 请求流程
|
||||
1. 客户端发送 HTTP 请求到 `/api/*`
|
||||
2. Nginx 将请求转发到 `public/api.php`
|
||||
3. api.php 解析路由,执行对应的中间件和控制器
|
||||
4. 控制器调用模型/服务层处理业务逻辑
|
||||
5. 返回 JSON 响应
|
||||
|
||||
### SSE 流式响应流程
|
||||
1. 客户端发送 POST 请求到 `/api/chat/completions`
|
||||
2. ChatController 设置 SSE 响应头
|
||||
3. AIService 通过 cURL 连接 AI 供应商 API
|
||||
4. 使用 CURLOPT_WRITEFUNCTION 回调逐块接收 AI 响应
|
||||
5. 每块数据通过 SSE 格式发送给客户端
|
||||
6. AI 响应完成后发送 `[DONE]` 信号
|
||||
|
||||
### 页面请求流程
|
||||
1. 客户端访问 PHP 页面(如 `/chat.php`)
|
||||
2. Nginx 通过 PHP-FPM 执行 PHP 文件
|
||||
3. PHP 文件引入布局模板和视图
|
||||
4. 客户端通过 AJAX 加载数据
|
||||
71
docs/DEPLOY.md
Normal file
71
docs/DEPLOY.md
Normal file
@@ -0,0 +1,71 @@
|
||||
# 部署指南
|
||||
|
||||
## 宝塔面板部署
|
||||
|
||||
### 1. 环境准备
|
||||
|
||||
在宝塔面板"软件商店"中安装:
|
||||
- Nginx 1.22+
|
||||
- PHP 8.0(在 PHP 设置中确保以下扩展已启用:pdo_mysql、curl、mbstring、json、openssl)
|
||||
- MySQL 5.7 或 8.0
|
||||
|
||||
### 2. 创建网站
|
||||
|
||||
1. 网站 → 添加站点
|
||||
2. 填写域名
|
||||
3. PHP 版本选择 **PHP 8.0**
|
||||
4. 数据库选择"不创建"
|
||||
|
||||
### 3. 部署代码
|
||||
|
||||
```bash
|
||||
cd /www/wwwroot/your-domain.com
|
||||
git clone <repository-url> .
|
||||
composer install
|
||||
```
|
||||
|
||||
### 4. 设置权限
|
||||
|
||||
```bash
|
||||
chmod -R 755 .
|
||||
chown -R www:www .
|
||||
```
|
||||
|
||||
### 5. 配置 Nginx
|
||||
|
||||
1. 站点设置 → 网站目录 → 运行目录改为 `/public`
|
||||
2. 站点设置 → 配置文件 → 添加 `docs/baota-nginx-snippet.conf` 中的配置
|
||||
|
||||
### 6. 运行安装向导
|
||||
|
||||
访问 `http://your-domain.com/install.php`
|
||||
|
||||
### 7. SSL 配置(可选)
|
||||
|
||||
站点设置 → SSL → Let's Encrypt → 申请证书
|
||||
|
||||
## 手动部署
|
||||
|
||||
### Nginx 配置
|
||||
|
||||
参考 `docs/nginx.conf` 文件,将 `root` 指向 `public/` 目录。
|
||||
|
||||
### PHP-FPM 配置
|
||||
|
||||
确保 PHP-FPM 监听 socket 配置正确(通常为 `/tmp/php-cgi-80.sock` 或 `127.0.0.1:9000`)。
|
||||
|
||||
### 目录权限
|
||||
|
||||
```bash
|
||||
chown -R www-data:www-data /path/to/ai-chat
|
||||
chmod 755 /path/to/ai-chat/uploads
|
||||
chmod 755 /path/to/ai-chat/config
|
||||
```
|
||||
|
||||
## 安全建议
|
||||
|
||||
1. 配置 SSL 证书(HTTPS)
|
||||
2. 修改数据库默认端口
|
||||
3. 定期更新 PHP 和 Nginx 版本
|
||||
4. 限制 uploads 目录的执行权限
|
||||
5. 禁止访问 app/、config/、vendor/ 等非公开目录(已在 Nginx 配置中设置)
|
||||
35
docs/PERSONALITY.md
Normal file
35
docs/PERSONALITY.md
Normal file
@@ -0,0 +1,35 @@
|
||||
# 人格系统说明
|
||||
|
||||
## 概述
|
||||
|
||||
人格系统允许用户选择不同的 AI 角色,每个角色有特定的系统提示词,从而影响 AI 的回答风格和专业领域。
|
||||
|
||||
## 预设人格
|
||||
|
||||
系统安装后自动创建以下 5 个预设人格:
|
||||
|
||||
| 名称 | 图标 | 说明 |
|
||||
|------|------|------|
|
||||
| 智能助手 | 🤖 | 全能智能助手,善于回答各种问题 |
|
||||
| 代码专家 | 💻 | 专业编程专家,精通多种语言和框架 |
|
||||
| 翻译官 | 🌐 | 多语言翻译专家 |
|
||||
| 写作助手 | ✍️ | 专业写作助手,擅长各类文体 |
|
||||
| 数学家 | 🔢 | 数学专家,善于解答数学问题 |
|
||||
|
||||
预设人格不可编辑和删除。
|
||||
|
||||
## 自定义人格
|
||||
|
||||
管理员可以在配置页面(`/config.php`)创建自定义人格:
|
||||
|
||||
- **名称**:人格的显示名称
|
||||
- **图标**:Emoji 图标
|
||||
- **提示词**:系统提示词(定义 AI 的角色和行为)
|
||||
- **描述**:简短描述
|
||||
|
||||
## 技术实现
|
||||
|
||||
- 人格数据存储在 `personalities` 数据表中
|
||||
- 预设人格 `is_preset = 1`,自定义人格 `is_preset = 0`
|
||||
- 选择人格后,其 `prompt` 字段作为 `system_prompt` 传递给 AI API
|
||||
- 人格 API 路径:`/api/personalities`
|
||||
86
docs/baota-nginx-snippet.conf
Normal file
86
docs/baota-nginx-snippet.conf
Normal file
@@ -0,0 +1,86 @@
|
||||
# ============================================================
|
||||
# AI Chat 宝塔面板 Nginx 配置片段(PHP-FPM 版本)
|
||||
# ============================================================
|
||||
#
|
||||
# 【使用说明】
|
||||
# 1. 本文件为宝塔面板专用的 Nginx 配置片段,不是完整的站点配置文件。
|
||||
# 2. 使用步骤:
|
||||
# a. 在宝塔面板中创建网站,PHP 版本选择 PHP 8.0
|
||||
# b. 将网站根目录指向 /path/to/ai-chat/public
|
||||
# c. 登录宝塔面板 → 网站 → 找到你的站点 → 点击"设置"
|
||||
# d. 在左侧菜单选择"配置文件"
|
||||
# e. 将下方"配置内容开始"到"配置内容结束"之间的内容,
|
||||
# 复制并粘贴到 server { } 块内(注意不要重复嵌套 server 块)
|
||||
# f. 保存即可生效
|
||||
# 3. 前置准备:
|
||||
# - 在宝塔面板中安装 PHP 8.0 并启用
|
||||
# - 安装 MySQL 5.7+ 或 8.0
|
||||
# - 安装 Composer
|
||||
# - 运行 composer install 安装 PHP 依赖
|
||||
#
|
||||
# ============================================================
|
||||
|
||||
# ↓↓↓ 配置内容开始 ↓↓↓
|
||||
|
||||
# === AI Chat 网站配置(PHP-FPM) ===
|
||||
|
||||
# 1. 默认路由
|
||||
location / {
|
||||
try_files $uri $uri/ /login.php;
|
||||
}
|
||||
|
||||
# 2. API 路由 - 转发到 api.php
|
||||
location /api/ {
|
||||
try_files $uri /api.php$is_args$args;
|
||||
fastcgi_pass unix:/tmp/php-cgi-80.sock;
|
||||
fastcgi_index index.php;
|
||||
include fastcgi_params;
|
||||
fastcgi_param SCRIPT_FILENAME $document_root/api.php;
|
||||
fastcgi_param PATH_INFO $fastcgi_path_info;
|
||||
fastcgi_param REQUEST_URI $request_uri;
|
||||
}
|
||||
|
||||
# 3. SSE 流式响应特殊配置(AI对话需要)
|
||||
# 针对 Chat API 的 SSE(Server-Sent Events)流式响应,
|
||||
# 需要禁用所有缓冲以确保实时输出
|
||||
location /api/chat/completions {
|
||||
fastcgi_pass unix:/tmp/php-cgi-80.sock;
|
||||
include fastcgi_params;
|
||||
fastcgi_param SCRIPT_FILENAME $document_root/api.php;
|
||||
fastcgi_param REQUEST_URI $request_uri;
|
||||
# SSE 关键配置
|
||||
fastcgi_buffering off;
|
||||
proxy_buffering off;
|
||||
add_header X-Accel-Buffering no;
|
||||
# 超时设置(5分钟)
|
||||
fastcgi_read_timeout 300s;
|
||||
}
|
||||
|
||||
# 4. 静态资源缓存
|
||||
location /assets/ {
|
||||
expires 30d;
|
||||
add_header Cache-Control "public, immutable";
|
||||
}
|
||||
|
||||
# 5. 上传文件访问
|
||||
location /uploads/ {
|
||||
expires 7d;
|
||||
add_header Cache-Control "public";
|
||||
}
|
||||
|
||||
# 6. 禁止访问非公开目录(安全)
|
||||
location ~ /(app|config|vendor|tests|scripts|)/ {
|
||||
deny all;
|
||||
return 404;
|
||||
}
|
||||
|
||||
# 7. 禁止访问隐藏文件
|
||||
location ~ /\. {
|
||||
deny all;
|
||||
return 404;
|
||||
}
|
||||
|
||||
# 8. 上传文件大小限制
|
||||
client_max_body_size 10m;
|
||||
|
||||
# ↑↑↑ 配置内容结束 ↑↑↑
|
||||
110
docs/nginx.conf
Normal file
110
docs/nginx.conf
Normal file
@@ -0,0 +1,110 @@
|
||||
# AI Chat 应用 Nginx 配置(PHP-FPM 版本)
|
||||
# 将此文件复制到 /etc/nginx/sites-available/ 并创建软链接到 sites-enabled/
|
||||
#
|
||||
# 注意:本文件为完整的 Nginx 站点配置,适用于直接部署到 Nginx + PHP-FPM 的场景。
|
||||
# 如果你使用宝塔面板管理服务器,请参考 baota-nginx-snippet.conf 文件中的配置片段。
|
||||
#
|
||||
# 前置条件:
|
||||
# - PHP 8.0+ 已安装并启用 PHP-FPM
|
||||
# - MySQL 5.7+ 已安装
|
||||
# - Composer 已安装
|
||||
# - 项目已部署到 /path/to/ai-chat/ 目录
|
||||
|
||||
server {
|
||||
listen 80;
|
||||
server_name your-domain.com;
|
||||
|
||||
# 网站根目录指向 public/
|
||||
root /path/to/ai-chat/public;
|
||||
index index.php login.php install.php;
|
||||
|
||||
# 安全 Headers
|
||||
add_header X-Frame-Options "SAMEORIGIN" always;
|
||||
add_header X-Content-Type-Options "nosniff" always;
|
||||
add_header X-XSS-Protection "1; mode=block" always;
|
||||
add_header Referrer-Policy "strict-origin-when-cross-origin" always;
|
||||
|
||||
# Gzip 压缩
|
||||
gzip on;
|
||||
gzip_vary on;
|
||||
gzip_proxied any;
|
||||
gzip_comp_level 6;
|
||||
gzip_min_length 256;
|
||||
gzip_types
|
||||
text/plain
|
||||
text/css
|
||||
text/javascript
|
||||
application/javascript
|
||||
application/json
|
||||
application/xml
|
||||
image/svg+xml;
|
||||
|
||||
# 默认路由 - PHP 文件
|
||||
location / {
|
||||
try_files $uri $uri/ /login.php;
|
||||
}
|
||||
|
||||
# API 路由 - 所有 /api/ 请求转发到 api.php
|
||||
location /api/ {
|
||||
try_files $uri /api.php$is_args$args;
|
||||
fastcgi_pass unix:/tmp/php-cgi-80.sock;
|
||||
fastcgi_index index.php;
|
||||
include fastcgi_params;
|
||||
fastcgi_param SCRIPT_FILENAME $document_root/api.php;
|
||||
fastcgi_param PATH_INFO $fastcgi_path_info;
|
||||
fastcgi_param REQUEST_URI $request_uri;
|
||||
}
|
||||
|
||||
# SSE 流式响应专用配置(AI对话 API)
|
||||
location /api/chat/completions {
|
||||
fastcgi_pass unix:/tmp/php-cgi-80.sock;
|
||||
include fastcgi_params;
|
||||
fastcgi_param SCRIPT_FILENAME $document_root/api.php;
|
||||
fastcgi_param REQUEST_URI $request_uri;
|
||||
# SSE 关键配置:禁用所有缓冲
|
||||
fastcgi_buffering off;
|
||||
fastcgi_cache off;
|
||||
proxy_buffering off;
|
||||
# 禁用 Nginx 缓冲
|
||||
add_header X-Accel-Buffering no;
|
||||
# SSE 超时设置(5分钟)
|
||||
fastcgi_read_timeout 300s;
|
||||
}
|
||||
|
||||
# 静态资源直接服务
|
||||
location /assets/ {
|
||||
alias /path/to/ai-chat/assets/;
|
||||
expires 30d;
|
||||
add_header Cache-Control "public, immutable";
|
||||
}
|
||||
|
||||
# 上传文件直接服务
|
||||
location /uploads/ {
|
||||
alias /path/to/ai-chat/uploads/;
|
||||
expires 7d;
|
||||
add_header Cache-Control "public";
|
||||
}
|
||||
|
||||
# PHP 文件处理
|
||||
location ~ \.php$ {
|
||||
fastcgi_pass unix:/tmp/php-cgi-80.sock;
|
||||
fastcgi_index index.php;
|
||||
include fastcgi_params;
|
||||
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
|
||||
}
|
||||
|
||||
# 禁止访问非公开目录
|
||||
location ~ /(app|config|vendor|tests|scripts|\.cospec)/ {
|
||||
deny all;
|
||||
return 404;
|
||||
}
|
||||
|
||||
# 禁止访问隐藏文件
|
||||
location ~ /\. {
|
||||
deny all;
|
||||
return 404;
|
||||
}
|
||||
|
||||
# 上传文件大小限制
|
||||
client_max_body_size 10m;
|
||||
}
|
||||
0
public/.gitkeep
Normal file
0
public/.gitkeep
Normal file
138
public/api.php
Normal file
138
public/api.php
Normal file
@@ -0,0 +1,138 @@
|
||||
<?php
|
||||
|
||||
require_once __DIR__ . '/../vendor/autoload.php';
|
||||
|
||||
use App\Config\AppConfig;
|
||||
use App\Controllers\AuthController;
|
||||
use App\Controllers\SessionController;
|
||||
use App\Controllers\MessageController;
|
||||
use App\Controllers\ChatController;
|
||||
use App\Controllers\UploadController;
|
||||
use App\Controllers\ConfigController;
|
||||
use App\Controllers\InstallController;
|
||||
use App\Middleware\AuthMiddleware;
|
||||
use App\Middleware\AdminMiddleware;
|
||||
|
||||
// 设置响应头
|
||||
header('Content-Type: application/json; charset=utf-8');
|
||||
|
||||
// CORS 设置
|
||||
$corsOrigin = AppConfig::get('corsOrigin', '*');
|
||||
header("Access-Control-Allow-Origin: {$corsOrigin}");
|
||||
header('Access-Control-Allow-Methods: GET, POST, PUT, DELETE, OPTIONS');
|
||||
header('Access-Control-Allow-Headers: Content-Type, Authorization');
|
||||
|
||||
// 处理 OPTIONS 预检请求
|
||||
if ($_SERVER['REQUEST_METHOD'] === 'OPTIONS') {
|
||||
http_response_code(200);
|
||||
exit;
|
||||
}
|
||||
|
||||
// 路由表定义
|
||||
$routes = [
|
||||
// 认证路由(无需认证)
|
||||
['POST', 'auth/login', [AuthController::class, 'login'], false, false],
|
||||
|
||||
// 认证路由(需认证)
|
||||
['GET', 'auth/me', [AuthController::class, 'me'], true, false],
|
||||
|
||||
// 会话路由
|
||||
['GET', 'sessions', [SessionController::class, 'index'], true, false],
|
||||
['POST', 'sessions', [SessionController::class, 'create'], true, false],
|
||||
['PUT', 'sessions/{id}', [SessionController::class, 'update'], true, false],
|
||||
['DELETE', 'sessions/{id}', [SessionController::class, 'delete'], true, false],
|
||||
|
||||
// 消息路由
|
||||
['GET', 'sessions/{id}/messages', [MessageController::class, 'index'], true, false],
|
||||
['POST', 'sessions/{id}/messages', [MessageController::class, 'create'], true, false],
|
||||
|
||||
// 聊天路由
|
||||
['POST', 'chat/completions', [ChatController::class, 'completions'], true, false],
|
||||
|
||||
// 上传路由
|
||||
['POST', 'upload', [UploadController::class, 'upload'], true, false],
|
||||
|
||||
// 配置路由
|
||||
['GET', 'config', [ConfigController::class, 'getConfig'], true, false],
|
||||
['PUT', 'config', [ConfigController::class, 'updateConfig'], true, true],
|
||||
|
||||
// 人格路由
|
||||
['GET', 'personalities', [ConfigController::class, 'listPersonalities'], true, false],
|
||||
['POST', 'personalities', [ConfigController::class, 'createPersonality'], true, true],
|
||||
['PUT', 'personalities/{id}', [ConfigController::class, 'updatePersonality'], true, true],
|
||||
['DELETE', 'personalities/{id}', [ConfigController::class, 'deletePersonality'], true, true],
|
||||
|
||||
// 安装路由(无需认证)
|
||||
['GET', 'install/status', [InstallController::class, 'status'], false, false],
|
||||
['POST', 'install/test-db', [InstallController::class, 'testDb'], false, false],
|
||||
['POST', 'install/setup', [InstallController::class, 'setup'], false, false],
|
||||
];
|
||||
|
||||
// 解析请求路径
|
||||
$requestUri = $_SERVER['REQUEST_URI'];
|
||||
$path = parse_url($requestUri, PHP_URL_PATH);
|
||||
$basePath = '/api/';
|
||||
|
||||
// 去除 /api/ 前缀
|
||||
if (str_starts_with($path, $basePath)) {
|
||||
$path = substr($path, strlen($basePath));
|
||||
}
|
||||
$path = rtrim($path, '/');
|
||||
|
||||
$method = $_SERVER['REQUEST_METHOD'];
|
||||
|
||||
// 路由匹配
|
||||
$matched = false;
|
||||
|
||||
try {
|
||||
foreach ($routes as $route) {
|
||||
[$routeMethod, $routePattern, $handler, $needAuth, $needAdmin] = $route;
|
||||
|
||||
if ($method !== $routeMethod) {
|
||||
continue;
|
||||
}
|
||||
|
||||
// 将路由模式转换为正则表达式
|
||||
$paramNames = [];
|
||||
$regexPattern = preg_replace_callback('/\{(\w+)\}/', function ($m) use (&$paramNames) {
|
||||
$paramNames[] = $m[1];
|
||||
return '([^/]+)';
|
||||
}, $routePattern);
|
||||
|
||||
$regex = '#^' . $regexPattern . '$#';
|
||||
|
||||
if (preg_match($regex, $path, $matches)) {
|
||||
$matched = true;
|
||||
|
||||
// 提取路径参数
|
||||
$params = [];
|
||||
for ($i = 0; $i < count($paramNames); $i++) {
|
||||
$params[$paramNames[$i]] = $matches[$i + 1];
|
||||
}
|
||||
|
||||
// 认证检查
|
||||
if ($needAuth) {
|
||||
AuthMiddleware::handle();
|
||||
}
|
||||
|
||||
// 管理员检查
|
||||
if ($needAdmin) {
|
||||
AdminMiddleware::handle();
|
||||
}
|
||||
|
||||
// 调用控制器方法
|
||||
$controllerMethod = $handler[1];
|
||||
$handler[0]::$controllerMethod(...array_values($params));
|
||||
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (!$matched) {
|
||||
http_response_code(404);
|
||||
echo json_encode(['success' => false, 'message' => '接口不存在']);
|
||||
}
|
||||
} catch (\Throwable $e) {
|
||||
http_response_code(500);
|
||||
echo json_encode(['success' => false, 'message' => '服务器内部错误']);
|
||||
}
|
||||
5
public/chat.php
Normal file
5
public/chat.php
Normal file
@@ -0,0 +1,5 @@
|
||||
<?php
|
||||
// 聊天页面需要认证检查(但使用前端 JS 检查 token)
|
||||
require_once __DIR__ . '/../app/Views/layout/header.php';
|
||||
require_once __DIR__ . '/../app/Views/chat.php';
|
||||
require_once __DIR__ . '/../app/Views/layout/footer.php';
|
||||
4
public/config.php
Normal file
4
public/config.php
Normal file
@@ -0,0 +1,4 @@
|
||||
<?php
|
||||
require_once __DIR__ . '/../app/Views/layout/header.php';
|
||||
require_once __DIR__ . '/../app/Views/config.php';
|
||||
require_once __DIR__ . '/../app/Views/layout/footer.php';
|
||||
338
public/install.php
Normal file
338
public/install.php
Normal file
@@ -0,0 +1,338 @@
|
||||
<?php
|
||||
// 检查是否已安装
|
||||
$configFile = __DIR__ . '/../config/db-config.json';
|
||||
if (file_exists($configFile)) {
|
||||
header('Location: /login.php');
|
||||
exit;
|
||||
}
|
||||
?>
|
||||
<!DOCTYPE html>
|
||||
<html lang="zh-CN">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<title>AI Chat - 安装向导</title>
|
||||
<link rel="stylesheet" href="/assets/css/style.css">
|
||||
<style>
|
||||
.install-container { max-width: 700px; margin: 40px auto; padding: 0 20px; }
|
||||
.step-content { display: none; background: var(--bg-card); border-radius: var(--radius); padding: 24px; }
|
||||
.step-content.active { display: block; }
|
||||
.check-item { display: flex; justify-content: space-between; padding: 8px 0; border-bottom: 1px solid var(--border-color); }
|
||||
.check-item .status { font-weight: bold; }
|
||||
.check-item .status.pass { color: var(--success); }
|
||||
.check-item .status.fail { color: var(--danger); }
|
||||
.provider-item { background: var(--bg-secondary); padding: 16px; border-radius: var(--radius); margin-bottom: 12px; }
|
||||
.provider-item .form-group { margin-bottom: 8px; }
|
||||
h1 { text-align: center; margin-bottom: 30px; color: var(--primary); }
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="install-container">
|
||||
<h1>🤖 AI Chat 安装向导</h1>
|
||||
|
||||
<!-- 步骤指示器 -->
|
||||
<ul class="step-indicator">
|
||||
<li class="step active" data-step="1">1. 环境检查</li>
|
||||
<li class="step" data-step="2">2. 数据库配置</li>
|
||||
<li class="step" data-step="3">3. 应用配置</li>
|
||||
<li class="step" data-step="4">4. 管理员账户</li>
|
||||
<li class="step" data-step="5">5. AI供应商</li>
|
||||
</ul>
|
||||
|
||||
<!-- 步骤1:环境检查 -->
|
||||
<div class="step-content active" id="step1">
|
||||
<h2>环境检查</h2>
|
||||
<div id="envChecks">
|
||||
<!-- 由 JS 动态填充 -->
|
||||
</div>
|
||||
<div class="btn-group">
|
||||
<button class="btn btn-primary" onclick="InstallWizard.nextStep()">下一步</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 步骤2:数据库配置 -->
|
||||
<div class="step-content" id="step2">
|
||||
<h2>数据库配置</h2>
|
||||
<div class="form-group">
|
||||
<label>主机地址</label>
|
||||
<input type="text" id="dbHost" value="127.0.0.1" placeholder="数据库主机">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>端口</label>
|
||||
<input type="number" id="dbPort" value="3306" placeholder="数据库端口">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>数据库名</label>
|
||||
<input type="text" id="dbName" placeholder="数据库名称">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>用户名</label>
|
||||
<input type="text" id="dbUser" placeholder="数据库用户名">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>密码</label>
|
||||
<input type="password" id="dbPassword" placeholder="数据库密码">
|
||||
</div>
|
||||
<div id="dbTestResult"></div>
|
||||
<div class="btn-group">
|
||||
<button class="btn btn-secondary" onclick="InstallWizard.prevStep()">上一步</button>
|
||||
<button class="btn btn-secondary" onclick="InstallWizard.testDb()">测试连接</button>
|
||||
<button class="btn btn-primary" onclick="InstallWizard.nextStep()">下一步</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 步骤3:应用配置 -->
|
||||
<div class="step-content" id="step3">
|
||||
<h2>应用配置</h2>
|
||||
<div class="form-group">
|
||||
<label>JWT 密钥</label>
|
||||
<div style="display:flex;gap:8px;">
|
||||
<input type="text" id="jwtSecret" placeholder="留空则自动生成" style="flex:1;">
|
||||
<button class="btn btn-secondary btn-sm" onclick="document.getElementById('jwtSecret').value=InstallWizard.generateSecret()">自动生成</button>
|
||||
</div>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>JWT 过期时间(秒)</label>
|
||||
<input type="number" id="jwtExpiry" value="86400" placeholder="默认 86400(24小时)">
|
||||
</div>
|
||||
<div class="btn-group">
|
||||
<button class="btn btn-secondary" onclick="InstallWizard.prevStep()">上一步</button>
|
||||
<button class="btn btn-primary" onclick="InstallWizard.nextStep()">下一步</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 步骤4:管理员账户 -->
|
||||
<div class="step-content" id="step4">
|
||||
<h2>创建管理员账户</h2>
|
||||
<div class="form-group">
|
||||
<label>用户名</label>
|
||||
<input type="text" id="adminUsername" placeholder="管理员用户名">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>密码</label>
|
||||
<input type="password" id="adminPassword" placeholder="至少6位密码">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>确认密码</label>
|
||||
<input type="password" id="adminPasswordConfirm" placeholder="再次输入密码">
|
||||
</div>
|
||||
<div class="btn-group">
|
||||
<button class="btn btn-secondary" onclick="InstallWizard.prevStep()">上一步</button>
|
||||
<button class="btn btn-primary" onclick="InstallWizard.nextStep()">下一步</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 步骤5:AI供应商 -->
|
||||
<div class="step-content" id="step5">
|
||||
<h2>AI 供应商配置</h2>
|
||||
<p style="color:var(--text-secondary);margin-bottom:16px;">至少配置一个 AI 供应商</p>
|
||||
<div id="providerList">
|
||||
<!-- 由 JS 动态管理 -->
|
||||
</div>
|
||||
<button class="btn btn-secondary" onclick="InstallWizard.addProvider()" style="margin-top:8px;">+ 添加供应商</button>
|
||||
<div id="installResult"></div>
|
||||
<div class="btn-group">
|
||||
<button class="btn btn-secondary" onclick="InstallWizard.prevStep()">上一步</button>
|
||||
<button class="btn btn-primary" id="installBtn" onclick="InstallWizard.runInstall()">完成安装</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<script>
|
||||
const InstallWizard = {
|
||||
currentStep: 1,
|
||||
totalSteps: 5,
|
||||
|
||||
init() {
|
||||
this.checkEnv();
|
||||
this.addProvider(); // 默认添加一个供应商表单
|
||||
},
|
||||
|
||||
checkEnv() {
|
||||
const checks = [
|
||||
{ name: 'PHP 版本 >= 8.0', pass: <?php echo version_compare(PHP_VERSION, '8.0.0', '>=') ? 'true' : 'false'; ?> },
|
||||
{ name: 'PDO 扩展', pass: <?php echo extension_loaded('pdo') ? 'true' : 'false'; ?> },
|
||||
{ name: 'cURL 扩展', pass: <?php echo extension_loaded('curl') ? 'true' : 'false'; ?> },
|
||||
{ name: 'uploads/ 目录可写', pass: <?php echo is_writable(__DIR__ . '/../uploads') ? 'true' : 'false'; ?> },
|
||||
{ name: 'config/ 目录可写', pass: <?php echo is_writable(__DIR__ . '/../config') ? 'true' : 'false'; ?> }
|
||||
];
|
||||
|
||||
const container = document.getElementById('envChecks');
|
||||
container.innerHTML = checks.map(c => `
|
||||
<div class="check-item">
|
||||
<span>${c.name}</span>
|
||||
<span class="status ${c.pass ? 'pass' : 'fail'}">${c.pass ? '✓ 通过' : '✗ 未通过'}</span>
|
||||
</div>
|
||||
`).join('');
|
||||
},
|
||||
|
||||
nextStep() {
|
||||
if (this.currentStep === 4) {
|
||||
// 验证管理员密码
|
||||
const pwd = document.getElementById('adminPassword').value;
|
||||
const confirm = document.getElementById('adminPasswordConfirm').value;
|
||||
const username = document.getElementById('adminUsername').value;
|
||||
if (!username) { alert('请输入管理员用户名'); return; }
|
||||
if (pwd.length < 6) { alert('密码至少6位'); return; }
|
||||
if (pwd !== confirm) { alert('两次密码不一致'); return; }
|
||||
}
|
||||
if (this.currentStep < this.totalSteps) {
|
||||
this.currentStep++;
|
||||
this.updateSteps();
|
||||
}
|
||||
},
|
||||
|
||||
prevStep() {
|
||||
if (this.currentStep > 1) {
|
||||
this.currentStep--;
|
||||
this.updateSteps();
|
||||
}
|
||||
},
|
||||
|
||||
updateSteps() {
|
||||
document.querySelectorAll('.step-content').forEach((el, i) => {
|
||||
el.classList.toggle('active', i + 1 === this.currentStep);
|
||||
});
|
||||
document.querySelectorAll('.step-indicator .step').forEach((el, i) => {
|
||||
el.classList.remove('active', 'completed');
|
||||
if (i + 1 === this.currentStep) el.classList.add('active');
|
||||
if (i + 1 < this.currentStep) el.classList.add('completed');
|
||||
});
|
||||
},
|
||||
|
||||
async testDb() {
|
||||
const result = document.getElementById('dbTestResult');
|
||||
try {
|
||||
const response = await fetch('/api/install/test-db', {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify({
|
||||
host: document.getElementById('dbHost').value,
|
||||
port: parseInt(document.getElementById('dbPort').value),
|
||||
user: document.getElementById('dbUser').value,
|
||||
password: document.getElementById('dbPassword').value,
|
||||
database: document.getElementById('dbName').value
|
||||
})
|
||||
});
|
||||
const data = await response.json();
|
||||
result.innerHTML = `<div class="alert ${data.success ? 'alert-success' : 'alert-error'}">${data.message}</div>`;
|
||||
} catch (err) {
|
||||
result.innerHTML = '<div class="alert alert-error">连接失败: ' + err.message + '</div>';
|
||||
}
|
||||
},
|
||||
|
||||
addProvider() {
|
||||
const list = document.getElementById('providerList');
|
||||
const index = list.children.length;
|
||||
const html = `
|
||||
<div class="provider-item" data-index="${index}">
|
||||
<div class="form-group">
|
||||
<label>供应商名称</label>
|
||||
<input type="text" class="provider-name" placeholder="如:OpenAI、DeepSeek">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>API URL</label>
|
||||
<input type="text" class="provider-url" placeholder="如:https://api.openai.com">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>API Key</label>
|
||||
<input type="password" class="provider-key" placeholder="API 密钥">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>可用模型(逗号分隔)</label>
|
||||
<input type="text" class="provider-models" placeholder="如:gpt-3.5-turbo, gpt-4">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label>供应商类型</label>
|
||||
<select class="provider-type">
|
||||
<option value="newapi">OpenAI 兼容</option>
|
||||
<option value="openai">OpenAI 官方</option>
|
||||
<option value="claude">Claude (Anthropic)</option>
|
||||
</select>
|
||||
</div>
|
||||
${index > 0 ? '<button class="btn btn-danger btn-sm" onclick="this.parentElement.remove()">删除</button>' : ''}
|
||||
</div>
|
||||
`;
|
||||
list.insertAdjacentHTML('beforeend', html);
|
||||
},
|
||||
|
||||
generateSecret() {
|
||||
const chars = '0123456789abcdef';
|
||||
let result = '';
|
||||
for (let i = 0; i < 64; i++) {
|
||||
result += chars[Math.floor(Math.random() * chars.length)];
|
||||
}
|
||||
return result;
|
||||
},
|
||||
|
||||
async runInstall() {
|
||||
const btn = document.getElementById('installBtn');
|
||||
const result = document.getElementById('installResult');
|
||||
|
||||
// 收集供应商数据
|
||||
const providers = [];
|
||||
document.querySelectorAll('.provider-item').forEach(item => {
|
||||
const models = item.querySelector('.provider-models').value.split(',').map(m => m.trim()).filter(m => m);
|
||||
providers.push({
|
||||
name: item.querySelector('.provider-name').value,
|
||||
apiUrl: item.querySelector('.provider-url').value,
|
||||
apiKey: item.querySelector('.provider-key').value,
|
||||
models: models,
|
||||
type: item.querySelector('.provider-type').value,
|
||||
enabled: true
|
||||
});
|
||||
});
|
||||
|
||||
if (providers.length === 0 || !providers[0].name || !providers[0].apiKey) {
|
||||
alert('请至少配置一个完整的供应商');
|
||||
return;
|
||||
}
|
||||
|
||||
btn.disabled = true;
|
||||
btn.textContent = '安装中...';
|
||||
|
||||
const setupData = {
|
||||
username: document.getElementById('adminUsername').value,
|
||||
password: document.getElementById('adminPassword').value,
|
||||
dbConfig: {
|
||||
host: document.getElementById('dbHost').value,
|
||||
port: parseInt(document.getElementById('dbPort').value),
|
||||
user: document.getElementById('dbUser').value,
|
||||
password: document.getElementById('dbPassword').value,
|
||||
database: document.getElementById('dbName').value
|
||||
},
|
||||
appConfig: {
|
||||
jwtSecret: document.getElementById('jwtSecret').value || undefined,
|
||||
jwtExpiry: parseInt(document.getElementById('jwtExpiry').value) || 86400
|
||||
},
|
||||
providers: providers
|
||||
};
|
||||
|
||||
try {
|
||||
const response = await fetch('/api/install/setup', {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify(setupData)
|
||||
});
|
||||
const data = await response.json();
|
||||
|
||||
if (data.success) {
|
||||
result.innerHTML = '<div class="alert alert-success">✓ 安装成功!正在跳转到登录页...</div>';
|
||||
setTimeout(() => { window.location.href = '/login.php'; }, 2000);
|
||||
} else {
|
||||
result.innerHTML = '<div class="alert alert-error">安装失败: ' + data.message + '</div>';
|
||||
btn.disabled = false;
|
||||
btn.textContent = '完成安装';
|
||||
}
|
||||
} catch (err) {
|
||||
result.innerHTML = '<div class="alert alert-error">安装失败: ' + err.message + '</div>';
|
||||
btn.disabled = false;
|
||||
btn.textContent = '完成安装';
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
InstallWizard.init();
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
||||
4
public/login.php
Normal file
4
public/login.php
Normal file
@@ -0,0 +1,4 @@
|
||||
<?php
|
||||
require_once __DIR__ . '/../app/Views/layout/header.php';
|
||||
require_once __DIR__ . '/../app/Views/login.php';
|
||||
require_once __DIR__ . '/../app/Views/layout/footer.php';
|
||||
0
scripts/.gitkeep
Normal file
0
scripts/.gitkeep
Normal file
0
tests/.gitkeep
Normal file
0
tests/.gitkeep
Normal file
0
tests/backend/.gitkeep
Normal file
0
tests/backend/.gitkeep
Normal file
0
tests/frontend/.gitkeep
Normal file
0
tests/frontend/.gitkeep
Normal file
0
uploads/.gitkeep
Normal file
0
uploads/.gitkeep
Normal file
Reference in New Issue
Block a user