Dicom 学习笔记-DICOM C-Get 消息服务

引言


  前篇介绍了 DICOM C-Find 消息服务,本文结合开源 DICOM 库 fo-dicom 详细介绍一下 C-Get 服务。

C-Get 消息服务


  C-Get 服务主要用于获取影像,用于一个 DIMSE-service-user 在同等的DIMSE-service-user 上查询复合 SOP 实例的属性满足查询条件给出的一组属性的复合 SOP 实例,并取回这些符合条件的复合 SOP 实例,同时在这个过程中将触发一个或多个 C-STORE 子操作过程,所有的操作(包含 C-STORE 子操作)均在同一个 association 连接中。
  有关C-Get 服务流程图,我通过不同的 C-Get SCP 的实现,出现了两种流程图,参考如下:

  • 每个 C-Store 子操作完成后都会有一个 C-Get 响应
C-Get workflow 1

-C-Get 响应只会在 C-Store 子操作全部完成后发送

C-Get workflow 2

C-Get SCU


  通过结合 DICOM 开源库 fo-dicom 来实现 C-Get SCU,部分代码如下:
需要引用命名空间【Dicom.Network】和【System.Threading】

using Dicom.Network;
using System.Threading;
var client = new DicomClient();

// 添加能够接收的抽象语法 abstract syntax,通过查看 DICOM 文件的 SOPClassUID 可获得,否则不会触发 C-Store 子操作
var pc = DicomPresentationContext.GetScpRolePresentationContext(DicomUID.CTImageStorage);
client.AdditionalPresentationContexts.Add(pc);

var counter = 0;
var locker = new object();
client.OnCStoreRequest = (DicomCStoreRequest request) =>
{
    lock (locker)
    {
        ++counter;
        Console.WriteLine(DateTime.Now.ToString() + " recived, count is " + counter);
    }
    // 可以通过 request.Dataset 获取到 DICOM 文件

    return new DicomCStoreResponse(request, DicomStatus.Success);
};

var get = new DicomCGetRequest({StudyInstanceUID});

var handle = new ManualResetEventSlim();
get.OnResponseReceived = (DicomCGetRequest requ, DicomCGetResponse response) =>
{
    if (response.Remaining == 0)
    {
        handle.Set();
    }
};
client.AddRequest(get);
client.Send({C-Get SCP IP}, {C-Get SCP Port}, false, {C-Get SCU AE Title}, {C-Get SCP AE Title});
handle.Wait();
  • StudyInstanceUID:检查唯一标识;
  • C-Get SCP IP:C-Get 服务端的 IP 地址或机器名;
  • C-Get SCP Port:C-Get 服务端的端口;
  • C-Get SCU AE Title:C-Get 客户端应用实体的名称;
  • C-Get SCP AE Title:C-Get 服务端应用实体的名称;

C-Get SCP


  在本文开头部分给出了两张流程图,这是通过不同 C-Get SCP 的实现进行抓包得到的,针对这块流程中不同的地方在 DICOM 标准中到底是如何定义的,可以在 DICOM 标准的第4部分【C.1.4 Service Definition】 中找到,具体的描述如下:

A C-GET service conveys the following semantics:

The SCU supplies Unique Key values to identify an entity at the level of the retrieval. The SCP generates C-STORE sub-oper- ations for the corresponding storage SOP Instances identified by the Unique Key values. These C-STORE sub-operations occur on the same Association as the C-GET service and the SCU/SCP roles will be reversed for the C-STORE.

The SCP may optionally generate responses to the C-GET with status equal to Pending during the processing of the C-STORE sub-operations. These C-GET responses indicate the number of Remaining C-STORE sub-operations and the number of C- STORE sub-operations returning the status of Success, Warning, and Failed.

When the number of Remaining C-STORE sub-operations reaches zero, the SCP generates a final response with a status equal to Success, Warning, Failed, or Refused. This response may indicate the number of C-STORE sub-operations returning the status of Success, Warning, and Failed. If the status of a C-STORE sub-operation was Failed a UID List will be returned.

The SCU may cancel the C-GET service by issuing a C-GET-CANCEL request at any time during the processing of the C- GET. The SCP terminates all incomplete C-STORE sub-operations and returns a status of Canceled.

