Skip to content
项目
群组
代码片段
帮助
当前项目
正在载入...
登录 / 注册
切换导航面板
G
go-ipfs
概览
概览
详情
活动
周期分析
版本库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
统计图
问题
0
议题
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
CI / CD
CI / CD
流水线
作业
日程表
图表
维基
Wiki
代码片段
代码片段
成员
成员
折叠边栏
关闭边栏
活动
图像
聊天
创建新问题
作业
提交
问题看板
Open sidebar
jihao
go-ipfs
Commits
fd75b646
提交
fd75b646
authored
7月 25, 2015
作者:
Jeromy
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
refactor http client code
License: MIT Signed-off-by:
Jeromy
<
jeromyj@gmail.com
>
上级
2b06ffaa
隐藏空白字符变更
内嵌
并排
正在显示
1 个修改的文件
包含
58 行增加
和
61 行删除
+58
-61
client.go
commands/http/client.go
+58
-61
没有找到文件。
commands/http/client.go
浏览文件 @
fd75b646
...
...
@@ -184,55 +184,16 @@ func getResponse(httpRes *http.Response, req cmds.Request) (cmds.Response, error
res
.
SetCloser
(
httpRes
.
Body
)
if
len
(
httpRes
.
Header
.
Get
(
streamHeader
))
>
0
&&
contentType
!=
"application/json"
{
//
if output is a stream, we can just use the body reader
if
contentType
!=
"application/json"
{
//
for all non json output types, just stream back the output
res
.
SetOutput
(
&
httpResponseReader
{
httpRes
})
return
res
,
nil
}
else
if
len
(
httpRes
.
Header
.
Get
(
channelHeader
))
>
0
{
// if output is coming from a channel, decode each chunk
outChan
:=
make
(
chan
interface
{})
go
func
()
{
dec
:=
json
.
NewDecoder
(
&
httpResponseReader
{
httpRes
})
outputType
:=
reflect
.
TypeOf
(
req
.
Command
()
.
Type
)
ctx
:=
req
.
Context
()
for
{
var
v
interface
{}
var
err
error
if
outputType
!=
nil
{
v
=
reflect
.
New
(
outputType
)
.
Interface
()
err
=
dec
.
Decode
(
v
)
}
else
{
err
=
dec
.
Decode
(
&
v
)
}
// since we are just looping reading on the response, the only way to
// know we are 'done' is for the consumer to close the response body.
// doing so doesnt throw an io.EOF, but we want to treat it like one.
if
err
!=
nil
&&
strings
.
Contains
(
err
.
Error
(),
"read on closed response body"
)
{
err
=
io
.
EOF
}
if
err
!=
nil
&&
err
!=
io
.
EOF
{
log
.
Error
(
err
)
return
}
select
{
case
<-
ctx
.
Done
()
:
close
(
outChan
)
return
default
:
}
if
err
==
io
.
EOF
{
close
(
outChan
)
return
}
outChan
<-
v
}
}()
go
readStreamedJson
(
req
,
httpRes
,
outChan
)
res
.
SetOutput
((
<-
chan
interface
{})(
outChan
))
return
res
,
nil
...
...
@@ -240,22 +201,24 @@ func getResponse(httpRes *http.Response, req cmds.Request) (cmds.Response, error
dec
:=
json
.
NewDecoder
(
&
httpResponseReader
{
httpRes
})
// If we ran into an error
if
httpRes
.
StatusCode
>=
http
.
StatusBadRequest
{
e
:=
cmds
.
Error
{}
if
httpRes
.
StatusCode
==
http
.
StatusNotFound
{
switch
{
case
httpRes
.
StatusCode
==
http
.
StatusNotFound
:
// handle 404s
e
.
Message
=
"Command not found."
e
.
Code
=
cmds
.
ErrClient
}
else
if
contentType
==
"text/plain"
{
case
contentType
==
"text/plain"
:
// handle non-marshalled errors
buf
:=
bytes
.
NewBuffer
(
nil
)
io
.
Copy
(
buf
,
httpRes
.
Body
)
e
.
Message
=
string
(
buf
.
Bytes
())
e
.
Code
=
cmds
.
ErrNormal
}
else
{
default
:
// handle marshalled errors
err
=
dec
.
Decode
(
&
e
)
if
err
!=
nil
{
...
...
@@ -265,25 +228,59 @@ func getResponse(httpRes *http.Response, req cmds.Request) (cmds.Response, error
res
.
SetError
(
e
,
e
.
Code
)
}
else
{
outputType
:=
reflect
.
TypeOf
(
req
.
Command
()
.
Type
)
var
v
interface
{}
if
outputType
!=
nil
{
v
=
reflect
.
New
(
outputType
)
.
Interface
()
err
=
dec
.
Decode
(
v
)
}
else
{
err
=
dec
.
Decode
(
&
v
)
}
if
err
!=
nil
&&
err
!=
io
.
EOF
{
return
nil
,
err
return
res
,
nil
}
outputType
:=
reflect
.
TypeOf
(
req
.
Command
()
.
Type
)
v
,
err
:=
decodeTypedVal
(
outputType
,
dec
)
if
err
!=
nil
&&
err
!=
io
.
EOF
{
return
nil
,
err
}
res
.
SetOutput
(
v
)
return
res
,
nil
}
func
readStreamedJson
(
req
cmds
.
Request
,
httpRes
*
http
.
Response
,
out
chan
<-
interface
{})
{
defer
close
(
out
)
dec
:=
json
.
NewDecoder
(
&
httpResponseReader
{
httpRes
})
outputType
:=
reflect
.
TypeOf
(
req
.
Command
()
.
Type
)
ctx
:=
req
.
Context
()
for
{
v
,
err
:=
decodeTypedVal
(
outputType
,
dec
)
if
err
!=
nil
{
// since we are just looping reading on the response, the only way to
// know we are 'done' is for the consumer to close the response body.
// doing so doesnt throw an io.EOF, but we want to treat it like one.
if
!
(
strings
.
Contains
(
err
.
Error
(),
"read on closed response body"
)
||
err
==
io
.
EOF
)
{
log
.
Error
(
err
)
}
return
}
if
v
!=
nil
{
res
.
SetOutput
(
v
)
select
{
case
<-
ctx
.
Done
()
:
return
case
out
<-
v
:
}
}
}
return
res
,
nil
func
decodeTypedVal
(
t
reflect
.
Type
,
dec
*
json
.
Decoder
)
(
interface
{},
error
)
{
var
v
interface
{}
var
err
error
if
t
!=
nil
{
v
=
reflect
.
New
(
t
)
.
Interface
()
err
=
dec
.
Decode
(
v
)
}
else
{
err
=
dec
.
Decode
(
&
v
)
}
return
v
,
err
}
type
httpResponseReader
struct
{
...
...
编写
预览
Markdown
格式
0%
重试
或
添加新文件
添加附件
取消
您添加了
0
人
到此讨论。请谨慎行事。
请先完成此评论的编辑!
取消
请
注册
或者
登录
后发表评论