DICOM UPS (WorkItems) Subscription 訂閱機制

Posted on Sat, Dec 2, 2023 DICOM

因為有隱匿高手把我的 Raccoon 的 UPS 轉換成 SQL 版本,但是實作的讓我說不出話,只好恢復一下自己實作 DICOM UPS 訂閱機制的記憶。不過想了想,UPS Subscription 實在是有點複雜,必須來記錄一下它的概念。

UPS 的操作

首先,我們來看看 UPS 有哪些操作:

不外乎,UPS 就跟一般世界的商業邏輯差不多,擁有 CRUD 的功能,但這個 Subscribe,到底在訂閱什麼?

UPS 的訂閱

節錄自: Subscribe Transaction

This transaction creates a Subscription to a Worklist or Workitem resource. It corresponds to the UPS DIMSE N-ACTION operation "Subscribe to Receive UPS Event Reports".
Once a Subscription has been created the user agent will receive notifications containing Event Reports for events associated with the Subscription's resource. To receive the notifications generated by Subscriptions, the user agent must have first opened a Notification Connection between itself and the origin server using the Open Notification Connection transaction; see Section 8.10.4.

我們主要看第二段,它說明了:當 Subscription 建立時,User 會收到 Subscription 的訊息,而要收到訊息的話,User 必須先建立與 server 的 Open Notification Connection transaction

那這個 Open Notification Connection transaction 又是個什麼鬼?

一點開上面的 Section 8.10.4,你就可以看到熟悉的字眼:The connection uses the WebSocket protocol. The connection can use the same TCP port as the HTTP connection, but they are separate connections.

哦!原來是 WebSocket

這樣一看,腦子就通了,這不就跟 yt 一樣,(1) 訂閱頻道,(2) 頻道有新影片時,傳送通知到帳號。轉換成 UPS 的概念就是 (1) 訂閱 UPS,(2) 進行相關操作,透過 WebSocket 把相關操作的 Subscription DICOM Json 傳送給 User

UPS 的訂閱種類

以下是 UPS 提供的 3 種訂閱種類,那筆者對於不同種類有不同的見解,若有誤,還請大家多多包涵以及指教。

看完後,我腦中所留下的實作內容大概是這樣:

我只能說,想那麼多幹嘛,看看 dcm4che,別人都實作好了,拿出來參考吧

訂閱觸發流程

其實上面說那麼多,這邊才是最主要的,不過也是個人紀錄當時的我,是如何實作 Subscription 的。

建立 WorkItem Subscription 觸發流程

我們可能對某個 aeTitle 建立 WorkItem 有興趣,就可以透過 Subscription 去知道某種 WorkItem 被建立了!

那這邊講一下流程:

sequenceDiagram
    User->>GlobalSubscriptions: Create Global Subscriptions
    GlobalSubscriptions->>Subscriptions: Find not subscirbed subscriptions with AE Title
    Subscriptions-->>GlobalSubscriptions: return hit subsciptions
    GlobalSubscriptions-->>User: Successful
sequenceDiagram
    Note left of User: global subscriptions already created
    User->>WorkItem: create Work Item
    WorkItem->>GlobalSubscriptions: found matcing global subscriptions
    GlobalSubscriptions->>Subscriptions: Create subscriptions for work item
    Subscriptions-->>GlobalSubscriptions: return
    GlobalSubscriptions-->>WorkItem: return
    WorkItem->>WorkItem: subscribed if any subscription exist
    WorkItem->>WorkItem: trigger event
    WorkItem-->>User: return