Merge pull request #785 from brndnmtthws/master
Add more details on using macaroons with GRPC.
This commit is contained in:
commit
9479b085e2
@ -62,6 +62,11 @@ The following dependencies are required.
|
||||
<artifactId>netty-tcnative-boringssl-static</artifactId>
|
||||
<version>2.0.7.Final</version>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>commons-codec</groupId>
|
||||
<artifactId>commons-codec</artifactId>
|
||||
<version>1.11</version>
|
||||
</dependency>
|
||||
</dependencies>
|
||||
```
|
||||
In the build section, we'll need to configure the following things :
|
||||
@ -98,28 +103,79 @@ In the build section, we'll need to configure the following things :
|
||||
```
|
||||
#### Main.java
|
||||
```java
|
||||
import io.grpc.Attributes;
|
||||
import io.grpc.CallCredentials;
|
||||
import io.grpc.ManagedChannel;
|
||||
import io.grpc.Metadata;
|
||||
import io.grpc.MethodDescriptor;
|
||||
import io.grpc.Status;
|
||||
import io.grpc.netty.GrpcSslContexts;
|
||||
import io.grpc.netty.NettyChannelBuilder;
|
||||
import io.netty.handler.ssl.SslContext;
|
||||
import lnrpc.LightningGrpc;
|
||||
import lnrpc.LightningGrpc.LightningBlockingStub;
|
||||
import lnrpc.Rpc.*;
|
||||
import lnrpc.Rpc.GetInfoRequest;
|
||||
import lnrpc.Rpc.GetInfoResponse;
|
||||
import org.apache.commons.codec.binary.Hex;
|
||||
|
||||
import javax.net.ssl.SSLException;
|
||||
import java.io.File;
|
||||
import java.io.IOException;
|
||||
import java.nio.file.Files;
|
||||
import java.nio.file.Paths;
|
||||
import java.util.concurrent.Executor;
|
||||
|
||||
public class Main {
|
||||
static class MacaroonCallCredential implements CallCredentials {
|
||||
private final String macaroon;
|
||||
|
||||
MacaroonCallCredential(String macaroon) {
|
||||
this.macaroon = macaroon;
|
||||
}
|
||||
|
||||
public void thisUsesUnstableApi() {}
|
||||
|
||||
public void applyRequestMetadata(
|
||||
MethodDescriptor < ? , ? > methodDescriptor,
|
||||
Attributes attributes,
|
||||
Executor executor,
|
||||
final MetadataApplier metadataApplier
|
||||
) {
|
||||
String authority = attributes.get(ATTR_AUTHORITY);
|
||||
System.out.println(authority);
|
||||
executor.execute(new Runnable() {
|
||||
public void run() {
|
||||
try {
|
||||
Metadata headers = new Metadata();
|
||||
Metadata.Key < String > macaroonKey = Metadata.Key.of("macaroon", Metadata.ASCII_STRING_MARSHALLER);
|
||||
headers.put(macaroonKey, macaroon);
|
||||
metadataApplier.apply(headers);
|
||||
} catch (Throwable e) {
|
||||
metadataApplier.fail(Status.UNAUTHENTICATED.withCause(e));
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
private static final String CERT_PATH = "/Users/user/Library/Application Support/Lnd/tls.cert";
|
||||
private static final String MACAROON_PATH = "/Users/user/Library/Application Support/Lnd/admin.macaroon";
|
||||
private static final String HOST = "localhost";
|
||||
private static final int PORT = 10009;
|
||||
|
||||
public static void main(String... args) throws SSLException {
|
||||
public static void main(String...args) throws IOException {
|
||||
SslContext sslContext = GrpcSslContexts.forClient().trustManager(new File(CERT_PATH)).build();
|
||||
NettyChannelBuilder channelBuilder = NettyChannelBuilder.forAddress(HOST, PORT);
|
||||
ManagedChannel channel = channelBuilder.sslContext(sslContext).build();
|
||||
LightningBlockingStub stub = LightningGrpc.newBlockingStub(channel);
|
||||
|
||||
String macaroon =
|
||||
Hex.encodeHexString(
|
||||
Files.readAllBytes(Paths.get(MACAROON_PATH))
|
||||
);
|
||||
|
||||
LightningBlockingStub stub = LightningGrpc
|
||||
.newBlockingStub(channel)
|
||||
.withCallCredentials(new MacaroonCallCredential(macaroon));
|
||||
|
||||
|
||||
GetInfoResponse response = stub.getInfo(GetInfoRequest.getDefaultInstance());
|
||||
System.out.println(response.getIdentityPubkey());
|
||||
|
@ -97,6 +97,23 @@ timeout can be changed with the `--macaroontimeout` option; this can be
|
||||
increased for making RPC calls between systems whose clocks are more than 60s
|
||||
apart.
|
||||
|
||||
## Using Macaroons with GRPC clients
|
||||
|
||||
When interacting with `lnd` using the GRPC interface, the macaroons are encoded
|
||||
as a hex string over the wire and can be passed to `lnd` by specifying the
|
||||
hex-encoded macaroon as GRPC metadata:
|
||||
|
||||
GET https://localhost:8080/v1/getinfo
|
||||
Grpc-Metadata-macaroon: <macaroon>
|
||||
|
||||
Where `<macaroon>` is the hex encoded binary data from the macaroon file itself.
|
||||
|
||||
A very simple example using `curl` may look something like this:
|
||||
|
||||
curl --insecure --header "Grpc-Metadata-macaroon: $(xxd -ps -u -c 1000 $HOME/.lnd/admin.macaroon)" https://localhost:8080/v1/getinfo
|
||||
|
||||
Have a look at the [Java GRPC example](/docs/grpc/java.md) for programmatic usage details.
|
||||
|
||||
## Future improvements to the `lnd` macaroon implementation
|
||||
|
||||
The existing macaroon implementation in `lnd` and `lncli` lays the groundwork
|
||||
|
Loading…
Reference in New Issue
Block a user