这里已经说明了是可选的,更详细的 C-Get 过程在 DICOM 标准第7部分【9.1.3.2】有描述,不过针对这块没有描述,所以才出现如文章开头两种流程图。C-Get SCP 可以通过派生 DicomService 服务类来实现 Dicom 服务的基本框架,然后实现 IDicomServiceProvider 和 IDicomCGetProvider 接口来实现,部分代码可以参考这里,核心部分是实现 OnCGetRequest 方法。

C-Get 过程分析


  我这里选择本文开头第一种流程图的抓包数据进行分析,由于包的数据量比较大,我这里过滤掉不能被解码成 DICOM 协议的包,只分析能被解码成 DICOM 协议的包,先看前面部分:

C-Get pcap

红色框内的两行是两个 AE 建立 association 的过程:

  1. C-Get SCU(10.3.13.202)向 C-Get SCP(10.3.2.209) 发送 A-ASSOCIATE 请求;
  2. C-Get SCP(10.3.2.209)响应 C-Get SCU(10.3.13.202)的 A-ASSOCIATE 请求,然后两个 AE 就建立了一个 association;

接着蓝色框第一行就是 C-Get SCU(10.3.13.202)向 C-Get SCP(10.3.2.209) 发送 C-Get 请求了;
蓝色框第二行是 C-Get SCP(10.3.2.209)经过查找,找到满足条件的复合 SOP 实例对象,返回找到的信息,从下图红色框可以看到有3个子操作待执行,这里表示找到了3个满足条件的复合 SOP 实例对象,接下来会触发3个 C-Store 子操作。

C-Get Response

接着就是 C-Store 归档数据的操作了

C-Store

一个复合 SOP 实例对象数据归档的最后一个包是一个畸形数据包【Malformed packet】:

Malformed packet

从上图可以看出,前面传输的数据包都将在收到 Frame 609后进行重组,Frame 609就在 Malformed Packet 这个数据包,下面的蓝色框是这个数据包将前面所有收到的数据包重组之后,解析 SOP 对象得到的 TAG 值;
然后 C-Get SCU(10.3.13.202)向 C-Get SCP(10.3.2.209) 发送 C-Store 响应,并告知状态为 Success;

C-Store Response

接着 C-Get SCP(10.3.2.209)向 C-Get SCU(10.3.13.202)发送 C-Get 请求的响应,内容包括当前的状态、剩下多少个子操作待执行、已完成多少个子操作和失败的子操作个数:

C-Get Response

这个数据包结束后就又是下一个 SOP 实例对象的归档数据包了,和上面一样,这里不再继续分析。
当所有的 C-Store 子操作结束后 C-Get SCP(10.3.2.209)会向 C-Get SCU(10.3.13.202)发送一个 C-Get 请求的响应,这里会返回 C-Get 请求的状态,如果前面的子操作都成功的话,这里会返回状态为 Success:

C-Get Response

最后 C-Get SCU(10.3.13.202)向 C-Get SCP(10.3.2.209) 发送 A-RELEASE 请求断开 association;
C-Get SCP(10.3.2.209)响应 C-Get SCU(10.3.13.202)的 A-RELEASE 请求,然后断开两个 AE 之间的 association;
至此,整个 DICOM 协议相关的包就全部发送完毕。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,937评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,503评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,712评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,668评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,677评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,601评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,975评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,637评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,881评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,621评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,710评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,387评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,971评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,947评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,189评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,805评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,449评论 2 342

推荐阅读更多精彩内容

  • rljs by sennchi Timeline of History Part One The Cognitiv...
    sennchi阅读 7,281评论 0 10
  • 引言   前篇介绍了 DICOM C-Store 消息服务,本文结合开源 DICOM 库 fo-dicom 详细介...
    Statmoon阅读 7,140评论 0 4
  • 3721
    汝南江阅读 136评论 0 0
  • 南方的梅雨 似乎 比往年來的早一些 淅淅瀝瀝 濕漉漉的 古寺瓦棱上的青苔 如得到心愛禮物的童心 著實更艷 更青 而...
    瘋人阅读 509评论 2 7
  • 春来叶落惊新蕾, 染血踟蹰发几枝? 望尽花都南国路, 守回赤胆北山陂。 千芳映日藏螓首, 独木披风露峨眉。 皆道倾...
    烟雨雨巷阅读 444评论 1 3