-
Notifications
You must be signed in to change notification settings - Fork 6
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: instant job event notifications
This commit introduces a more efficient mechanism for notifications related to job events, such as creation, updating, and deletion. Unlike the previous architecture where clients had to constantly poll the server for updates, the new implementation uses server-sent events (SSE) to allow client to receive updates from the server , improving efficiency and reducing network overhead. Key Features: - Introduced a new SSE-enabled endpoint, `/jobs/events`, which accepts filter parameters to customize the stream of events received. - Clients can subscribe to `/jobs/events` for to receive updates on job events that meet specified filter criteria. - Provided a client reference implementation through `wfxctl`. Signed-off-by: Michael Adler <[email protected]>
- Loading branch information
1 parent
9dc0031
commit c7c68b7
Showing
63 changed files
with
3,297 additions
and
29 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,114 @@ | ||
package api | ||
|
||
/* | ||
* SPDX-FileCopyrightText: 2023 Siemens AG | ||
* | ||
* SPDX-License-Identifier: Apache-2.0 | ||
* | ||
* Author: Michael Adler <[email protected]> | ||
*/ | ||
|
||
import ( | ||
"context" | ||
"encoding/json" | ||
"io" | ||
"net/http" | ||
"os" | ||
"strings" | ||
"sync" | ||
"testing" | ||
"time" | ||
|
||
"github.com/rs/zerolog" | ||
"github.com/rs/zerolog/log" | ||
|
||
"github.com/siemens/wfx/generated/model" | ||
"github.com/siemens/wfx/internal/handler/job" | ||
"github.com/siemens/wfx/internal/handler/job/events" | ||
"github.com/siemens/wfx/internal/handler/job/status" | ||
"github.com/siemens/wfx/internal/handler/workflow" | ||
"github.com/siemens/wfx/workflow/dau" | ||
"github.com/steinfletcher/apitest" | ||
"github.com/stretchr/testify/assert" | ||
"github.com/stretchr/testify/require" | ||
) | ||
|
||
func TestJobEventsSubscribe(t *testing.T) { | ||
log.Logger = zerolog.New(zerolog.ConsoleWriter{Out: os.Stdout, TimeFormat: time.Stamp}) | ||
|
||
db := newInMemoryDB(t) | ||
wf := dau.DirectWorkflow() | ||
_, err := workflow.CreateWorkflow(context.Background(), db, wf) | ||
require.NoError(t, err) | ||
|
||
north, south := createNorthAndSouth(t, db) | ||
|
||
handlers := []http.Handler{north, south} | ||
for i, name := range allAPIs { | ||
handler := handlers[i] | ||
t.Run(name, func(t *testing.T) { | ||
var jobID string | ||
{ | ||
job, err := job.CreateJob(context.Background(), db, | ||
&model.JobRequest{ClientID: "foo", Workflow: wf.Name}) | ||
require.NoError(t, err) | ||
jobID = job.ID | ||
} | ||
require.NotEmpty(t, jobID) | ||
|
||
var wg sync.WaitGroup | ||
wg.Add(1) | ||
go func() { | ||
defer wg.Done() | ||
|
||
ch, _ := events.AddSubscriber(context.Background(), events.FilterParams{IDs: []string{jobID}}) | ||
// wait for event created by our status.Update | ||
<-ch | ||
// now our GET request should have received the response as well, | ||
// add some extra time to be safe | ||
time.Sleep(100 * time.Millisecond) | ||
events.ShutdownSubscribers() | ||
}() | ||
|
||
wg.Add(1) | ||
go func() { | ||
defer wg.Done() | ||
// wait for subscriber which is created by our GET request below and our test goroutine | ||
for events.SubscriberCount() != 2 { | ||
time.Sleep(50 * time.Millisecond) | ||
} | ||
// update job | ||
_, err = status.Update(context.Background(), db, jobID, &model.JobStatus{State: "INSTALLING"}, model.EligibleEnumCLIENT) | ||
require.NoError(t, err) | ||
}() | ||
|
||
result := apitest.New(). | ||
Handler(handler). | ||
Get("/api/wfx/v1/jobs/events").Query("ids", jobID). | ||
Expect(t). | ||
Status(http.StatusOK). | ||
Header("Content-Type", "text/event-stream"). | ||
End() | ||
|
||
data, _ := io.ReadAll(result.Response.Body) | ||
body := string(data) | ||
require.NotEmpty(t, body) | ||
|
||
// check body ends with two newlines as required by SSE spec | ||
assert.True(t, strings.HasSuffix(body, "\n\n")) | ||
|
||
// check body starts with data: | ||
assert.True(t, strings.HasPrefix(body, "data: ")) | ||
|
||
// check content is a job and state is INSTALLING | ||
var ev events.JobEvent | ||
err = json.Unmarshal([]byte(strings.TrimPrefix(body, "data: ")), &ev) | ||
require.NoError(t, err) | ||
assert.Equal(t, "INSTALLING", ev.Job.Status.State) | ||
assert.Equal(t, wf.Name, ev.Job.Workflow.Name) | ||
|
||
wg.Wait() | ||
events.ShutdownSubscribers() | ||
}) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.