laravel-echo 此路由不支持 POST 方法
laravel-echo The POST method is not supported for this route
所以我正在尝试使用 Laravel Echo 连接到私人频道。我能够连接到 public 频道,但我在验证(使用正确的 Bearer 令牌登录)私人频道时遇到问题 - 从 laravel-echo-server start
收到以下错误
L A R A V E L E C H O S E R V E R
version 1.6.2
⚠ Starting server in DEV mode...
✔ Running at localhost on port 6001
✔ Channels are ready.
✔ Listening for http events...
✔ Listening for redis events...
Server ready!
[10:58:39 PM] - Preparing authentication request to: https://example.co.uk
[10:58:39 PM] - Sending auth request to: https://example.co.uk/api/broadcasting/auth/
⚠ [10:58:39 PM] - K1cJaXs7jWMBNpF9AAAA could not be authenticated to private-sonect_database_user.2
{"errors":{"message":"The POST method is not supported for this route. Supported methods: GET, HEAD."}}
Client can not be authenticated, got HTTP status 401
我对错误 POST method is not supported
感到困惑,因为所有 POST、PUT、PATCH 和 DELETE 都在我的 Laravel 项目中工作。此外,我可以使用带有 POST 请求的邮递员点击 https://example.co.uk/api/broadcasting/auth/
。
这是 php artisan route:list
的示例
| Domain | Method | URI | Name | Action | Middleware |
| | GET|POST|HEAD | api/broadcasting/auth | | Illuminate\Broadcasting\BroadcastController@authenticate | auth:api |
PlayerEvent.php
namespace App\Events;
use App\Models\User;
use Illuminate\Broadcasting\Channel;
use Illuminate\Broadcasting\InteractsWithSockets;
use Illuminate\Broadcasting\PresenceChannel;
use Illuminate\Broadcasting\PrivateChannel;
use Illuminate\Contracts\Broadcasting\ShouldBroadcast;
use Illuminate\Contracts\Broadcasting\ShouldBroadcastNow;
use Illuminate\Foundation\Events\Dispatchable;
use Illuminate\Queue\SerializesModels;
class PlayerEvent implements ShouldBroadcast
{
use Dispatchable, InteractsWithSockets, SerializesModels;
public $user;
/**
* Create a new event instance.
*
* @param User $user
* @return void
*/
public function __construct(User $user)
{
$this->user = $user;
}
/**
* Get the channels the event should broadcast on.
*
* @return \Illuminate\Broadcasting\Channel|array
*/
public function broadcastOn()
{
return new PrivateChannel('user.' . $this->user->id);
}
/**
* The event's broadcast name.
*
* @return string
*/
public function broadcastAs()
{
return 'PlayerEvent';
}
/**
* The event's broadcast name.
*
* @return array
*/
public function broadcastWith()
{
return ['message' => 'Hello world'];
}
}
BroadcaseServiceProvider.php
public function boot()
{
Broadcast::routes(["prefix" => "api", "middleware" => "auth:api"]);
require base_path('routes/channels.php');
}
config/app.php
/*
* Application Service Providers...
*/
App\Providers\AppServiceProvider::class,
App\Providers\AuthServiceProvider::class,
App\Providers\BroadcastServiceProvider::class,
App\Providers\EventServiceProvider::class,
App\Providers\RouteServiceProvider::class,
routes/channel.php
Broadcast::channel('user.{user}', function ($user) {
return true;
});
laravel-echo-server.json
{
"authHost": "https://example.co.uk",
"authEndpoint": "/api/broadcasting/auth/",
"clients": [
{
"appId": "4cb2c72910d85007",
"key": "d3af62258887434b9900b89a937cb7f7"
}
],
"database": "redis",
"databaseConfig": {
"redis": {},
"sqlite": {
"databasePath": "/database/laravel-echo-server.sqlite"
},
"publishPresence": true
},
"devMode": true,
"host": null,
"port": "6001",
"protocol": "https",
"socketio": {},
"secureOptions": 67108864,
"sslCertPath": "/var/www/example.co.uk/letsencrypt/live/example.co.uk/fullchain.pem",
"sslKeyPath": "/var/www/example.co.uk/letsencrypt/live/example.co.uk/privkey.pem",
"sslCertChainPath": "",
"sslPassphrase": "",
"subscribers": {
"http": true,
"redis": true
},
"apiOriginAllow": {
"allowCors": true,
"allowOrigin": "http://localhost:4200/",
"allowMethods": "GET, POST, PUT, PATCH, DELETE",
"allowHeaders": "Origin, Content-Type, X-Auth-Token, X-Requested-With, Accept, Authorization, X-CSRF-TOKEN, X-Socket-Id"
}
}
我在前端使用 angular
echo.service.ts
import {Injectable} from "@angular/core";
import {environment} from "../../../environments/environment";
import Echo from "laravel-echo";
import {UserModel} from "../models/user.model";
import * as io from "socket.io-client";
@Injectable({
providedIn: 'root'
})
export class EchoService {
public echo: Echo;
constructor() {
window['io'] = io;
this.echo = window['Echo'] = new Echo({
broadcaster: 'socket.io',
host: environment.websocketUrl, // https://example.co.uk:6001
auth: {
headers: {
Authorization: `Bearer ${localStorage.getItem('token')}`
}
}
});
}
/** privateChannel('user', 'PlayerEvent', this.user); Called from elsewhere */
privateChannel(channelName: string, eventName: string, user: UserModel) {
console.log('privateChannel');
this.echo.private( environment.broadcastChannelPrefix + channelName + '.' + user.id)
.listen('.' + eventName, data => {
console.log(data);
});
}
}
Laravel: 7.12.0
laravel-回显:^1.8.0
socket.io-客户端:^2.3.0
Angular: 9.1.9
所以我想通了。有点明显,当我意识到...
这与我的开发环境有关 - 我有一个虚拟机 运行,我在其中编辑了我的 /etc/hosts/
文件以指向 example.co.uk。由于 laravel-echo 从服务器对用户进行身份验证,example.co.uk 现在指向我的实时服务器,该路由未正确设置。
将我在虚拟机上的 /etc/hosts
文件更改为指向:
192.168.33.10 example.co.uk
修复了问题
所以我正在尝试使用 Laravel Echo 连接到私人频道。我能够连接到 public 频道,但我在验证(使用正确的 Bearer 令牌登录)私人频道时遇到问题 - 从 laravel-echo-server start
L A R A V E L E C H O S E R V E R
version 1.6.2
⚠ Starting server in DEV mode...
✔ Running at localhost on port 6001
✔ Channels are ready.
✔ Listening for http events...
✔ Listening for redis events...
Server ready!
[10:58:39 PM] - Preparing authentication request to: https://example.co.uk
[10:58:39 PM] - Sending auth request to: https://example.co.uk/api/broadcasting/auth/
⚠ [10:58:39 PM] - K1cJaXs7jWMBNpF9AAAA could not be authenticated to private-sonect_database_user.2
{"errors":{"message":"The POST method is not supported for this route. Supported methods: GET, HEAD."}}
Client can not be authenticated, got HTTP status 401
我对错误 POST method is not supported
感到困惑,因为所有 POST、PUT、PATCH 和 DELETE 都在我的 Laravel 项目中工作。此外,我可以使用带有 POST 请求的邮递员点击 https://example.co.uk/api/broadcasting/auth/
。
这是 php artisan route:list
| Domain | Method | URI | Name | Action | Middleware |
| | GET|POST|HEAD | api/broadcasting/auth | | Illuminate\Broadcasting\BroadcastController@authenticate | auth:api |
PlayerEvent.php
namespace App\Events;
use App\Models\User;
use Illuminate\Broadcasting\Channel;
use Illuminate\Broadcasting\InteractsWithSockets;
use Illuminate\Broadcasting\PresenceChannel;
use Illuminate\Broadcasting\PrivateChannel;
use Illuminate\Contracts\Broadcasting\ShouldBroadcast;
use Illuminate\Contracts\Broadcasting\ShouldBroadcastNow;
use Illuminate\Foundation\Events\Dispatchable;
use Illuminate\Queue\SerializesModels;
class PlayerEvent implements ShouldBroadcast
{
use Dispatchable, InteractsWithSockets, SerializesModels;
public $user;
/**
* Create a new event instance.
*
* @param User $user
* @return void
*/
public function __construct(User $user)
{
$this->user = $user;
}
/**
* Get the channels the event should broadcast on.
*
* @return \Illuminate\Broadcasting\Channel|array
*/
public function broadcastOn()
{
return new PrivateChannel('user.' . $this->user->id);
}
/**
* The event's broadcast name.
*
* @return string
*/
public function broadcastAs()
{
return 'PlayerEvent';
}
/**
* The event's broadcast name.
*
* @return array
*/
public function broadcastWith()
{
return ['message' => 'Hello world'];
}
}
BroadcaseServiceProvider.php
public function boot()
{
Broadcast::routes(["prefix" => "api", "middleware" => "auth:api"]);
require base_path('routes/channels.php');
}
config/app.php
/*
* Application Service Providers...
*/
App\Providers\AppServiceProvider::class,
App\Providers\AuthServiceProvider::class,
App\Providers\BroadcastServiceProvider::class,
App\Providers\EventServiceProvider::class,
App\Providers\RouteServiceProvider::class,
routes/channel.php
Broadcast::channel('user.{user}', function ($user) {
return true;
});
laravel-echo-server.json
{
"authHost": "https://example.co.uk",
"authEndpoint": "/api/broadcasting/auth/",
"clients": [
{
"appId": "4cb2c72910d85007",
"key": "d3af62258887434b9900b89a937cb7f7"
}
],
"database": "redis",
"databaseConfig": {
"redis": {},
"sqlite": {
"databasePath": "/database/laravel-echo-server.sqlite"
},
"publishPresence": true
},
"devMode": true,
"host": null,
"port": "6001",
"protocol": "https",
"socketio": {},
"secureOptions": 67108864,
"sslCertPath": "/var/www/example.co.uk/letsencrypt/live/example.co.uk/fullchain.pem",
"sslKeyPath": "/var/www/example.co.uk/letsencrypt/live/example.co.uk/privkey.pem",
"sslCertChainPath": "",
"sslPassphrase": "",
"subscribers": {
"http": true,
"redis": true
},
"apiOriginAllow": {
"allowCors": true,
"allowOrigin": "http://localhost:4200/",
"allowMethods": "GET, POST, PUT, PATCH, DELETE",
"allowHeaders": "Origin, Content-Type, X-Auth-Token, X-Requested-With, Accept, Authorization, X-CSRF-TOKEN, X-Socket-Id"
}
}
我在前端使用 angular
echo.service.ts
import {Injectable} from "@angular/core";
import {environment} from "../../../environments/environment";
import Echo from "laravel-echo";
import {UserModel} from "../models/user.model";
import * as io from "socket.io-client";
@Injectable({
providedIn: 'root'
})
export class EchoService {
public echo: Echo;
constructor() {
window['io'] = io;
this.echo = window['Echo'] = new Echo({
broadcaster: 'socket.io',
host: environment.websocketUrl, // https://example.co.uk:6001
auth: {
headers: {
Authorization: `Bearer ${localStorage.getItem('token')}`
}
}
});
}
/** privateChannel('user', 'PlayerEvent', this.user); Called from elsewhere */
privateChannel(channelName: string, eventName: string, user: UserModel) {
console.log('privateChannel');
this.echo.private( environment.broadcastChannelPrefix + channelName + '.' + user.id)
.listen('.' + eventName, data => {
console.log(data);
});
}
}
Laravel: 7.12.0
laravel-回显:^1.8.0
socket.io-客户端:^2.3.0
Angular: 9.1.9
所以我想通了。有点明显,当我意识到...
这与我的开发环境有关 - 我有一个虚拟机 运行,我在其中编辑了我的 /etc/hosts/
文件以指向 example.co.uk。由于 laravel-echo 从服务器对用户进行身份验证,example.co.uk 现在指向我的实时服务器,该路由未正确设置。
将我在虚拟机上的 /etc/hosts
文件更改为指向:
192.168.33.10 example.co.uk
修复了问题