Hyperledger Composer + loopback-datasource-juggler + Java:编码问题

时间:2018-03-06 12:29:23

标签: java loopback hyperledger-composer

我使用 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',请参阅hereignoreHttpErrors(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.jssource 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;。 我是否在错误的位置搜索导致错误的原因?

2 个答案:

答案 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();