4 "golang.org/x/net/context"
14 transport roundTripper
17 // NewSession returns a session using the connection. The Context ctx provides
18 // a context for out of bad messages, such as flushes, that may be sent by the
19 // session. The session can effectively shutdown with this context.
20 func NewSession(ctx context.Context, conn net.Conn) (Session, error) {
21 ch := newChannel(conn, codec9p{}, DefaultMSize) // sets msize, effectively.
23 // negotiate the protocol version
24 version, err := clientnegotiate(ctx, ch, DefaultVersion)
33 transport: newTransport(ctx, ch),
37 var _ Session = &client{}
39 func (c *client) Version() (int, string) {
40 return c.msize, c.version
43 func (c *client) Auth(ctx context.Context, afid Fid, uname, aname string) (Qid, error) {
50 resp, err := c.transport.send(ctx, m)
55 rauth, ok := resp.(MessageRauth)
57 return Qid{}, ErrUnexpectedMsg
63 func (c *client) Attach(ctx context.Context, fid, afid Fid, uname, aname string) (Qid, error) {
71 resp, err := c.transport.send(ctx, m)
76 rattach, ok := resp.(MessageRattach)
78 return Qid{}, ErrUnexpectedMsg
81 return rattach.Qid, nil
84 func (c *client) Clunk(ctx context.Context, fid Fid) error {
85 resp, err := c.transport.send(ctx, MessageTclunk{
92 _, ok := resp.(MessageRclunk)
94 return ErrUnexpectedMsg
100 func (c *client) Remove(ctx context.Context, fid Fid) error {
101 resp, err := c.transport.send(ctx, MessageTremove{
108 _, ok := resp.(MessageRremove)
110 return ErrUnexpectedMsg
116 func (c *client) Walk(ctx context.Context, fid Fid, newfid Fid, names ...string) ([]Qid, error) {
118 return nil, ErrWalkLimit
121 resp, err := c.transport.send(ctx, MessageTwalk{
130 rwalk, ok := resp.(MessageRwalk)
132 return nil, ErrUnexpectedMsg
135 return rwalk.Qids, nil
138 func (c *client) Read(ctx context.Context, fid Fid, p []byte, offset int64) (n int, err error) {
139 resp, err := c.transport.send(ctx, MessageTread{
141 Offset: uint64(offset),
142 Count: uint32(len(p)),
148 rread, ok := resp.(MessageRread)
150 return 0, ErrUnexpectedMsg
153 return copy(p, rread.Data), nil
156 // Readdir implements Plan9 wire protocol specifics for reading Dir entries.
157 func (c *client) Readdir(ctx context.Context, fid Fid) ([]*Dir, error) {
159 errChan := make(chan error)
160 doneChan := make(chan []*Dir)
162 reader, writer := io.Pipe()
163 buf := make([]byte, 256)
168 n, err := c.Read(ctx, fid, buf, offset)
176 writer.Write(buf[:n])
185 err := DecodeDir(NewCodec(), reader, dir)
186 if err != nil && err != io.EOF {
190 dirs = append(dirs, dir)
198 case err := <-errChan:
200 case dirs := <-doneChan:
205 func (c *client) Write(ctx context.Context, fid Fid, p []byte, offset int64) (n int, err error) {
206 resp, err := c.transport.send(ctx, MessageTwrite{
208 Offset: uint64(offset),
215 rwrite, ok := resp.(MessageRwrite)
217 return 0, ErrUnexpectedMsg
220 return int(rwrite.Count), nil
223 func (c *client) Open(ctx context.Context, fid Fid, mode Flag) (Qid, uint32, error) {
224 resp, err := c.transport.send(ctx, MessageTopen{
232 ropen, ok := resp.(MessageRopen)
234 return Qid{}, 0, ErrUnexpectedMsg
237 return ropen.Qid, ropen.IOUnit, nil
240 func (c *client) Create(ctx context.Context, parent Fid, name string, perm uint32, mode Flag) (Qid, uint32, error) {
241 resp, err := c.transport.send(ctx, MessageTcreate{
251 rcreate, ok := resp.(MessageRcreate)
253 return Qid{}, 0, ErrUnexpectedMsg
256 return rcreate.Qid, rcreate.IOUnit, nil
259 func (c *client) Stat(ctx context.Context, fid Fid) (Dir, error) {
260 resp, err := c.transport.send(ctx, MessageTstat{Fid: fid})
265 rstat, ok := resp.(MessageRstat)
267 return Dir{}, ErrUnexpectedMsg
270 return rstat.Stat, nil
273 func (c *client) WStat(ctx context.Context, fid Fid, dir Dir) error {
274 resp, err := c.transport.send(ctx, MessageTwstat{
282 _, ok := resp.(MessageRwstat)
284 return ErrUnexpectedMsg