mirror of
https://github.com/pocketbase/pocketbase.git
synced 2025-04-05 01:55:46 +02:00
[#468] added additional realtime events
This commit is contained in:
parent
98cc8e2aee
commit
23fbfab63a
@ -6,6 +6,9 @@
|
|||||||
```go
|
```go
|
||||||
app.OnBeforeBootstrap()
|
app.OnBeforeBootstrap()
|
||||||
app.OnAfterBootstrap()
|
app.OnAfterBootstrap()
|
||||||
|
app.OnRealtimeDisconnectRequest()
|
||||||
|
app.OnRealtimeBeforeMessageSend()
|
||||||
|
app.OnRealtimeAfterMessageSend()
|
||||||
```
|
```
|
||||||
|
|
||||||
- Refactored the `migrate` command to support **external JavaScript migration files** using an embedded JS interpreter ([goja](https://github.com/dop251/goja)).
|
- Refactored the `migrate` command to support **external JavaScript migration files** using an embedded JS interpreter ([goja](https://github.com/dop251/goja)).
|
||||||
|
@ -42,7 +42,14 @@ func (api *realtimeApi) connect(c echo.Context) error {
|
|||||||
// register new subscription client
|
// register new subscription client
|
||||||
client := subscriptions.NewDefaultClient()
|
client := subscriptions.NewDefaultClient()
|
||||||
api.app.SubscriptionsBroker().Register(client)
|
api.app.SubscriptionsBroker().Register(client)
|
||||||
defer api.app.SubscriptionsBroker().Unregister(client.Id())
|
defer func() {
|
||||||
|
api.app.OnRealtimeDisconnectRequest().Trigger(&core.RealtimeDisconnectEvent{
|
||||||
|
HttpContext: c,
|
||||||
|
Client: client,
|
||||||
|
})
|
||||||
|
|
||||||
|
api.app.SubscriptionsBroker().Unregister(client.Id())
|
||||||
|
}()
|
||||||
|
|
||||||
c.Response().Header().Set("Content-Type", "text/event-stream; charset=UTF-8")
|
c.Response().Header().Set("Content-Type", "text/event-stream; charset=UTF-8")
|
||||||
c.Response().Header().Set("Cache-Control", "no-store")
|
c.Response().Header().Set("Cache-Control", "no-store")
|
||||||
@ -51,12 +58,12 @@ func (api *realtimeApi) connect(c echo.Context) error {
|
|||||||
// https://nginx.org/en/docs/http/ngx_http_proxy_module.html#proxy_buffering
|
// https://nginx.org/en/docs/http/ngx_http_proxy_module.html#proxy_buffering
|
||||||
c.Response().Header().Set("X-Accel-Buffering", "no")
|
c.Response().Header().Set("X-Accel-Buffering", "no")
|
||||||
|
|
||||||
event := &core.RealtimeConnectEvent{
|
connectEvent := &core.RealtimeConnectEvent{
|
||||||
HttpContext: c,
|
HttpContext: c,
|
||||||
Client: client,
|
Client: client,
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := api.app.OnRealtimeConnectRequest().Trigger(event); err != nil {
|
if err := api.app.OnRealtimeConnectRequest().Trigger(connectEvent); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -65,10 +72,31 @@ func (api *realtimeApi) connect(c echo.Context) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// signalize established connection (aka. fire "connect" message)
|
// signalize established connection (aka. fire "connect" message)
|
||||||
fmt.Fprint(c.Response(), "id:"+client.Id()+"\n")
|
connectMsgEvent := &core.RealtimeMessageEvent{
|
||||||
fmt.Fprint(c.Response(), "event:PB_CONNECT\n")
|
HttpContext: c,
|
||||||
fmt.Fprint(c.Response(), "data:{\"clientId\":\""+client.Id()+"\"}\n\n")
|
Client: client,
|
||||||
c.Response().Flush()
|
Message: &subscriptions.Message{
|
||||||
|
Name: "PB_CONNECT",
|
||||||
|
Data: `{"clientId":"` + client.Id() + `"}`,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
connectMsgErr := api.app.OnRealtimeBeforeMessageSend().Trigger(connectMsgEvent, func(e *core.RealtimeMessageEvent) error {
|
||||||
|
w := e.HttpContext.Response()
|
||||||
|
fmt.Fprint(w, "id:"+client.Id()+"\n")
|
||||||
|
fmt.Fprint(w, "event:"+e.Message.Name+"\n")
|
||||||
|
fmt.Fprint(w, "data:"+e.Message.Data+"\n\n")
|
||||||
|
w.Flush()
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
if connectMsgErr != nil {
|
||||||
|
if api.app.IsDebug() {
|
||||||
|
log.Println("Realtime connection closed (failed to deliver PB_CONNECT):", client.Id(), connectMsgErr)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if err := api.app.OnRealtimeAfterMessageSend().Trigger(connectMsgEvent); err != nil && api.app.IsDebug() {
|
||||||
|
log.Println("OnRealtimeAfterMessageSend PB_CONNECT error:", err)
|
||||||
|
}
|
||||||
|
|
||||||
// start an idle timer to keep track of inactive/forgotten connections
|
// start an idle timer to keep track of inactive/forgotten connections
|
||||||
idleDuration := 5 * time.Minute
|
idleDuration := 5 * time.Minute
|
||||||
@ -88,11 +116,29 @@ func (api *realtimeApi) connect(c echo.Context) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
w := c.Response()
|
msgEvent := &core.RealtimeMessageEvent{
|
||||||
fmt.Fprint(w, "id:"+client.Id()+"\n")
|
HttpContext: c,
|
||||||
fmt.Fprint(w, "event:"+msg.Name+"\n")
|
Client: client,
|
||||||
fmt.Fprint(w, "data:"+msg.Data+"\n\n")
|
Message: &msg,
|
||||||
w.Flush()
|
}
|
||||||
|
msgErr := api.app.OnRealtimeBeforeMessageSend().Trigger(msgEvent, func(e *core.RealtimeMessageEvent) error {
|
||||||
|
w := e.HttpContext.Response()
|
||||||
|
fmt.Fprint(w, "id:"+e.Client.Id()+"\n")
|
||||||
|
fmt.Fprint(w, "event:"+e.Message.Name+"\n")
|
||||||
|
fmt.Fprint(w, "data:"+e.Message.Data+"\n\n")
|
||||||
|
w.Flush()
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
if msgErr != nil {
|
||||||
|
if api.app.IsDebug() {
|
||||||
|
log.Println("Realtime connection closed (failed to deliver message):", client.Id(), msgErr)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := api.app.OnRealtimeAfterMessageSend().Trigger(msgEvent); err != nil && api.app.IsDebug() {
|
||||||
|
log.Println("OnRealtimeAfterMessageSend error:", err)
|
||||||
|
}
|
||||||
|
|
||||||
idleTimer.Stop()
|
idleTimer.Stop()
|
||||||
idleTimer.Reset(idleDuration)
|
idleTimer.Reset(idleDuration)
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
package apis_test
|
package apis_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
"net/http"
|
"net/http"
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
@ -10,6 +11,7 @@ import (
|
|||||||
"github.com/pocketbase/pocketbase/core"
|
"github.com/pocketbase/pocketbase/core"
|
||||||
"github.com/pocketbase/pocketbase/models"
|
"github.com/pocketbase/pocketbase/models"
|
||||||
"github.com/pocketbase/pocketbase/tests"
|
"github.com/pocketbase/pocketbase/tests"
|
||||||
|
"github.com/pocketbase/pocketbase/tools/hook"
|
||||||
"github.com/pocketbase/pocketbase/tools/subscriptions"
|
"github.com/pocketbase/pocketbase/tools/subscriptions"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -25,7 +27,56 @@ func TestRealtimeConnect(t *testing.T) {
|
|||||||
`data:{"clientId":`,
|
`data:{"clientId":`,
|
||||||
},
|
},
|
||||||
ExpectedEvents: map[string]int{
|
ExpectedEvents: map[string]int{
|
||||||
"OnRealtimeConnectRequest": 1,
|
"OnRealtimeConnectRequest": 1,
|
||||||
|
"OnRealtimeBeforeMessageSend": 1,
|
||||||
|
"OnRealtimeAfterMessageSend": 1,
|
||||||
|
"OnRealtimeDisconnectRequest": 1,
|
||||||
|
},
|
||||||
|
AfterTestFunc: func(t *testing.T, app *tests.TestApp, e *echo.Echo) {
|
||||||
|
if len(app.SubscriptionsBroker().Clients()) != 0 {
|
||||||
|
t.Errorf("Expected the subscribers to be removed after connection close, found %d", len(app.SubscriptionsBroker().Clients()))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "PB_CONNECT interrupt",
|
||||||
|
Method: http.MethodGet,
|
||||||
|
Url: "/api/realtime",
|
||||||
|
ExpectedStatus: 200,
|
||||||
|
ExpectedEvents: map[string]int{
|
||||||
|
"OnRealtimeConnectRequest": 1,
|
||||||
|
"OnRealtimeBeforeMessageSend": 1,
|
||||||
|
"OnRealtimeDisconnectRequest": 1,
|
||||||
|
},
|
||||||
|
BeforeTestFunc: func(t *testing.T, app *tests.TestApp, e *echo.Echo) {
|
||||||
|
app.OnRealtimeBeforeMessageSend().Add(func(e *core.RealtimeMessageEvent) error {
|
||||||
|
if e.Message.Name == "PB_CONNECT" {
|
||||||
|
return errors.New("PB_CONNECT error")
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
},
|
||||||
|
AfterTestFunc: func(t *testing.T, app *tests.TestApp, e *echo.Echo) {
|
||||||
|
if len(app.SubscriptionsBroker().Clients()) != 0 {
|
||||||
|
t.Errorf("Expected the subscribers to be removed after connection close, found %d", len(app.SubscriptionsBroker().Clients()))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "Skipping/ignoring messages",
|
||||||
|
Method: http.MethodGet,
|
||||||
|
Url: "/api/realtime",
|
||||||
|
ExpectedStatus: 200,
|
||||||
|
ExpectedEvents: map[string]int{
|
||||||
|
"OnRealtimeConnectRequest": 1,
|
||||||
|
"OnRealtimeBeforeMessageSend": 1,
|
||||||
|
"OnRealtimeAfterMessageSend": 1,
|
||||||
|
"OnRealtimeDisconnectRequest": 1,
|
||||||
|
},
|
||||||
|
BeforeTestFunc: func(t *testing.T, app *tests.TestApp, e *echo.Echo) {
|
||||||
|
app.OnRealtimeBeforeMessageSend().Add(func(e *core.RealtimeMessageEvent) error {
|
||||||
|
return hook.StopPropagation
|
||||||
|
})
|
||||||
},
|
},
|
||||||
AfterTestFunc: func(t *testing.T, app *tests.TestApp, e *echo.Echo) {
|
AfterTestFunc: func(t *testing.T, app *tests.TestApp, e *echo.Echo) {
|
||||||
if len(app.SubscriptionsBroker().Clients()) != 0 {
|
if len(app.SubscriptionsBroker().Clients()) != 0 {
|
||||||
|
15
core/app.go
15
core/app.go
@ -176,6 +176,21 @@ type App interface {
|
|||||||
// the SSE client connection.
|
// the SSE client connection.
|
||||||
OnRealtimeConnectRequest() *hook.Hook[*RealtimeConnectEvent]
|
OnRealtimeConnectRequest() *hook.Hook[*RealtimeConnectEvent]
|
||||||
|
|
||||||
|
// OnRealtimeDisconnectRequest hook is triggered on disconnected/interrupted
|
||||||
|
// SSE client connection.
|
||||||
|
OnRealtimeDisconnectRequest() *hook.Hook[*RealtimeDisconnectEvent]
|
||||||
|
|
||||||
|
// OnRealtimeBeforeMessage hook is triggered right before sending
|
||||||
|
// an SSE message to a client.
|
||||||
|
//
|
||||||
|
// Returning [hook.StopPropagation] will prevent sending the message.
|
||||||
|
// Returning any other non-nil error will close the realtime connection.
|
||||||
|
OnRealtimeBeforeMessageSend() *hook.Hook[*RealtimeMessageEvent]
|
||||||
|
|
||||||
|
// OnRealtimeBeforeMessage hook is triggered right after sending
|
||||||
|
// an SSE message to a client.
|
||||||
|
OnRealtimeAfterMessageSend() *hook.Hook[*RealtimeMessageEvent]
|
||||||
|
|
||||||
// OnRealtimeBeforeSubscribeRequest hook is triggered before changing
|
// OnRealtimeBeforeSubscribeRequest hook is triggered before changing
|
||||||
// the client subscriptions, allowing you to further validate and
|
// the client subscriptions, allowing you to further validate and
|
||||||
// modify the submitted change.
|
// modify the submitted change.
|
||||||
|
18
core/base.go
18
core/base.go
@ -64,6 +64,9 @@ type BaseApp struct {
|
|||||||
|
|
||||||
// realtime api event hooks
|
// realtime api event hooks
|
||||||
onRealtimeConnectRequest *hook.Hook[*RealtimeConnectEvent]
|
onRealtimeConnectRequest *hook.Hook[*RealtimeConnectEvent]
|
||||||
|
onRealtimeDisconnectRequest *hook.Hook[*RealtimeDisconnectEvent]
|
||||||
|
onRealtimeBeforeMessageSend *hook.Hook[*RealtimeMessageEvent]
|
||||||
|
onRealtimeAfterMessageSend *hook.Hook[*RealtimeMessageEvent]
|
||||||
onRealtimeBeforeSubscribeRequest *hook.Hook[*RealtimeSubscribeEvent]
|
onRealtimeBeforeSubscribeRequest *hook.Hook[*RealtimeSubscribeEvent]
|
||||||
onRealtimeAfterSubscribeRequest *hook.Hook[*RealtimeSubscribeEvent]
|
onRealtimeAfterSubscribeRequest *hook.Hook[*RealtimeSubscribeEvent]
|
||||||
|
|
||||||
@ -153,6 +156,9 @@ func NewBaseApp(dataDir string, encryptionEnv string, isDebug bool) *BaseApp {
|
|||||||
|
|
||||||
// realtime API event hooks
|
// realtime API event hooks
|
||||||
onRealtimeConnectRequest: &hook.Hook[*RealtimeConnectEvent]{},
|
onRealtimeConnectRequest: &hook.Hook[*RealtimeConnectEvent]{},
|
||||||
|
onRealtimeDisconnectRequest: &hook.Hook[*RealtimeDisconnectEvent]{},
|
||||||
|
onRealtimeBeforeMessageSend: &hook.Hook[*RealtimeMessageEvent]{},
|
||||||
|
onRealtimeAfterMessageSend: &hook.Hook[*RealtimeMessageEvent]{},
|
||||||
onRealtimeBeforeSubscribeRequest: &hook.Hook[*RealtimeSubscribeEvent]{},
|
onRealtimeBeforeSubscribeRequest: &hook.Hook[*RealtimeSubscribeEvent]{},
|
||||||
onRealtimeAfterSubscribeRequest: &hook.Hook[*RealtimeSubscribeEvent]{},
|
onRealtimeAfterSubscribeRequest: &hook.Hook[*RealtimeSubscribeEvent]{},
|
||||||
|
|
||||||
@ -471,6 +477,18 @@ func (app *BaseApp) OnRealtimeConnectRequest() *hook.Hook[*RealtimeConnectEvent]
|
|||||||
return app.onRealtimeConnectRequest
|
return app.onRealtimeConnectRequest
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (app *BaseApp) OnRealtimeDisconnectRequest() *hook.Hook[*RealtimeDisconnectEvent] {
|
||||||
|
return app.onRealtimeDisconnectRequest
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *BaseApp) OnRealtimeBeforeMessageSend() *hook.Hook[*RealtimeMessageEvent] {
|
||||||
|
return app.onRealtimeBeforeMessageSend
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *BaseApp) OnRealtimeAfterMessageSend() *hook.Hook[*RealtimeMessageEvent] {
|
||||||
|
return app.onRealtimeAfterMessageSend
|
||||||
|
}
|
||||||
|
|
||||||
func (app *BaseApp) OnRealtimeBeforeSubscribeRequest() *hook.Hook[*RealtimeSubscribeEvent] {
|
func (app *BaseApp) OnRealtimeBeforeSubscribeRequest() *hook.Hook[*RealtimeSubscribeEvent] {
|
||||||
return app.onRealtimeBeforeSubscribeRequest
|
return app.onRealtimeBeforeSubscribeRequest
|
||||||
}
|
}
|
||||||
|
@ -61,6 +61,17 @@ type RealtimeConnectEvent struct {
|
|||||||
Client subscriptions.Client
|
Client subscriptions.Client
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type RealtimeDisconnectEvent struct {
|
||||||
|
HttpContext echo.Context
|
||||||
|
Client subscriptions.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
type RealtimeMessageEvent struct {
|
||||||
|
HttpContext echo.Context
|
||||||
|
Client subscriptions.Client
|
||||||
|
Message *subscriptions.Message
|
||||||
|
}
|
||||||
|
|
||||||
type RealtimeSubscribeEvent struct {
|
type RealtimeSubscribeEvent struct {
|
||||||
HttpContext echo.Context
|
HttpContext echo.Context
|
||||||
Client subscriptions.Client
|
Client subscriptions.Client
|
||||||
|
15
tests/app.go
15
tests/app.go
@ -222,6 +222,21 @@ func NewTestApp(optTestDataDir ...string) (*TestApp, error) {
|
|||||||
return nil
|
return nil
|
||||||
})
|
})
|
||||||
|
|
||||||
|
t.OnRealtimeDisconnectRequest().Add(func(e *core.RealtimeDisconnectEvent) error {
|
||||||
|
t.EventCalls["OnRealtimeDisconnectRequest"]++
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
|
||||||
|
t.OnRealtimeBeforeMessageSend().Add(func(e *core.RealtimeMessageEvent) error {
|
||||||
|
t.EventCalls["OnRealtimeBeforeMessageSend"]++
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
|
||||||
|
t.OnRealtimeAfterMessageSend().Add(func(e *core.RealtimeMessageEvent) error {
|
||||||
|
t.EventCalls["OnRealtimeAfterMessageSend"]++
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
|
||||||
t.OnRealtimeBeforeSubscribeRequest().Add(func(e *core.RealtimeSubscribeEvent) error {
|
t.OnRealtimeBeforeSubscribeRequest().Add(func(e *core.RealtimeSubscribeEvent) error {
|
||||||
t.EventCalls["OnRealtimeBeforeSubscribeRequest"]++
|
t.EventCalls["OnRealtimeBeforeSubscribeRequest"]++
|
||||||
return nil
|
return nil
|
||||||
|
Loading…
x
Reference in New Issue
Block a user