我使用 Hyperledger Composer 开发了业务网络定义,将其部署在 Hyperledger Fabric 示例链上(在 VirtualBox 上本地运行 - 使用Docker-containers安装Ubuntu)并使用composer-rest-server
(以及选项-c [cardname] -n always -w true
)启动Composer的REST服务器。然后我在浏览器中玩了一下,一切正常。
现在出现了问题:我想编写一个与此REST API交互的 Java 程序,并且API似乎无法解析我发送给Java的请求。我将Java生成的 JSON -string复制到浏览器,在那里运行它并且运行正常。但是,如果我以编程方式发送它,我会得到错误(见最后)。
这是我通过浏览器发送的内容
{"cId":"C_ID7","cDomain":"example.com"}
浏览器告诉我CURL应该看起来像这样
curl -X POST --header 'Content-Type: application/json' --header 'Accept: application/json' -d '{"cId":"C_ID7","cDomain":"example.com"}' 'http://localhost:3000/api/com.example.Company'
现在我想通过Java + JSOUP使用以下代码发送相同的消息:
Response resp = Jsoup.connect(baseURL + namespace + "Company").ignoreContentType(true).method(Method.POST)
.ignoreHttpErrors(true)
.requestBody("{\"cId\":\"C_ID7\",\"cDomain\":\"example.com\"}").execute();
相同(没有requestBody和Method.GET)适用于GET请求。 ignoreContentType(true)
是必要的,因为JSOUP不会处理' application / json',请参阅here。 ignoreHttpErrors(true)
是必要的,以解决Http 500错误,该错误隐藏了我在解决问题时所需要的真正错误。
所以最后我看到,当我执行上面的代码时,出现以下错误:
{"error":{"statusCode":500,"name":"Error","message":"Property names containing dot(s) are not supported. Model: com_example_Company, dynamic property: {\"cId\":\"C_ID7\",\"cDomain\":\"example.com\"}","stack":"Error: Property names containing dot(s) are not supported. Model: com_example_Company, dynamic property: {\"cId\":\"C_ID7\",\"cDomain\":\"example.com\"}\n at com_example_Company.ModelBaseClass._initProperties
(/home/[user]
]/.nvm/versions/node/v8.9.4/lib/node_modules/composer-rest-server/node_modules/loopback-datasource-juggler/lib/model.js:249:17)\n
at com_example_Company.ModelBaseClass (/home/[user]/.nvm/versions/node/v8.9.4/lib/node_modules/composer-rest-server/node_modules/loopback-datasource-juggler/lib/model.js:60:8)\n
at com_example_Company.Model (eval at createModelClassCtor
(/home/[user]/.nvm/versions/node/v8.9.4/lib/node_modules/composer-rest-server/node_modules/loopback-datasource-juggler/lib/model-builder.js:671:21), <anonymous>:12:24)\n
at com_example_Company.PersistedModel (eval at createModelClassCtor
(/home/[user]/.nvm/versions/node/v8.9.4/lib/node_modules/composer-rest-server/node_modules/loopback-datasource-juggler/lib/model-builder.js:671:21), <anonymous>:12:24)\n
at new com_example_Company (eval at createModelClassCtor
(/home/[user]/.nvm/versions/node/v8.9.4/lib/node_modules/composer-rest-server/node_modules/loopback-datasource-juggler/lib/model-builder.js:671:21), <anonymous>:12:24)\n
at Function.DataAccessObject.create (/home/[user]/.nvm/versions/node/v8.9.4/lib/node_modules/composer-rest-server/node_modules/loopback-datasource-juggler/lib/dao.js:359:13)\n
at SharedMethod.invoke
(/home/[user]/.nvm/versions/node/v8.9.4/lib/node_modules/composer-rest-server/node_modules/strong-remoting/lib/shared-method.js:270:25)\n
at HttpContext.invoke (/home/[user]/.nvm/versions/node/v8.9.4/lib/node_modules/composer-rest-server/node_modules/strong-remoting/lib/http-context.js:297:12)\n
at phaseInvoke (/home/[user]/.nvm/versions/node/v8.9.4/lib/node_modules/composer-rest-server/node_modules/strong-remoting/lib/remote-objects.js:676:9)\n at runHandler
(/home/[user]/.nvm/versions/node/v8.9.4/lib/node_modules/composer-rest-server/node_modules/strong-remoting/node_modules/loopback-phase/lib/phase.js:135:5)\n
at iterate (/home/[user]/.nvm/versions/node/v8.9.4/lib/node_modules/composer-rest-server/node_modules/strong-remoting/node_modules/loopback-phase/node_modules/async/lib/async.js:146:13)\n
at Object.async.eachSeries
(/home/[user]/.nvm/versions/node/v8.9.4/lib/node_modules/composer-rest-server/node_modules/strong-remoting/node_modules/loopback-phase/node_modules/async/lib/async.js:162:9)\n
at runHandlers (/home/[user]/.nvm/versions/node/v8.9.4/lib/node_modules/composer-rest-server/node_modules/strong-remoting/node_modules/loopback-phase/lib/phase.js:144:13)\n
at iterate
(/home/[user]/.nvm/versions/node/v8.9.4/lib/node_modules/composer-rest-server/node_modules/strong-remoting/node_modules/loopback-phase/node_modules/async/lib/async.js:146:13)\n
at /home/[user]/.nvm/versions/node/v8.9.4/lib/node_modules/composer-rest-server/node_modules/strong-remoting/node_modules/loopback-phase/node_modules/async/lib/async.js:157:25\n
at /home/[user]/.nvm/versions/node/v8.9.4/lib/node_modules/composer-rest-server/node_modules/strong-remoting/node_modules/loopback-phase/node_modules/async/lib/async.js:154:25"}}
显然[user]是我的用户名。所以我查看了 loopback-datasource-juggler 源代码,发现在model-builder.js
(source on Github)中,不是解析字符串的不同属性,而是给出了完整的字符串(&#34; {\&#34; cId \&#34;:\&#34; C_ID7 \&#34;,\&#34; cDomain \&#34;:\&#34;示例.COM \&#34)。在第269行中,构建器检查属性名称中是否有点字符。因为在我们的例子中属性名称是完整的JSON字符串,所以有点,程序失败。
不幸的是,此时似乎不支持XML。
现在:这怎么可能发生?我唯一的猜测是由Java完成的编码事情,因为相同的JSON在浏览器中有效。可能导致这种情况的任何想法?
还有一个小问题:代码中的错误消息是&#39;&#39;不支持包含点的属性名称。 &#39; +&#39;型号:%s,属性:%s&#39;&#39;,但在我收到的错误中说了一些关于&#34;动态属性&#34;。 我是否在错误的位置搜索导致错误的原因?
答案 0 :(得分:0)
curl示例是REST API(而不仅仅是浏览器)了解业务网络REST API端点的原因。
对于您的requestBody,您将需要完全限定的类(即具有前导命名空间的资产/参与者/事务类) - 例如:
curl -X POST --header 'Content-Type: application/json' --header 'Accept: application/json' -d '{ \
"$class": "org.example.biznet.SampleAsset", \
"assetId": "1", \
"value": "103300" \
}' 'http://localhost:3000/api/SampleAsset'
所以你的将是
Response resp = Jsoup.connect(baseURL + namespace + "Company").ignoreContentType(true).method(Method.POST)
.ignoreHttpErrors(true)
.requestBody("{\"\$class\": \"org.example.biznet.SampleAsset\", \"cId\":\"C_ID7\", \"cDomain\":\"example.com\"}").execute();
我相信你还需要至少一个其他的双引号(现在总数:14)和一个结束的大括号。
答案 1 :(得分:0)
由于您已经指定了'Content-Type: application/json'
,因此requestBody中不需要双引号。以下应该起作用:
Response resp = Jsoup.connect(baseURL + namespace + "Company").ignoreContentType(true).method(Method.POST)
.ignoreHttpErrors(true)
.requestBody({\"cId\":\"C_ID7\",\"cDomain\":\"example.com\"}).execute();