gurl is a unified command-line tool that supports both gRPC and HTTP/HTTPS requests.
gurl combines the functionality of grpcurl (for gRPC servers) and curl (for HTTP/HTTPS endpoints) into a single tool:
- gRPC mode (default): Interact with gRPC servers using JSON/text input
- HTTP mode (
--curl): Acts as a standard HTTP client likecurlfor REST APIs and web services
In standard mode, gurl is essentially grpcurl - a tool for interacting with gRPC servers. It's
basically curl for gRPC servers.
The main purpose is to invoke RPC methods on a gRPC server from the
command-line. gRPC servers use a binary encoding on the wire
(protocol buffers, or "protobufs"
for short). So they are basically impossible to interact with using regular curl
(and older versions of curl that do not support HTTP/2 are of course non-starters).
This program accepts messages using JSON encoding, which is much more friendly for both
humans and scripts.
With this tool you can also browse the schema for gRPC services, either by querying
a server that supports server reflection,
by reading proto source files, or by loading in compiled "protoset" files (files that contain
encoded file descriptor protos).
In fact, the way the tool transforms JSON request data into a binary encoded protobuf
is using that very same schema. So, if the server you interact with does not support
reflection, you will either need the proto source files that define the service or need
protoset files that gurl can use.
This repo also provides a library package, github.com/worthies/gurl, that has
functions for simplifying the construction of other command-line tools that dynamically
invoke gRPC endpoints. This code is a great example of how to use the various packages of
the protoreflect library, and shows
off what they can do.
See also the grpcurl talk at GopherCon 2018.
When --curl is enabled, gurl acts as a standard HTTP/HTTPS client, making it compatible
with regular web services and REST APIs. This mode supports common curl options and behavior.
gurl supports all kinds of RPC methods, including streaming methods. You can even
operate bi-directional streaming methods interactively by running gurl from an
interactive terminal and using stdin as the request body!
gurl supports both secure/TLS servers and plain-text servers (i.e. no TLS) and has
numerous options for TLS configuration. It also supports mutual TLS, where the client is
required to present a client certificate.
As mentioned above, gurl works seamlessly if the server supports the reflection
service. If not, you can supply the .proto source files or you can supply protoset
files (containing compiled descriptors, produced by protoc) to gurl.
- Standard HTTP methods: GET, POST, PUT, DELETE, etc.
- Custom headers with
-Hor--header - Request body data with
-dor--data - TLS support with certificate validation
- Client certificates for mutual TLS
- Unix domain socket support
- Verbose output with
-v - Output to file with
-o - And more curl-compatible options
Download the binary from the releases page.
On macOS, grpcurl is available via Homebrew:
brew install grpcurlFor platforms that support Docker, you can download an image that lets you run grpcurl:
# Download image
docker pull fullstorydev/grpcurl:latest
# Run the tool
docker run fullstorydev/grpcurl api.grpc.me:443 listNote that there are some pitfalls when using docker:
- If you need to interact with a server listening on the host's loopback network, you must specify the host as
host.docker.internalinstead oflocalhost(for Mac or Windows) OR have the container use the host network with-network="host"(Linux only). - If you need to provide proto source files or descriptor sets, you must mount the folder containing the files as a volume (
-v $(pwd):/protos) and adjust the import paths to container paths accordingly. - If you want to provide the request message via stdin, using the
-d @option, you need to use the-iflag on the docker command.
There are numerous other ways to install grpcurl, thanks to support from third parties that
have created recipes/packages for it. These include other ways to install grpcurl on a variety
of environments, including Windows and myriad Linux distributions.
You can see more details and the full list of other packages for grpcurl at repology.org:
https://repology.org/project/grpcurl/information
You can install grpcurl using the snap package:
snap install grpcurl
If you already have the Go SDK installed, you can build gurl from source:
# Clone this repository
git clone https://github.com/worthies/gURL.git
cd gURL
# Build the binary
go build -o gurl ./cmd/gurl
# Optionally install to $GOPATH/bin
go install ./cmd/gurlThis installs the command into the bin sub-folder of wherever your $GOPATH
environment variable points. (If you have no GOPATH environment variable set,
the default install location is $HOME/go/bin). If this directory is already in
your $PATH, then you should be good to go.
The usage doc for the tool explains the numerous options:
gurl -helpIn the sections below, you will find numerous examples demonstrating how to use
gurl in gRPC mode (default behavior).
Invoking an RPC on a trusted server (e.g. TLS without self-signed key or custom CA)
that requires no client certs and supports server reflection is the simplest thing to
do with gurl. This minimal invocation sends an empty request body:
gurl grpc.server.com:443 my.custom.server.Service/Method
# no TLS
gurl -plaintext grpc.server.com:80 my.custom.server.Service/MethodTo send a non-empty request, use the -d argument. Note that all arguments must come
before the server address and method name:
gurl -d '{"id": 1234, "tags": ["foo","bar"]}' \
grpc.server.com:443 my.custom.server.Service/MethodAs can be seen in the example, the supplied body must be in JSON format. The body will be parsed and then transmitted to the server in the protobuf binary format.
If you want to include gurl in a command pipeline, such as when using jq to
create a request body, you can use -d @, which tells gurl to read the actual
request body from stdin:
gurl -d @ grpc.server.com:443 my.custom.server.Service/Method <<EOM
{
"id": 1234,
"tags": [
"foor",
"bar"
]
}
EOMAdding of headers / metadata to a rpc request is possible via the -H name:value command line option. Multiple headers can be added in a similar fashion.
Example :
gurl -H header1:value1 -H header2:value2 -d '{"id": 1234, "tags": ["foo","bar"]}' grpc.server.com:443 my.custom.server.Service/MethodFor more usage guide, check out the help docs via gurl -help
gurl can operate in a cURL-compatible mode where common cURL-style
options are accepted and mapped to gurl's flags. Enable this mode with
--curl. The following mappings are supported:
-H, --header <name: value>→-H "name: value"(pass custom headers)-d, --data <data>→-d <data>(request body data)--data-raw, --data-binary, --data-ascii→-d <data>(all treated as request data)
-k, --insecure→-insecure(skip TLS verification)--cacert <file>→-cacert <file>(CA certificate)-E, --cert <cert[:password]>→-cert <cert>(client certificate)--key <file>→-key <file>(private key)
--connect-timeout <seconds>→-connect-timeout <seconds>(connection timeout)-m, --max-time <seconds>→-max-time <seconds>(maximum request time)--keepalive-time <seconds>→-keepalive-time <seconds>(keepalive interval)--unix-socket <path>→ sets-unixand uses the socket path as address--abstract-unix-socket <path>→ same as--unix-socket
-A, --user-agent <name>→-user-agent <name>(set User-Agent header)-u, --user <user:password>→ addsAuthorization: Basicheader-v, --verbose→-v(verbose output)-h, --help→-help(show help)-V, --version→-version(show version)
--url <url>→ extracts host:port from URL- URLs starting with
http://orhttps://→ automatically extracts host:port
Short flags can be combined: -kv is equivalent to -k -v (insecure + verbose)
The following cURL options are not applicable to gRPC and are silently ignored:
- HTTP-specific:
--compressed,--location,--cookie,--form,--get,--head - Output options:
--output,--remote-name,--dump-header - Proxy options:
--proxy,--noproxy,--proxy-user - Protocol versions:
--http1.0,--http1.1,--http2
Examples:
# Basic usage with headers and data
gurl --curl --header "authorization: Bearer TOKEN" \
--data '{"id":1}' grpc.server.com:443 my.Service/Method
# Using short flags (combined)
gurl --curl -kv -d '{"name":"test"}' localhost:50051 my.Service/Method
# Unix socket
gurl --curl --unix-socket /tmp/grpc.sock my.Service/Method
# With client certificates
gurl --curl --cert client.crt --key client.key \
--cacert ca.crt grpc.server.com:443 my.Service/Method
# URL-style addressing
gurl --curl -d '{"id":1}' https://api.example.com/my.Service/Method
# With timeout and keepalive
gurl --curl --connect-timeout 5 --max-time 30 \
--keepalive-time 10 server:443 my.Service/Method
```### Listing Services
To list all services exposed by a server, use the "list" verb. When using `.proto` source
or protoset files instead of server reflection, this lists all services defined in the
source or protoset files.
```shell
# Server supports reflection
gurl localhost:8787 list
# Using compiled protoset files
gurl -protoset my-protos.bin list
# Using proto sources
gurl -import-path ../protos -proto my-stuff.proto list
# Export proto files (use -proto-out-dir to specify the output directory)
gurl -plaintext -proto-out-dir "out_protos" "localhost:8787" describe my.custom.server.Service
# Export protoset file (use -protoset-out to specify the output file)
gurl -plaintext -protoset-out "out.protoset" "localhost:8787" describe my.custom.server.Service
The "list" verb also lets you see all methods in a particular service:
gurl localhost:8787 list my.custom.server.ServiceThe "describe" verb will print the type of any symbol that the server knows about or that is found in a given protoset file. It also prints a description of that symbol, in the form of snippets of proto source. It won't necessarily be the original source that defined the element, but it will be equivalent.
# Server supports reflection
gurl localhost:8787 describe my.custom.server.Service.MethodOne
# Using compiled protoset files
gurl -protoset my-protos.bin describe my.custom.server.Service.MethodOne
# Using proto sources
gurl -import-path ../protos -proto my-stuff.proto describe my.custom.server.Service.MethodOneThe gurl tool can operate on a variety of sources for descriptors. The descriptors
are required, in order for gurl to understand the RPC schema, translate inputs
into the protobuf binary format as well as translate responses from the binary format
into text. The sections below document the supported sources and what command-line flags
are needed to use them.
Without any additional command-line flags, gurl will try to use server reflection.
Examples for how to set up server reflection can be found here.
When using reflection, the server address (host:port or path to Unix socket) is required
even for "list" and "describe" operations, so that gurl can connect to the server
and ask it for its descriptors.
To use gurl on servers that do not support reflection, you can use .proto source
files.
In addition to using -proto flags to point gurl at the relevant proto source file(s),
you may also need to supply -import-path flags to tell gurl the folders from which
dependencies can be imported.
Just like when compiling with protoc, you do not need to provide an import path for the
location of the standard protos included with protoc (which contain various "well-known
types" with a package definition of google.protobuf). These files are "known" by gurl
as a snapshot of their descriptors is built into the gurl binary.
When using proto sources, you can omit the server address (host:port or path to Unix socket) when using the "list" and "describe" operations since they only need to consult the proto source files.
You can also use compiled protoset files with gurl. If you are scripting gurl and
need to re-use the same proto sources for many invocations, you will see better performance
by using protoset files (since it skips the parsing and compilation steps with each
invocation).
Protoset files contain binary encoded google.protobuf.FileDescriptorSet protos. To create
a protoset file, invoke protoc with the *.proto files that define the service:
protoc --proto_path=. \
--descriptor_set_out=myservice.protoset \
--include_imports \
my/custom/server/service.protoThe --descriptor_set_out argument is what tells protoc to produce a protoset,
and the --include_imports argument is necessary for the protoset to contain
everything that gurl needs to process and understand the schema.
When using protosets, you can omit the server address (host:port or path to Unix socket) when using the "list" and "describe" operations since they only need to consult the protoset files.