Blob


1 package p9p
3 import (
4 "golang.org/x/net/context"
5 "io"
7 "net"
8 )
10 type client struct {
11 version string
12 msize int
13 ctx context.Context
14 transport roundTripper
15 }
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)
25 if err != nil {
26 return nil, err
27 }
29 return &client{
30 version: version,
31 msize: ch.MSize(),
32 ctx: ctx,
33 transport: newTransport(ctx, ch),
34 }, nil
35 }
37 var _ Session = &client{}
39 func (c *client) Version() (int, string) {
40 return c.msize, c.version
41 }
43 func (c *client) Auth(ctx context.Context, afid Fid, uname, aname string) (Qid, error) {
44 m := MessageTauth{
45 Afid: afid,
46 Uname: uname,
47 Aname: aname,
48 }
50 resp, err := c.transport.send(ctx, m)
51 if err != nil {
52 return Qid{}, nil
53 }
55 rauth, ok := resp.(MessageRauth)
56 if !ok {
57 return Qid{}, ErrUnexpectedMsg
58 }
60 return rauth.Qid, nil
61 }
63 func (c *client) Attach(ctx context.Context, fid, afid Fid, uname, aname string) (Qid, error) {
64 m := MessageTattach{
65 Fid: fid,
66 Afid: afid,
67 Uname: uname,
68 Aname: aname,
69 }
71 resp, err := c.transport.send(ctx, m)
72 if err != nil {
73 return Qid{}, err
74 }
76 rattach, ok := resp.(MessageRattach)
77 if !ok {
78 return Qid{}, ErrUnexpectedMsg
79 }
81 return rattach.Qid, nil
82 }
84 func (c *client) Clunk(ctx context.Context, fid Fid) error {
85 resp, err := c.transport.send(ctx, MessageTclunk{
86 Fid: fid,
87 })
88 if err != nil {
89 return err
90 }
92 _, ok := resp.(MessageRclunk)
93 if !ok {
94 return ErrUnexpectedMsg
95 }
97 return nil
98 }
100 func (c *client) Remove(ctx context.Context, fid Fid) error {
101 resp, err := c.transport.send(ctx, MessageTremove{
102 Fid: fid,
103 })
104 if err != nil {
105 return err
108 _, ok := resp.(MessageRremove)
109 if !ok {
110 return ErrUnexpectedMsg
113 return nil
116 func (c *client) Walk(ctx context.Context, fid Fid, newfid Fid, names ...string) ([]Qid, error) {
117 if len(names) > 16 {
118 return nil, ErrWalkLimit
121 resp, err := c.transport.send(ctx, MessageTwalk{
122 Fid: fid,
123 Newfid: newfid,
124 Wnames: names,
125 })
126 if err != nil {
127 return nil, err
130 rwalk, ok := resp.(MessageRwalk)
131 if !ok {
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{
140 Fid: fid,
141 Offset: uint64(offset),
142 Count: uint32(len(p)),
143 })
144 if err != nil {
145 return 0, err
148 rread, ok := resp.(MessageRread)
149 if !ok {
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)
164 offset := int64(0)
166 go func() {
167 for {
168 n, err := c.Read(ctx, fid, buf, offset)
169 if err != nil {
170 errChan <- err
171 break
173 if n == 0 {
174 break
176 writer.Write(buf[:n])
178 writer.Close()
179 }()
181 go func() {
182 dirs := []*Dir{}
183 dir := &Dir{}
184 for {
185 err := DecodeDir(NewCodec(), reader, dir)
186 if err != nil && err != io.EOF {
187 errChan <- err
188 break
190 dirs = append(dirs, dir)
191 if err != nil {
192 doneChan <- dirs
195 }()
197 select {
198 case err := <-errChan:
199 return nil, err
200 case dirs := <-doneChan:
201 return dirs, nil
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{
207 Fid: fid,
208 Offset: uint64(offset),
209 Data: p,
210 })
211 if err != nil {
212 return 0, err
215 rwrite, ok := resp.(MessageRwrite)
216 if !ok {
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{
225 Fid: fid,
226 Mode: mode,
227 })
228 if err != nil {
229 return Qid{}, 0, err
232 ropen, ok := resp.(MessageRopen)
233 if !ok {
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{
242 Fid: parent,
243 Name: name,
244 Perm: perm,
245 Mode: mode,
246 })
247 if err != nil {
248 return Qid{}, 0, err
251 rcreate, ok := resp.(MessageRcreate)
252 if !ok {
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})
261 if err != nil {
262 return Dir{}, err
265 rstat, ok := resp.(MessageRstat)
266 if !ok {
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{
275 Fid: fid,
276 Stat: dir,
277 })
278 if err != nil {
279 return err
282 _, ok := resp.(MessageRwstat)
283 if !ok {
284 return ErrUnexpectedMsg
287 return nil