@hayes/thrift-client v0.9.4-alpha.2
Thrift Client
Thrift client library for NodeJS written in TypeScript.
Supports communicating with Thrift services over HTTP or TCP.
Usage
We're going to go through this step-by-step.
- Install dependencies
- Define our service
- Run codegen on our Thrift IDL
- Create a client
- Make service calls with our client
- Add observability with Zipkin
Install
All Thrift Server libraries define inter-dependencies as peer dependencies to avoid type collisions.
$ npm install --save-dev @creditkarma/thrift-typescript
$ npm install --save @creditkarma/thrift-server-core
$ npm install --save @creditkarma/thrift-client
$ npm install --save request
$ npm install --save @types/request
Example Service
service Calculator {
i32 add(1: i32 left, 2: i32 right)
i32 subtract(1: i32 left, 2: i32 right)
}
Codegen
Requires @creditkarma/thrift-typescript >= v2.0.0
Add a script to your package.json to codegen. The 'target' option is important to make thrift-typescript generate for this library instead of the Apache libraries.
"scripts": {
"codegen": "thrift-typescript --target thrift-server --sourceDir thrift --outDir codegen
}
Then you can run codegen:
$ npm run codegen
Creating an HTTP Client
There are two ways to create HTTP clients with the public API.
- Use the
createHttpClient
factory function. - Manually create your own
HttpConnection
object
createHttpClient
Using the createHttpClient
function you pass in two arguments, the first is your Thrift service client class and the second is a map of options to configure the underlying HTTP connection.
When creating a client using this method Thrift Client uses the Request library for making HTTP requests.
import {
createHttpClient
} from '@creditkaram/thrift-client'
import { CoreOptions } from 'request'
import { Calculator } from './codegen/calculator'
// Create Thrift client
const thriftClient: Calculator.Client<CoreOptions> = createHttpClient(Calculator.Client, {
serviceName: 'calculator-service',
hostName: 'localhost',
port: 8080,
requestOptions: {} // CoreOptions to pass to Request
})
Options
The available options are:
- serviceName (optional): The name of your service. Used for logging.
- hostName (required): The name of the host to connect to.
- port (required): The port number to attach to on the host.
- path (optional): The path on which the Thrift service is listening. Defaults to '/thrift'.
- https (optional): Boolean value indicating whether to use https. Defaults to false.
- transport (optional): Name of the Thrift transport type to use. Defaults to 'buffered'.
- protocol (optional): Name of the Thrift protocol type to use. Defaults to 'binary'.
- requestOptions (optional): Options to pass to the underlying Request library. Defaults to {}.
- register (optional): A list of filters to apply to your client. More on this later.
Currently @creditkarma/thrift-server-core"
only supports buffered transport and binary or compact protocols.
type TransportType = 'buffered'
The possible protocol types are:
type ProtocolType = 'binary' | 'compact'
Manual Creation
Manually creating your Thrift client allows you to choose the use of another HTTP client library or to reuse a previously created instance of Request.
import {
RequestInstance,
HttpConnection,
IHttpConnectionOptions,
} from '@creditkaram/thrift-client'
import * as request from 'request'
import { CoreOptions } from 'request'
import { Calculator } from './codegen/calculator'
const clientConfig: IHttpConnectionOptions = {
hostName: 'localhost',
port: 3000,
path: '/',
transport: 'buffered',
protocol: 'binary',
}
// Create Thrift client
const requestClient: RequestInstance = request.defaults({})
const connection: HttpConnection =
new HttpConnection(requestClient, clientConfig)
const thriftClient: Calculator.Client<CoreOptions> = new Calculator.Client(connection)
Here HttpConnection
is a class that extends the ThriftConnection
abstract class. You could create custom connections, for instance TCP, by extending the same class.
Also of note here is that the type IHttpConnectionOptions
does not accept the requestOptions
parameter. Options to Request
here would be passed directly to the call to request.defaults({})
.
Creating a TCP Client
Creating a TCP client is much like creating an HTTP client.
- Use the
createTcpClient
factory function. - Manually create your own
TcpConnection
object
Note: Our TcpConnection object uses an underlying connection pool instead of all client requests reusing the same connection. This pool is configurable. We use GenericPool for managing our connection pool. Pool options are passed directly through to GenericPool.
createTcpClient
Using the createTcpClient
function you pass in two arguments, the first is your Thrift service client class and the second is a map of options to configure the underlying TCP connection.
import {
createTcpClient
} from '@creditkaram/thrift-client'
import { Calculator } from './codegen/calculator'
// Create Thrift client
const thriftClient: Calculator.Client<CoreOptions> = createTcpClient(Calculator.Client, {
serviceName: 'calculator-service',
hostName: 'localhost',
port: 8080,
})
Options
The available options are:
- serviceName (optional): The name of your service. Used for logging.
- hostName (required): The name of the host to connect to.
- port (required): The port number to attach to on the host.
- timeout (optional): Connection timeout in milliseconds. Defaults to 5000.
- transport (optional): Name of the Thrift transport type to use. Defaults to 'buffered'.
- protocol (optional): Name of the Thrift protocol type to use. Defaults to 'binary'.
- pool (optional): Options to configure the underlying connection pool.
- register (optional): A list of filters to apply to your client. More on this later.
Manual Creation
import {
TcpConnection,
} from '@creditkaram/thrift-client'
import { Calculator } from './codegen/calculator'
const connection: TcpConnection = new TcpConnection({
hostName: 'localhost',
port: 3000,
transport: 'buffered',
protocol: 'binary',
})
const thriftClient: Calculator.Client<CoreOptions> = new Calculator.Client(connection)
Here TcpConnection
is a class that extends the ThriftConnection
abstract class. You could create custom connections, for instance TCP, by extending the same class.
Making Service Calls with our Client
However we chose to make our client, we use them in the same way.
Notice the optional context parameter. All service client methods can take an optional context parameter. This context refers to the request options for Request library (CoreOptions). These options will be deep merged with any default options (passed in on instantiation) before sending a service request. This context can be used to do useful things like tracing or authentication. Usually this will be used for changing headers on a per-request basis.
Related to context you will notice that our Thrift service client is a generic Calculator.Client<ThriftContext<CoreOptions>>
. This type parameter refers to the type of the context, here the ThriftContext<CoreOptions>
which extends the options interface from the Request library.
import {
createHttpClient
} from '@creditkaram/thrift-client'
import { CoreOptions } from 'request'
import * as express from 'express'
import { Calculator } from './codegen/calculator'
const serverConfig = {
hostName: 'localhost',
port: 8080,
}
// Create Thrift client
const thriftClient: Calculator.Client<ThriftContext<CoreOptions>> = createHttpClient(Calculator.Client, {
hostName: 'localhost',
port: 8080,
requestOptions: {} // CoreOptions to pass to Request
})
// This receives a query like "http://localhost:8080/add?left=5&right=3"
app.get('/add', (req: express.Request, res: express.Response): void => {
// Request contexts allow you to do tracing and auth
const context: CoreOptions = {
headers: {
'x-trace-id': 'my-trace-id'
}
}
// Client methods return a Promise of the expected result
thriftClient.add(req.query.left, req.query.right, context).then((result: number) => {
res.send(result)
}, (err: any) => {
res.status(500).send(err)
})
})
app.listen(serverConfig.port, () => {
console.log(`Web server listening at http://${serverConfig.hostName}:${serverConfig.port}`)
})
Client Filters
Sometimes you'll want to universally filter or modify requests and/or responses. This is done with filters. If you've used many server libraries you are probably used to the idea of plugins or middleware.
A filter is an object that consists of a handler function and an optional list of client method names to apply the filter to.
Filters are applied in the order in which they are registered.
interface IRequestResponse {
statusCode: number
headers: IRequestHeaders
body: Buffer
}
type NextFunction<Options> =
(data?: Buffer, options?: Options) => Promise<IRequestResponse>
interface IThriftClientFilterConfig<Options> {
methods?: Array<string>
handler(request: IThriftRequest<Options>, next: NextFunction<Options>): Promise<IRequestResponse>
}
The handler
function receives as its arguments the IThriftRequest
object and the next RequestHandler
in the chain. When you are done applying your filter you return
the call to next
. When calling next
you can optionally pass along a modified Thrift data Buffer
or new options to apply to the request. Options
is almost always going to be CoreOptions
for the underlying request library.
IThriftRequest
The IThriftRequest
object wraps the outgoing data with some useful metadata about the current request.
export interface IThriftRequest<Context> {
data: Buffer
traceId?: TraceId
methodName: string
uri: string
context: Context
}
The data
attribute is the outgoing Thrift payload. The methodName
is the name of the Thrift service method being called.
The other interesting one is context
. The context is the data passed through in the service method call.
If we look back at our previous client example we find this:
// This receives a query like "http://localhost:8080/add?left=5&right=3"
app.get('/add', (req: express.Request, res: express.Response): void => {
// Request contexts allow you to do tracing and auth
const context: CoreOptions = {
headers: {
'x-trace-id': 'my-trace-id'
}
}
// Client methods return a Promise of the expected result
thriftClient.add(req.query.left, req.query.right, context).then((result: number) => {
res.send(result)
}, (err: any) => {
res.status(500).send(err)
})
})
Where the context passed through is an instance of CoreOptions
. These options are passed through to modify the outgoing request. Any filter in the filter chain can modify this data.
To modify this context one needs to pass the updated context to the next
function.
return next(request.data, updatedContext)
Applying Filters to Outgoing Requests
Something you may want to do with middlware is to apply some common HTTP headers to every outgoing request from your service. Maybe there is a token your service should attach to every outgoing request.
You could do something like this:
import {
createHttpClient,
IThriftRequest,
} from '@creditkaram/thrift-client'
import { Calculator } from './codegen/calculator'
const thriftClient: Calculator.Client = createHttpClient(Calculator.Client, {
hostName: 'localhost',
port: 8080,
register: [ {
handler(request: IThriftRequest<CoreOptions>, next: NextFunction<CoreOptions>): Promise<IRequestResponse> {
return next(request.data, {
headers: {
'x-fake-token': 'fake-token',
},
})
},
} ]
})
This sends data along unaltered, but adds a header x-fake-token
to the outgoing request. When you send along options, the options are deep merged with any previous options that were applied.
Applying Filters to Incoming Responses
To apply filters to the response you would call .then
on the next
function. This would allow you to inspect or modify the response before allowing it to proceed up the chain.
import {
createHttpClient,
IThriftRequest
} from '@creditkaram/thrift-client'
import { Calculator } from './codegen/calculator'
const thriftClient: Calculator.Client = createHttpClient(Calculator.Client, {
hostName: 'localhost',
port: 8080,
register: [ {
handler(request: IThriftRequest<CoreOptions>, next: NextFunction<CoreOptions>): Promise<IRequestResponse> {
return next().then((res: IRequestResponse) => {
if (validateResponse(res.body)) {
return res
} else {
throw new Error('Invalid data returned')
}
})
},
} ]
})
Adding Filters to HttpConnection Object
When you're not using createHttpClient
you can add filters directly to the connection instance.
// Create thrift client
const requestClient: RequestInstance = request.defaults({})
const connection: HttpConnection =
new HttpConnection(requestClient, clientConfig)
connection.register({
handler(request: IThriftRequest<CoreOptions>, next: NextFunction<CoreOptions>): Promise<IRequestResponse> {
return next(request.data, {
headers: {
'x-fake-token': 'fake-token',
},
})
},
})
const thriftClient: Calculator.Client = new Calculator.Client(connection)
The optional register
option takes an array of filters to apply. Unsurprisingly they are applied in the order you pass them in.
TCP Filters
When using Thrift over HTTP we can use HTTP headers to pass context/metadata between services (tracing, auth). When using TCP we don't have this. Among the options to solve this is to prepend an object onto the head of our TCP payload. @creditkarma/thrift-client
comes with two filters for helping with this situation.
ThriftContextFilter
This plugin writes a Thrift struct onto the head of an outgoing payload and reads a struct off of the head of an incoming payload.
import {
createTcpClient,
ThriftContextFilter,
} from '@creditkarma/thrift-client'
import {
RequestContext,
ResponseContext,
} from './codegen/metadata'
import {
Calculator,
} from './codegen/calculator'
const thriftClient: Calculator.Client<RequestContext> =
createTcpClient(Calculator.Client, {
hostName: 'localhost',
port: 8080,
register: [ ThriftContextFilter<RequestContext, ResponseContext>({
RequestContextClass: RequestContext,
}) ]
})
thriftClient.add(5, 6, new RequestContext({ traceId: 3827293 })).then((response: number) => {
// Do stuff...
})
Options
Available options for ThriftContextFilter:
- RequestContextClass (required): A class (extending StructLike) that is to be prepended to outgoing requests.
- ResponseContextClass (optional): A class (extending StructLike) that is prepended to incoming responses. Defaults to nothing.
- transportType (optional): The type of transport to use. Currently only 'buffered'.
- protocolType (optional): The type of protocol to use, either 'binary' or 'compact'.
TTwitterClientFilter
This plugin can be used in conjuction with Twitter's open source Finagle project to add and receive the headers that project expects.
import {
createTcpClient,
TTwitterClientFilter,
} from '@creditkarma/thrift-client'
import {
Calculator,
} from './codegen/calculator'
const thriftClient: Calculator.Client<RequestContext> =
createTcpClient(Calculator.Client, {
hostName: 'localhost',
port: 8080,
register: [ TTwitterClientFilter({
localServiceName: 'calculator-client',
remoteServiceName: 'calculator-service',
destHeader: 'calculator-service',
endpoint: 'http://localhost:9411/api/v1/spans',
sampleRate: 1,
}) ]
})
thriftClient.add(5, 6).then((response: number) => {
// Do stuff...
})
Note: The Twitter types are generated and exported under the name TTwitter
Options
Available options for TTwitterClientFilter:
- localServiceName (required): The name of your local service/application.
- remoteServiceName (required): The name of the remote service you are calling.
- destHeader (optional): A name for the destination added to the RequestHeader object Finagle expects. Defaults to the value of
remoteServiceName
. - isUpgraded (optional): Is the service using TTwitter context. Defaults to true.
- clientId (optional): A unique identifier for the client. Defaults to undefined.
- debug (optional): Zipkin debug mode. Defaults to false.
- endpoint (optional): Zipkin endpoint. Defaults to ''.
- sampleRate (optional): Zipkin samplerate. Defaults to 0.1.
- httpInterval (optional): Rate (in milliseconds) at which to send traces to Zipkin collector. Defaults to 1000.
- transportType (optional): The type of transport to use. Currently only 'buffered'.
- protocolType (optional): The type of protocol to use. Currently only 'binary'.
Observability
Distributed tracing is provided out-of-the-box with Zipkin. Distributed tracing allows you to track a request across multiple service calls to see where latency is in your system or to see where a particular request is failing. Also, just to get a complete picture of how many services a request of a particular kind touch.
Zipkin tracing is added to your Thrift client through filters.
import {
createHttpClient,
ZipkinClientFilter,
} from '@creditkaram/thrift-client'
import { Calculator } from './codegen/calculator'
const thriftClient: Calculator.Client<ThriftContext<CoreOptions>> =
createHttpClient(Calculator.Client, {
hostName: 'localhost',
port: 8080,
register: [ ZipkinClientFilter({
localServiceName: 'calculator-client',
remoteServiceName: 'calculator-service',
endpoint: 'http://localhost:9411/api/v1/spans',
sampleRate: 0.1,
}) ]
})
In order for tracing to be useful the services you are communicating with will also need to be setup with Zipkin tracing. Plugins are available for thrift-server-hapi
and thrift-server-express
. The provided plugins in Thrift Server only support HTTP transport at the moment.
Options
- localServiceName (required): The name of your service.
- remoteServiceName (optional): The name of the service you are calling.
- debug (optional): In debug mode all requests are sampled.
- endpoint (optional): URL of your collector (where to send traces).
- sampleRate (optional): Percentage (from 0 to 1) of requests to sample. Defaults to 0.1.
- httpInterval (optional): Sampling data is batched to reduce network load. This is the rate (in milliseconds) at which to empty the sample queue. Defaults to 1000.
If the endpoint is set then the plugin will send sampling data to the given endpoint over HTTP. If the endpoint is not set then sampling data will just be logged to the console.
Tracing Non-Thrift Endpoints
Sometimes, as part of completing your service request, you may need to gather data from both Thrift and non-Thrift endpoints. To get a complete picture you need to trace all of these calls. You can add Zipkin to other requests with instrumentation provided by the OpenZipkin project.
When constructing instrumentation provided by another library you need to use the same Tracer
in order to maintain the correct trace context. You can import this shared Tracer
through a call to getTracerForService
. This assumes a Tracer
has already been created for your service by usage of one of the Thrift Zipkin plugins.
import { getTracerForService } from '@creditkarma/thrift-server-core'
import * as wrapRequest from 'zipkin-instrumentation-request'
import * as request from 'request'
const tracer = getTracerForService('calculator-client')
const zipkinRequest = wrapRequest(request, { tracer, remoteServiceName: 'calculator-service' })
zipkinRequest.get(url, (err, resp, body) => {
// Do something
})
Running the Sample Application
Included in this repo is a sample application that uses thrift-client
and thrift-server-hapi
. To get the sample application up and running you need to do a few things.
First, clone the thrift-server
repo:
$ git clone https://github.com/creditkarma/thrift-server.git
Then, cd
into the thrift-server
directory and run npm install
and npm run build
.
$ cd thrift-server
$ npm install
$ npm run build
The thrift-server
project uses lerna to manage inter-library dependencies. The npm install
command will obviously install all your dependencies, but it will also perform a lerna bootstrap
that will set up sym-links between all the libraries within the mono-repo.
Now that everything is linked and built we can go to the thrift-client
package and start the example application:
$ cd packages/thrift-client
$ npm start
This will start a web server on localhost:8080. The sample app has a UI you can visit from a web browser.
Running Zipkin
The example app is configured to emit Zipkin traces. To view these traces run the Zipkin Docker image:
$ docker run -d -p 9411:9411 openzipkin/zipkin
Contributing
For more information about contributing new features and bug fixes, see our Contribution Guidelines. External contributors must sign Contributor License Agreement (CLA)
License
This project is licensed under Apache License Version 2.0