Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
机器未来
Paddle
提交
fd893755
P
Paddle
项目概览
机器未来
/
Paddle
与 Fork 源项目一致
Fork自
PaddlePaddle / Paddle
通知
1
Star
1
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
1
列表
看板
标记
里程碑
合并请求
0
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
P
Paddle
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
1
Issue
1
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
提交
fd893755
编写于
6月 20, 2017
作者:
H
Helin Wang
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
Master save and load state from etcd
上级
f277726c
变更
6
隐藏空白更改
内联
并排
Showing
6 changed file
with
330 addition
and
48 deletion
+330
-48
go/cmd/master/master.go
go/cmd/master/master.go
+47
-8
go/master/client_internal_test.go
go/master/client_internal_test.go
+19
-2
go/master/client_test.go
go/master/client_test.go
+19
-2
go/master/etcd_store.go
go/master/etcd_store.go
+133
-0
go/master/service.go
go/master/service.go
+109
-33
go/pserver/cclient/cclient.go
go/pserver/cclient/cclient.go
+3
-3
未找到文件。
go/cmd/master/master.go
浏览文件 @
fd893755
...
@@ -5,41 +5,80 @@ import (
...
@@ -5,41 +5,80 @@ import (
"net/http"
"net/http"
"net/rpc"
"net/rpc"
"strconv"
"strconv"
"strings"
"sync"
"time"
"time"
"github.com/namsral/flag"
"github.com/namsral/flag"
log
"github.com/sirupsen/logrus"
"github.com/PaddlePaddle/Paddle/go/master"
"github.com/PaddlePaddle/Paddle/go/master"
)
)
type
inMemStore
struct
{
mu
sync
.
Mutex
buf
[]
byte
}
func
(
m
*
inMemStore
)
Save
(
b
[]
byte
)
error
{
m
.
mu
.
Lock
()
defer
m
.
mu
.
Unlock
()
m
.
buf
=
b
return
nil
}
func
(
m
*
inMemStore
)
Load
()
([]
byte
,
error
)
{
m
.
mu
.
Lock
()
defer
m
.
mu
.
Unlock
()
return
m
.
buf
,
nil
}
func
main
()
{
func
main
()
{
port
:=
flag
.
Int
(
"port"
,
8080
,
"port of the master server."
)
port
:=
flag
.
Int
(
"port"
,
8080
,
"port of the master server."
)
faultTolerance
:=
flag
.
Bool
(
"fault_tolerance"
,
false
,
"enable fault tolerance (requires etcd)."
)
ttlSec
:=
flag
.
Int
(
"ttl"
,
60
,
"etcd lease TTL in seconds."
)
endpoints
:=
flag
.
String
(
"endpoints"
,
""
,
"comma separated etcd endpoints. If empty, fault tolerance will not be enabled."
)
taskTimeoutDur
:=
flag
.
Duration
(
"task_timout_dur"
,
20
*
time
.
Minute
,
"task timout duration."
)
taskTimeoutDur
:=
flag
.
Duration
(
"task_timout_dur"
,
20
*
time
.
Minute
,
"task timout duration."
)
taskTimeoutMax
:=
flag
.
Int
(
"task_timeout_max"
,
3
,
"max timtout count for each task before it being declared failed task."
)
taskTimeoutMax
:=
flag
.
Int
(
"task_timeout_max"
,
3
,
"max timtout count for each task before it being declared failed task."
)
chunkPerTask
:=
flag
.
Int
(
"chunk_per_task"
,
10
,
"chunk per task."
)
chunkPerTask
:=
flag
.
Int
(
"chunk_per_task"
,
10
,
"chunk per task."
)
flag
.
Parse
()
flag
.
Parse
()
if
*
faultTolerance
{
if
*
endpoints
==
""
{
panic
(
"fault tolernance not implemented."
)
log
.
Warningln
(
"-endpoints not set, fault tolerance not be enabled."
)
}
var
store
master
.
Store
if
*
endpoints
!=
""
{
eps
:=
strings
.
Split
(
*
endpoints
,
","
)
var
err
error
store
,
err
=
master
.
NewEtcdStore
(
eps
,
master
.
DefaultLockPath
,
master
.
DefaultStatePath
,
*
ttlSec
)
if
err
!=
nil
{
log
.
Fatal
(
err
)
}
}
else
{
store
=
&
inMemStore
{}
}
s
,
err
:=
master
.
NewService
(
store
,
*
chunkPerTask
,
*
taskTimeoutDur
,
*
taskTimeoutMax
)
if
err
!=
nil
{
log
.
Fatal
(
err
)
}
}
s
:=
master
.
NewService
(
*
chunkPerTask
,
*
taskTimeoutDur
,
*
taskTimeoutMax
)
err
=
rpc
.
Register
(
s
)
err
:=
rpc
.
Register
(
s
)
if
err
!=
nil
{
if
err
!=
nil
{
panic
(
err
)
log
.
Fatal
(
err
)
}
}
rpc
.
HandleHTTP
()
rpc
.
HandleHTTP
()
l
,
err
:=
net
.
Listen
(
"tcp"
,
":"
+
strconv
.
Itoa
(
*
port
))
l
,
err
:=
net
.
Listen
(
"tcp"
,
":"
+
strconv
.
Itoa
(
*
port
))
if
err
!=
nil
{
if
err
!=
nil
{
panic
(
err
)
log
.
Fatal
(
err
)
}
}
err
=
http
.
Serve
(
l
,
nil
)
err
=
http
.
Serve
(
l
,
nil
)
if
err
!=
nil
{
if
err
!=
nil
{
panic
(
err
)
log
.
Fatal
(
err
)
}
}
}
}
go/master/client_internal_test.go
浏览文件 @
fd893755
...
@@ -32,6 +32,19 @@ func (a TestAddresser) Address() string {
...
@@ -32,6 +32,19 @@ func (a TestAddresser) Address() string {
return
string
(
a
)
return
string
(
a
)
}
}
type
myStore
struct
{
buf
[]
byte
}
func
(
m
*
myStore
)
Save
(
b
[]
byte
)
error
{
m
.
buf
=
b
return
nil
}
func
(
m
*
myStore
)
Load
()
([]
byte
,
error
)
{
return
m
.
buf
,
nil
}
func
TestGetFinishTask
(
t
*
testing
.
T
)
{
func
TestGetFinishTask
(
t
*
testing
.
T
)
{
const
path
=
"/tmp/master_client_test_0"
const
path
=
"/tmp/master_client_test_0"
...
@@ -47,9 +60,13 @@ func TestGetFinishTask(t *testing.T) {
...
@@ -47,9 +60,13 @@ func TestGetFinishTask(t *testing.T) {
}
}
go
func
(
l
net
.
Listener
)
{
go
func
(
l
net
.
Listener
)
{
s
:=
NewService
(
chunkPerTask
,
time
.
Second
,
1
)
s
,
err
:=
NewService
(
&
myStore
{},
chunkPerTask
,
time
.
Second
,
1
)
if
err
!=
nil
{
panic
(
err
)
}
server
:=
rpc
.
NewServer
()
server
:=
rpc
.
NewServer
()
err
:
=
server
.
Register
(
s
)
err
=
server
.
Register
(
s
)
if
err
!=
nil
{
if
err
!=
nil
{
panic
(
err
)
panic
(
err
)
}
}
...
...
go/master/client_test.go
浏览文件 @
fd893755
...
@@ -15,6 +15,19 @@ import (
...
@@ -15,6 +15,19 @@ import (
"github.com/PaddlePaddle/recordio"
"github.com/PaddlePaddle/recordio"
)
)
type
myStore
struct
{
buf
[]
byte
}
func
(
m
*
myStore
)
Save
(
b
[]
byte
)
error
{
m
.
buf
=
b
return
nil
}
func
(
m
*
myStore
)
Load
()
([]
byte
,
error
)
{
return
m
.
buf
,
nil
}
func
TestNextRecord
(
t
*
testing
.
T
)
{
func
TestNextRecord
(
t
*
testing
.
T
)
{
const
(
const
(
path
=
"/tmp/master_client_TestFull"
path
=
"/tmp/master_client_TestFull"
...
@@ -33,9 +46,13 @@ func TestNextRecord(t *testing.T) {
...
@@ -33,9 +46,13 @@ func TestNextRecord(t *testing.T) {
}
}
go
func
(
l
net
.
Listener
)
{
go
func
(
l
net
.
Listener
)
{
s
:=
master
.
NewService
(
10
,
time
.
Second
,
1
)
s
,
err
:=
master
.
NewService
(
&
myStore
{},
10
,
time
.
Second
,
1
)
if
err
!=
nil
{
panic
(
err
)
}
server
:=
rpc
.
NewServer
()
server
:=
rpc
.
NewServer
()
err
:
=
server
.
Register
(
s
)
err
=
server
.
Register
(
s
)
if
err
!=
nil
{
if
err
!=
nil
{
panic
(
err
)
panic
(
err
)
}
}
...
...
go/master/etcd_store.go
0 → 100644
浏览文件 @
fd893755
package
master
import
(
"context"
"sync"
"github.com/coreos/etcd/clientv3"
"github.com/coreos/etcd/clientv3/concurrency"
log
"github.com/sirupsen/logrus"
)
const
(
// DefaultLockPath is the default etcd master lock path.
DefaultLockPath
=
"/master/lock"
// DefaultStatePath is the default etcd key for master state.
DefaultStatePath
=
"/master/state"
)
// EtcdStore is the Store implementation backed by etcd.
type
EtcdStore
struct
{
lockPath
string
statePath
string
ttlSec
int
client
*
clientv3
.
Client
mu
sync
.
Mutex
lock
*
concurrency
.
Mutex
}
// NewEtcdStore creates a new EtcdStore.
func
NewEtcdStore
(
endpoints
[]
string
,
lockPath
,
statePath
string
,
ttlSec
int
)
(
*
EtcdStore
,
error
)
{
cli
,
err
:=
clientv3
.
New
(
clientv3
.
Config
{
Endpoints
:
endpoints
,
DialTimeout
:
dialTimeout
,
})
if
err
!=
nil
{
return
nil
,
err
}
sess
,
err
:=
concurrency
.
NewSession
(
cli
,
concurrency
.
WithTTL
(
ttlSec
))
if
err
!=
nil
{
return
nil
,
err
}
lock
:=
concurrency
.
NewMutex
(
sess
,
lockPath
)
// It's fine for the lock to get stuck, in this case we have
// multiple master servers running (only configured to have
// one master running, but split-brain problem may cuase
// multiple master servers running), and the cluster management
// software will kill one of them.
log
.
Infof
(
"Trying to acquire lock at %s."
,
lockPath
)
err
=
lock
.
Lock
(
context
.
TODO
())
if
err
!=
nil
{
return
nil
,
err
}
log
.
Infof
(
"Successfully acquired lock at %s."
,
lockPath
)
e
:=
&
EtcdStore
{}
e
.
client
=
cli
e
.
lock
=
lock
e
.
lockPath
=
lockPath
e
.
statePath
=
statePath
e
.
ttlSec
=
ttlSec
return
e
,
nil
}
// Save saves the state into the etcd.
func
(
e
*
EtcdStore
)
Save
(
state
[]
byte
)
error
{
e
.
mu
.
Lock
()
defer
e
.
mu
.
Unlock
()
ctx
:=
context
.
TODO
()
put
:=
clientv3
.
OpPut
(
e
.
statePath
,
string
(
state
))
resp
,
err
:=
e
.
client
.
Txn
(
ctx
)
.
If
(
e
.
lock
.
IsOwner
())
.
Then
(
put
)
.
Commit
()
if
err
!=
nil
{
return
err
}
if
!
resp
.
Succeeded
{
log
.
Errorln
(
"No longer owns the lock, trying to lock and save again."
)
sess
,
err
:=
concurrency
.
NewSession
(
e
.
client
,
concurrency
.
WithTTL
(
e
.
ttlSec
))
if
err
!=
nil
{
return
err
}
e
.
lock
=
concurrency
.
NewMutex
(
sess
,
e
.
lockPath
)
log
.
Infof
(
"Try to acquire lock at %s."
,
e
.
lockPath
)
err
=
e
.
lock
.
Lock
(
context
.
TODO
())
if
err
!=
nil
{
return
err
}
log
.
Infof
(
"Successfully acquired lock at %s."
,
e
.
lockPath
)
return
e
.
Save
(
state
)
}
return
nil
}
// Load loads the state from etcd.
func
(
e
*
EtcdStore
)
Load
()
([]
byte
,
error
)
{
e
.
mu
.
Lock
()
ctx
:=
context
.
TODO
()
get
:=
clientv3
.
OpGet
(
e
.
statePath
)
resp
,
err
:=
e
.
client
.
Txn
(
ctx
)
.
If
(
e
.
lock
.
IsOwner
())
.
Then
(
get
)
.
Commit
()
if
err
!=
nil
{
return
nil
,
err
}
if
!
resp
.
Succeeded
{
log
.
Errorln
(
"No longer owns the lock, trying to lock and load again."
)
sess
,
err
:=
concurrency
.
NewSession
(
e
.
client
)
if
err
!=
nil
{
return
nil
,
err
}
e
.
lock
=
concurrency
.
NewMutex
(
sess
,
e
.
lockPath
)
e
.
lock
.
Lock
(
context
.
TODO
())
e
.
mu
.
Unlock
()
return
e
.
Load
()
}
kvs
:=
resp
.
Responses
[
0
]
.
GetResponseRange
()
.
Kvs
if
len
(
kvs
)
==
0
{
// No state exists
e
.
mu
.
Unlock
()
return
nil
,
nil
}
state
:=
kvs
[
0
]
.
Value
e
.
mu
.
Unlock
()
return
state
,
nil
}
go/master/service.go
浏览文件 @
fd893755
package
master
package
master
import
(
import
(
"bytes"
"compress/gzip"
"encoding/gob"
"errors"
"errors"
"os"
"os"
"path/filepath"
"path/filepath"
...
@@ -12,24 +15,54 @@ import (
...
@@ -12,24 +15,54 @@ import (
"github.com/PaddlePaddle/recordio"
"github.com/PaddlePaddle/recordio"
)
)
const
(
dialTimeout
=
5
*
time
.
Second
)
// Store is the interface for save and load the master state.
type
Store
interface
{
Save
([]
byte
)
error
Load
()
([]
byte
,
error
)
}
// Chunk is a chunk of data consisted of several data instances.
type
Chunk
struct
{
Path
string
Index
recordio
.
Index
// chunk index
}
// Task is the basic unit of data instances assigned to trainers.
type
Task
struct
{
ID
int
Chunks
[]
Chunk
}
type
taskEntry
struct
{
Epoch
int
NumTimeout
int
Task
Task
}
type
taskQueues
struct
{
Todo
[]
taskEntry
Pending
map
[
int
]
taskEntry
// map from task ID to task entry
Done
[]
taskEntry
Failed
[]
Task
}
// Service is the master server service.
// Service is the master server service.
type
Service
struct
{
type
Service
struct
{
chunksPerTask
int
chunksPerTask
int
timeoutDur
time
.
Duration
timeoutDur
time
.
Duration
timeoutMax
int
timeoutMax
int
ready
chan
struct
{}
ready
chan
struct
{}
store
Store
mu
sync
.
Mutex
mu
sync
.
Mutex
initDone
bool
initDone
bool
taskQueues
taskQueues
taskQueues
taskQueues
}
}
// Recover recovers service state from etcd.
func
Recover
()
(
*
Service
,
error
)
{
// TODO(helin): recover from snapshot state from etcd.
return
nil
,
nil
}
func
partition
(
chunks
[]
Chunk
,
chunksPerTask
int
)
[]
taskEntry
{
func
partition
(
chunks
[]
Chunk
,
chunksPerTask
int
)
[]
taskEntry
{
id
:=
0
id
:=
0
if
chunksPerTask
<=
0
{
if
chunksPerTask
<=
0
{
...
@@ -58,7 +91,7 @@ func partition(chunks []Chunk, chunksPerTask int) []taskEntry {
...
@@ -58,7 +91,7 @@ func partition(chunks []Chunk, chunksPerTask int) []taskEntry {
}
}
// NewService creates a new service.
// NewService creates a new service.
func
NewService
(
chunksPerTask
int
,
timeoutDur
time
.
Duration
,
timeoutMax
int
)
*
Service
{
func
NewService
(
store
Store
,
chunksPerTask
int
,
timeoutDur
time
.
Duration
,
timeoutMax
int
)
(
*
Service
,
error
)
{
s
:=
&
Service
{}
s
:=
&
Service
{}
s
.
chunksPerTask
=
chunksPerTask
s
.
chunksPerTask
=
chunksPerTask
s
.
timeoutDur
=
timeoutDur
s
.
timeoutDur
=
timeoutDur
...
@@ -66,38 +99,81 @@ func NewService(chunksPerTask int, timeoutDur time.Duration, timeoutMax int) *Se
...
@@ -66,38 +99,81 @@ func NewService(chunksPerTask int, timeoutDur time.Duration, timeoutMax int) *Se
s
.
taskQueues
=
taskQueues
{}
s
.
taskQueues
=
taskQueues
{}
s
.
taskQueues
.
Pending
=
make
(
map
[
int
]
taskEntry
)
s
.
taskQueues
.
Pending
=
make
(
map
[
int
]
taskEntry
)
s
.
ready
=
make
(
chan
struct
{})
s
.
ready
=
make
(
chan
struct
{})
return
s
s
.
store
=
store
}
recovered
,
err
:=
s
.
recover
()
if
err
!=
nil
{
return
nil
,
err
}
// Chunk is a chunk of data consisted of several data instances.
if
recovered
{
type
Chunk
struct
{
// Recovered. Now the state is already initialized,
Path
string
// and the master is ready.
Index
recordio
.
Index
// chunk index
s
.
initDone
=
true
}
close
(
s
.
ready
)
}
// Task is the basic unit of data instances assigned to trainers.
return
s
,
nil
type
Task
struct
{
ID
int
Chunks
[]
Chunk
}
}
type
taskEntry
struct
{
// recover recovers service state from etcd.
Epoch
int
func
(
s
*
Service
)
recover
()
(
bool
,
error
)
{
NumTimeout
int
state
,
err
:=
s
.
store
.
Load
()
Task
Task
if
err
!=
nil
{
}
return
false
,
err
}
type
taskQueues
struct
{
if
state
==
nil
{
Todo
[]
taskEntry
log
.
Infoln
(
"No state exists, not recovered."
)
Pending
map
[
int
]
taskEntry
// map from task ID to task entry
return
false
,
nil
Done
[]
taskEntry
}
Failed
[]
Task
log
.
Infof
(
"Loaded snapshot of size: %d bytes."
,
len
(
state
))
gr
,
err
:=
gzip
.
NewReader
(
bytes
.
NewReader
(
state
))
if
err
!=
nil
{
return
false
,
err
}
dec
:=
gob
.
NewDecoder
(
gr
)
var
tqs
taskQueues
err
=
dec
.
Decode
(
&
tqs
)
if
err
!=
nil
{
return
false
,
err
}
err
=
gr
.
Close
()
if
err
!=
nil
{
// Only close failed, recover actually succeed, so
// just log error.
log
.
Errorln
(
err
)
}
s
.
taskQueues
=
tqs
return
true
,
nil
}
}
// *must* be called with s.mu being held.
//
snapshot
*must* be called with s.mu being held.
func
(
s
*
Service
)
snapshot
()
error
{
func
(
s
*
Service
)
snapshot
()
error
{
// TODO(helin): snapshot state on etcd.
// TOOD(helin): etcd request has a size limit, so the snapshot
return
nil
// size is limited by the max request size. We should either
// divide the snapshot into smaller chunks and save under
// different keys, or configure the request size to be big
// enough:
// https://github.com/coreos/etcd/blob/2f84f3d8d8ed8f9537ab6ffa44a3a1c7eddfa9b1/embed/config.go#L44
var
buf
bytes
.
Buffer
gw
:=
gzip
.
NewWriter
(
&
buf
)
enc
:=
gob
.
NewEncoder
(
gw
)
err
:=
enc
.
Encode
(
s
.
taskQueues
)
if
err
!=
nil
{
return
err
}
err
=
gw
.
Close
()
if
err
!=
nil
{
return
err
}
state
:=
buf
.
Bytes
()
log
.
Infof
(
"Saving snapshot of size: %d bytes."
,
len
(
state
))
return
s
.
store
.
Save
(
state
)
}
}
func
readChunks
(
globPaths
[]
string
)
([]
Chunk
,
error
)
{
func
readChunks
(
globPaths
[]
string
)
([]
Chunk
,
error
)
{
...
@@ -207,12 +283,12 @@ func (s *Service) checkTimeoutFunc(taskID int, epoch int) func() {
...
@@ -207,12 +283,12 @@ func (s *Service) checkTimeoutFunc(taskID int, epoch int) func() {
t
.
NumTimeout
++
t
.
NumTimeout
++
if
t
.
NumTimeout
>
s
.
timeoutMax
{
if
t
.
NumTimeout
>
s
.
timeoutMax
{
log
.
Warningf
(
"Task %v timed out %d times, discard.
\n
"
,
t
.
Task
,
t
.
NumTimeout
)
log
.
Warningf
(
"Task %v timed out %d times, discard."
,
t
.
Task
,
t
.
NumTimeout
)
s
.
taskQueues
.
Failed
=
append
(
s
.
taskQueues
.
Failed
,
t
.
Task
)
s
.
taskQueues
.
Failed
=
append
(
s
.
taskQueues
.
Failed
,
t
.
Task
)
return
return
}
}
log
.
Warningf
(
"Task %v timed out %d times, retry.
\n
"
,
t
.
Task
,
t
.
NumTimeout
)
log
.
Warningf
(
"Task %v timed out %d times, retry."
,
t
.
Task
,
t
.
NumTimeout
)
s
.
taskQueues
.
Todo
=
append
(
s
.
taskQueues
.
Todo
,
t
)
s
.
taskQueues
.
Todo
=
append
(
s
.
taskQueues
.
Todo
,
t
)
}
}
}
}
...
...
go/pserver/cclient/cclient.go
浏览文件 @
fd893755
...
@@ -133,7 +133,7 @@ func paddle_init_param(client C.paddle_pserver_client, param C.paddle_parameter,
...
@@ -133,7 +133,7 @@ func paddle_init_param(client C.paddle_pserver_client, param C.paddle_parameter,
if
err
!=
nil
{
if
err
!=
nil
{
if
err
.
Error
()
==
pserver
.
AlreadyInitialized
{
if
err
.
Error
()
==
pserver
.
AlreadyInitialized
{
log
.
Warningf
(
"parameter %s already initialized, treat paddle_init_param as sucessful.
\n
"
,
name
)
log
.
Warningf
(
"parameter %s already initialized, treat paddle_init_param as sucessful."
,
name
)
return
C
.
PSERVER_OK
return
C
.
PSERVER_OK
}
}
log
.
Errorln
(
err
)
log
.
Errorln
(
err
)
...
@@ -200,7 +200,7 @@ func paddle_get_params(client C.paddle_pserver_client, dst **C.paddle_parameter,
...
@@ -200,7 +200,7 @@ func paddle_get_params(client C.paddle_pserver_client, dst **C.paddle_parameter,
for
i
,
p
:=
range
ps
{
for
i
,
p
:=
range
ps
{
pn
[
i
]
=
p
.
Name
pn
[
i
]
=
p
.
Name
}
}
log
.
Errorf
(
"pserver returned wrong number of parameters. Requested: %s, returned: %s.
\n
"
,
strings
.
Join
(
pn
,
", "
),
strings
.
Join
(
ns
,
", "
))
log
.
Errorf
(
"pserver returned wrong number of parameters. Requested: %s, returned: %s."
,
strings
.
Join
(
pn
,
", "
),
strings
.
Join
(
ns
,
", "
))
return
C
.
PSERVER_ERROR
return
C
.
PSERVER_ERROR
}
}
...
@@ -210,7 +210,7 @@ func paddle_get_params(client C.paddle_pserver_client, dst **C.paddle_parameter,
...
@@ -210,7 +210,7 @@ func paddle_get_params(client C.paddle_pserver_client, dst **C.paddle_parameter,
for
i
,
p
:=
range
ps
{
for
i
,
p
:=
range
ps
{
pn
[
i
]
=
p
.
Name
pn
[
i
]
=
p
.
Name
}
}
log
.
Errorf
(
"pserver returned wrong parameters, or not in requested order. Requested: %s, returned: %s.
\n
"
,
strings
.
Join
(
pn
,
", "
),
strings
.
Join
(
ns
,
", "
))
log
.
Errorf
(
"pserver returned wrong parameters, or not in requested order. Requested: %s, returned: %s."
,
strings
.
Join
(
pn
,
", "
),
strings
.
Join
(
ns
,
", "
))
return
C
.
PSERVER_ERROR
return
C
.
PSERVER_ERROR
}
}
}
}
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录