-
-
Notifications
You must be signed in to change notification settings - Fork 112
Getting Started
First, install the Cap'n Proto tools.
Then, install the Go compiler plugin by fetching the capnproto.org/go/capnp/v3
repo.
Note that the repository MUST be in $GOPATH/src
and not in $GOPATH/pkg
, the latter of which is the default target when using go get
with Go 1.13 and up.
To ensure this repository is stored at the correct path, there are two options:
GO111MODULE=off go get -u capnproto.org/go/capnp/v3/
- Run
git clone capnproto.org/go/capnp
within$GOPATH/src/capnproto.org/go
Cap'n Proto works by generating code: your schema can be converted to code for any language that Cap'n Proto supports.
Consider the following schema, stored in foo/books.capnp
:
using Go = import "/go.capnp";
@0x85d3acc39d94e0f8;
$Go.package("books");
$Go.import("foo/books");
struct Book {
title @0 :Text;
# Title of the book.
pageCount @1 :Int32;
# Number of pages in the book.
}
capnpc-go requires two annotations for all files: package
and import
. package
is needed to know what package to place at the head of the generated file and what identifier to use when referring to the type from another package. import
should be the fully qualified import path, and is used to generate the import statement from other packages and to detect when two types are in the same package. Compilation will fail unless these annotations are present.
To compile this schema into Go code, run the following command:
capnp compile -I$GOPATH/src/capnproto.org/go/capnp/std -ogo foo/books.capnp
This generates the file foo/books.capnp.go
.
The data structures contained in foo/books.capnp.go
are special Go structs that can be imported into your programs. Moreover, these can be written to byte-streams.
package main
import (
"os"
"foo/books"
"capnproto.org/go/capnp/v3"
)
func main() {
// Make a brand new empty message. A Message allocates Cap'n Proto structs.
msg, seg, err := capnp.NewMessage(capnp.SingleSegment(nil))
if err != nil {
panic(err)
}
// Create a new Book struct. Every message must have a root struct.
book, err := books.NewRootBook(seg)
if err != nil {
panic(err)
}
book.SetTitle("War and Peace")
book.SetPageCount(1440)
// Write the message to stdout.
err = capnp.NewEncoder(os.Stdout).Encode(msg)
if err != nil {
panic(err)
}
}
These datatypes can also be read from byte-streams, as follows:
package main
import (
"fmt"
"os"
"foo/books"
"capnproto.org/go/capnp/v3"
)
func main() {
// Read the message from stdin.
msg, err := capnp.NewDecoder(os.Stdin).Decode()
if err != nil {
panic(err)
}
// Extract the root struct from the message.
book, err := books.ReadRootBook(msg)
if err != nil {
panic(err)
}
// Access fields from the struct.
title, err := book.Title()
if err != nil {
panic(err)
}
pageCount := book.PageCount()
fmt.Printf("%q has %d pages\n", title, pageCount)
}
In addition, each type has a .Message()
method that returns a capnp.Message
, which can be directly marshaled into []byte
s using Message.Marshal
, and unmarshaled with a corresponding call to Message.Unmarshal
.
Lastly, packed encodings are supported via the following methods:
Serializing data structures is useful, but the real power of Cap'n Proto is the RPC Protocol. RPC (remote procedure call) is a form of network communication that allows you to make function calls between processes, often on different machines.
Just by providing a byte stream -- like a pipe or a network connection -- you can make method calls on objects in other processes.
The first step is to define an interface inside your schema:
using Go = import "/go.capnp";
@0xdb8274f9144abc7e;
$Go.package("hashes");
$Go.import("foo/hashes");
interface HashFactory {
newSha1 @0 () -> (hash :Hash);
}
interface Hash {
write @0 (data :Data) -> ();
sum @1 () -> (hash :Data);
}
Implementing the interface is straightforward:
package main
import (
"crypto/sha1"
"fmt"
"hash"
"io"
"foo/hashes"
"golang.org/x/net/context"
"capnproto.org/go/capnp/v3/rpc"
)
// hashFactory is a local implementation of HashFactory.
type hashFactory struct{}
func (hf hashFactory) NewSha1(_ context.Context, call hashes.HashFactory_newSha1) error {
// Create a new locally implemented Hash capability.
hs := hashes.Hash_ServerToClient(hashServer{sha1.New()})
// Notice that methods can return other interfaces.
return call.Results.SetHash(hs)
}
// hashServer is a local implementation of Hash.
type hashServer struct {
h hash.Hash
}
func (hs hashServer) Write(_ context.Context, call hashes.Hash_write) error {
data, err := call.Args().Data()
if err != nil {
return err
}
_, err = hs.h.Write(data)
return err
}
func (hs hashServer) Sum(call hashes.Hash_sum) error {
res, err := call.AllocResults()
if err != nil {
return err
}
b := hs.h.Sum(nil)
return res.SetHash(b)
}
func serveHash(ctx context.Context, rwc io.ReadWriteCloser) error {
// Create a new locally implemented HashFactory.
main := hashes.HashFactory_ServerToClient(hashFactory{})
// Listen for calls, using the HashFactory as the bootstrap interface.
conn := rpc.NewConn(rpc.NewStreamTransport(rwc), &rpc.Options{
BootstrapClient: main.Client,
})
defer conn.Close()
// Wait for connection to abort.
select {
case <-conn.Done():
return nil
case <-ctx.Done():
return conn.Close()
}
And then using it in a client:
func client(ctx context.Context, rwc io.ReadWriteCloser) error {
// Create a connection that we can use to get the HashFactory.
conn := rpc.NewConn(rpc.NewStreamTransport(rwc), nil) // nil sets default options
defer conn.Close()
// Get the "bootstrap" interface. This is the capability set with
// rpc.MainInterface on the remote side.
hf := hashes.HashFactory{Client: conn.Bootstrap(ctx)}
// Now we can call methods on hf, and they will be sent over c.
// The NewSha1 method does not have any parameters we can set, so we
// pass a nil function.
f, free := hf.NewSha1(ctx, nil)
defer free()
// 'NewSha1' returns a future, which allows us to pipeline calls to
// returned values before they are actually delivered. Here, we issue
// calls to an as-of-yet-unresolved Sha1 instance.
s := f.Hash()
// s refers to a remote Hash. Method calls are delivered in order.
f, free = s.Write(ctx, func(p hashes.Hash_write_Params) error {
return p.SetData([]byte("Hello, "))
})
defer free()
f, free = s.Write(ctx, func(p hashes.Hash_write_Params) error {
return p.SetData([]byte("World!"))
})
defer free()
// Get the sum, waiting for the result.
f, free = s.Sum(ctx, nil)
defer free()
result, err := f.Struct()
if err != nil {
return err
}
// Display the result.
sha1Val, err := result.Hash()
if err != nil {
return err
}
fmt.Printf("sha1: %x\n", sha1Val)
return nil
}
func main() {
ctx := context.Background()
c1, c2 := net.Pipe()
go serveHash(ctx, c1)
client(ctx, c2)
}
For the sake of simplicity, this example uses an in-memory pipe, but you can use TCP connections, Unix pipes, or any other type that implements io.ReadWriteCloser
.
The return type for a client call is a promise, not an immediate value.
It isn't until the Struct()
method is called on a method that the client
function blocks on the remote side.
This relies on a feature of the Cap'n Proto RPC protocol called promise pipelining.
The upshot is that this function only requires one network round trip to receive its results, even though there were multiple chained calls.
This is one of Cap'n Proto's key advantages.
For more details on writing schemas, see the Cap'n Proto language reference. The capnp package docs detail the encoding and client API, whereas the rpc package docs detail how to initiate an RPC connection.