Telescope π


A "babel for the Cosmos", Telescope is a TypeScript Transpiler for Cosmos Protobufs. Telescope is used to generate libraries for Cosmos blockchains. Simply point to your protobuffer files and create developer-friendly Typescript libraries for teams to build on your blockchain.
The following blockchain libraries (generated by Telescope) are available via npm
π₯ Checkout our video playlist to learn how to use telescope!
Table of contentsβ
- Telescope π
- Usage
Quickstartβ
Follow the instructions below to generate a new Typescript package that you can publish to npm.
First, install telescope:
npm install -g @osmonauts/telescope
Generateβ
Use the generate command to create a new package. 
telescope generate
cd ./your-new-project
yarn 
Add Protobufsβ
If you have .proto files, simply add them to a ./proto folder.
However, if you want to get started quickly using existing protos from our registry, simply use the install command.
telescope install
It's not necessary, but you may also specify specific packages, e.g.
telescope install @protobufs/osmosis
Transpileβ
To create the Typescript files, run the transpile command. 
telescope transpile
You should now see some .ts files generated in ./src. These are the real source files used in your application.
Buildβ
Finally, run install and build to generate the JS and types for publishing your module to npm.
yarn install
yarn build
Now you should have code inside of your ./src folder, ready for publishing via npm publish. Or, if you used the defaults, you can start developing and your code can be imported from ./src/codegen;
Usage
Programmatic Usageβ
First add telescope to your devDependencies:
yarn add --dev @osmonauts/telescope
Install helpers and cosmjs dependencies listed here
import { join } from 'path';
import telescope from '@osmonauts/telescope';
import { sync as rimraf } from 'rimraf';
const protoDirs = [join(__dirname, '/../proto')];
const outPath = join(__dirname, '../src/codegen');
rimraf(outPath);
telescope({
  protoDirs,
  outPath,
  // all options are totally optional ;)
  options: {
    aminoEncoding: {
        enabled: true
    },
    lcdClients: {
        enabled: false
    },
    rpcClients: {
        enabled: false,
        camelCase: true
    },
    // you can scope options to certain packages:
    packages: {
      nebula: {
        prototypes: {
          typingsFormat: {
            useExact: false
          }
        }
      },
      akash: {
        stargateClients: {
            enabled: true;
            includeCosmosDefaultTypes: false;
        },
        prototypes: {
          typingsFormat: {
              useExact: false
          }
        }
      }
    }
  }
}).then(()=>{
  console.log('β¨ all done!');
}).catch(e=>{
  console.error(e);
  process.exit(1);
})
Optionsβ
Amino Encodingβ
| option | description | defaults | 
|---|---|---|
| aminoEncoding.enabled | generate amino types and amino converters | true | 
| aminoEncoding.casingFn | set the amino-casing function for a project | snake() | 
| aminoEncoding.exceptions | set specific aminoType name exceptions | see code | 
| aminoEncoding.typeUrlToAmino | create functions for aminoType name exceptions | undefined | 
Prototypes Optionsβ
| option | description | defaults | 
|---|---|---|
| prototypes.enabled | enables the generation of proto encoding methods | true | 
| prototypes.includePackageVar | export a protoPackagevariable to indicate package name | false | 
| prototypes.excluded.packages | exclude a set of packages from transpilation | undefined | 
| prototypes.excluded.protos | exclude a set of proto files from transpilation | undefined | 
| prototypes.fieldDefaultIsOptional | boolean value representing default optionality of field | false | 
| prototypes.useOptionalNullable | use (gogoproto.nullable)values in determining optionality | true | 
| prototypes.allowUndefinedTypes | boolean value allowing Types to beundefined | false | 
| prototypes.optionalQueryParams | boolean value setting queryParams to be optional | false | 
| prototypes.optionalPageRequests | boolean value setting PageRequestfields to optional | false | 
Prototypes Methodsβ
| option | description | defaults | 
|---|---|---|
| prototypes.methods.encode | boolean to enable encodemethod on proto objects | true | 
| prototypes.methods.decode | boolean to enable decodemethod on proto objects | true | 
| prototypes.methods.fromJSON | boolean to enable fromJSONmethod on proto objects | true | 
| prototypes.methods.toJSON | boolean to enable toJSONmethod on proto objects | true | 
| prototypes.methods.fromPartial | boolean to enable fromPartialmethod on proto objects | true | 
| prototypes.methods.fromSDK | boolean to enable fromSDKmethod on proto objects | false | 
| prototypes.methods.toSDK | boolean to enable toSDKmethod on proto objects | false | 
LCD Client Optionsβ
| option | description | defaults | 
|---|---|---|
| lcdClients.enabled | generate LCD clients that can query proto Querymessages | true | 
| lcdClients.bundle | will generate factory bundle aggregate of all LCD Clients | true | 
| lcdClients.scoped | will generate factory of scoped LCD Clients | undefined | 
| lcdClients.scopedIsExclusive | will allow both scoped bundles and all RPC Clients | true | 
See LCD Clients for more info.
RPC Client Optionsβ
| option | description | defaults | 
|---|---|---|
| rpcClients.enabled | generate RPC clients that can interact with proto messages | true | 
| rpcClients.bundle | will generate factory bundle aggregate of all RPC Clients | true | 
| rpcClients.camelCase | use camel-case for RPC methods when generating RPC clients | true | 
| rpcClients.scoped | will generate factory of scoped RPC Clients | undefined | 
| rpcClients.scopedIsExclusive | will allow both scoped bundles and all RPC Clients | true | 
| rpcClients.enabledServices | which services to enable | [ Msg,Query,Service] | 
See RPC Clients for more info.
Stargate Client Optionsβ
| option | description | defaults | 
|---|---|---|
| stargateClients.includeCosmosDefaultTypes | if true, will include the cosmjs defaults with stargate clients | true(except cosmos package) | 
Typings and Formattingβ
| option | description | defaults | 
|---|---|---|
| prototypes.typingsFormat.useDeepPartial | defaults to true, but if disabled uses the PartialTS type | true | 
| prototypes.typingsFormat.useExact | defaults to false, but if enabled uses the ExactTS type | false | 
| prototypes.typingsFormat.timestamp | use either dateortimestampforTimestampproto type | "date" | 
| prototypes.typingsFormat.duration | use either durationorstringforDurationproto type | "duration" | 
Protobuf parserβ
| option | description | defaults | 
|---|---|---|
| prototypes.parser.keepCase | passes keepCaseto protobufparse()to keep original casing | true | 
| prototypes.parser.alternateCommentMode | passes alternateCommentModeto protobufparse()method | true | 
| prototypes.parser.preferTrailingComment | passes preferTrailingCommentto protobufparse()method | false | 
Typescript Disablingβ
| option | description | defaults | 
|---|---|---|
| tsDisable.disableAll | if true, will include //@ts-nocheckon every output file | false | 
| tsDisable.patterns | if set, will include //@ts-nocheckon matched patterns | [] | 
| tsDisable.files | if set, will include //@ts-nocheckon matched files | [] | 
ESLint Disablingβ
| option | description | defaults | 
|---|---|---|
| eslintDisable.disableAll | if true, will include /* eslint-disable */on every output file | false | 
| eslintDisable.patterns | if set, will include /* eslint-disable */on matched patterns | [] | 
| eslintDisable.files | if set, will include /* eslint-disable */on matched files | [] | 
Bundleβ
| option | description | defaults | 
|---|---|---|
| bundle.enabled | bundle all files into a scoped index file | true | 
Outputβ
| option | description | defaults | 
|---|---|---|
| removeUnusedImports | removes unused imports | true | 
| classesUseArrowFunctions | classes use arrow functions instead of bind()ing in constructors | false | 
Typesβ
Timestampβ
The representation of google.protobuf.Timestamp is configurable by the prototypes.typingsFormat.timestamp option.
| Protobuf type | Default/ date='date' | date='timestamp' | 
|---|---|---|
| google.protobuf.Timestamp | Date | { seconds: Long, nanos: number } | 
TODO
-  add date='string'option
Durationβ
The representation of google.protobuf.Duration is configurable by the prototypes.typingsFormat.duration option.
| Protobuf type | Default/ duration='duration' | duration='string' | |
|---|---|---|---|
| google.protobuf.Duration | { seconds: Long, nanos: number } | string | 
Composing Messagesβ
This example shows messages from the osmojs, which was built with Telescope.
Import the osmosis object from osmojs. In this case, we're show the messages available from the osmosis.gamm.v1beta1 module:
import { osmosis } from 'osmojs';
const {
    joinPool,
    exitPool,
    exitSwapExternAmountOut,
    exitSwapShareAmountIn,
    joinSwapExternAmountIn,
    joinSwapShareAmountOut,
    swapExactAmountIn,
    swapExactAmountOut
} = osmosis.gamm.v1beta1.MessageComposer.withTypeUrl;
Now you can construct messages. If you use vscode or another typescript-enabled IDE, you should also be able to use ctrl+space to see auto-completion of the fields required for the message.
import { coin } from '@cosmjs/amino';
const msg = swapExactAmountIn({
  sender,
  routes,
  tokenIn: coin(amount, denom),
  tokenOutMinAmount
});
Calculating Feesβ
Make sure to create a fee object in addition to your message.
import { coins } from '@cosmjs/amino';
const fee = {
    amount: coins(0, 'uosmo'),
    gas: '250000'
}
if you are broadcasting multiple messages in a batch, you should simulate your tx and estimate the fee
import { Dec, IntPretty } from '@keplr-wallet/unit';
const gasEstimated = await stargateClient.simulate(address, msgs, memo);
const fee = {
  amount: coins(0, 'uosmo'),
  gas: new IntPretty(new Dec(gasEstimated).mul(new Dec(1.3)))
    .maxDecimals(0)
    .locale(false)
    .toString()
};
Stargate Clientsβ
Every module gets their own signing client. This example demonstrates for the osmosis module.
Use getSigningOsmosisClient to get your SigningStargateClient, with the Osmosis proto/amino messages full-loaded. No need to manually add amino types, just require and initialize the client:
import { getSigningOsmosisClient } from 'osmojs';
const client = await getSigningOsmosisClient({
  rpcEndpoint,
  signer // OfflineSigner
});
Creating Signersβ
To broadcast messages, you'll want to use either keplr or an OfflineSigner from cosmjs using mnemonics.
Amino Signerβ
Likely you'll want to use the Amino, so unless you need proto, you should use this one:
import { getOfflineSigner as getOfflineSignerAmino } from 'cosmjs-utils';
Proto Signerβ
import { getOfflineSigner as getOfflineSignerProto } from 'cosmjs-utils';
WARNING: NOT RECOMMENDED TO USE PLAIN-TEXT MNEMONICS. Please take care of your security and use best practices such as AES encryption and/or methods from 12factor applications.
import { chains } from 'chain-registry';
const mnemonic =
  'unfold client turtle either pilot stock floor glow toward bullet car science';
  const chain = chains.find(({ chain_name }) => chain_name === 'osmosis');
  const signer = await getOfflineSigner({
    mnemonic,
    chain
  });
Broadcasting messagesβ
Now that you have your client, you can broadcast messages:
import { signAndBroadcast } from '@osmosnauts/helpers';
const res = await signAndBroadcast({
  client, // SigningStargateClient
  chainId: 'osmosis-1', // use 'osmo-test-5' for testnet
  address,
  msgs: [msg],
  fee,
  memo: ''
});
LCD Clientsβ
For querying data via REST endpoints, you can use LCD Clients. For a better developer experience, you can generate a factory of scoped bundles of all LCD Clients with the lcdClients option. 
const options: TelescopeOptions = {
    lcdClients: {
        enabled: true;
    }
};
If you use the lcdClients.scoped array, you can scope to only the modules of your interest.
const options: TelescopeOptions = {
  lcdClients: {
    enabled: true,
    scoped: [
      {
        dir: 'osmosis',
        filename: 'custom-lcd-client.ts',
        packages: [
          'cosmos.bank.v1beta1',
          'cosmos.gov.v1beta1',
          'osmosis.gamm.v1beta1'
        ],
        addToBundle: true,
        methodName: 'createCustomLCDClient'
      },
      {
        dir: 'evmos',
        filename: 'custom-lcd-client.ts',
        packages: [
          'cosmos.bank.v1beta1',
          'cosmos.gov.v1beta1',
          'evmos.erc20.v1'
        ],
        addToBundle: true,
        methodName: 'createEvmosLCDClient'
      }
    ]
  }
};
This will generate a nice helper in the ClientFactory, which you can then use to query multiple modules from a single object:
import { osmosis } from './codegen';
const main = async () => {
   const client = await osmosis.ClientFactory.createLCDClient({ restEndpoint: REST_ENDPOINT });
   // now you can query the modules
   const pool = await client.osmosis.gamm.v1beta1.pool({ poolId: "1" });
   const balance = await client.cosmos.bank.v1beta1.allBalances({ address: 'osmo1addresshere' });
};
LCD Clients Classesβ
If you want to instantiate a single client, for any module that has a Query type, there will be a LCDQueryClient object:
import { osmosis } from "osmojs";
export const main = async () => {
    const requestClient = new LCDClient({ restEndpoint: REST_ENDPOINT });
    const client = new osmosis.gamm.v1beta1.LCDQueryClient({ requestClient });
    const pools = await client.pools();
    console.log(pools);
};
main().then(() => {
    console.log('all done')
})
RPC Clientsβ
For querying data via RPC endpoints, you can use RPC Clients. For a better developer experience, you can generate a factory of scoped bundles of all RPC Clients with the rpcClients option. 
const options: TelescopeOptions = {
  rpcClients: {
    enabled: true
  }
};
If you use the rpcClients.scoped array, you can scope to only the modules of your interest.
const options: TelescopeOptions = {
  rpcClients: {
    enabled: true,
    camelCase: true,
    scoped: [
      {
        dir: 'osmosis',
        filename: 'osmosis-rpc-client.ts',
        packages: [
          'cosmos.bank.v1beta1',
          'cosmos.gov.v1beta1',
          'osmosis.gamm.v1beta1'
        ],
        addToBundle: true,
        methodNameQuery: 'createRPCQueryClient',
        methodNameTx: 'createRPCTxClient'
      }
    ]
  }
};
This will generate helpers createRPCQueryClient and createRPCTxClient in the ClientFactory, which you can then use to query multiple modules from a single object:
import { osmosis } from './proto';
const main = async () => {
  const client = await osmosis.ClientFactory.createRPCQueryClient({ rpcEndpoint });
  // now you can query the modules
  const pool = await client.osmosis.gamm.v1beta1.pool({ poolId: "1" });
  const balance = await client.cosmos.bank.v1beta1.allBalances({ address: 'osmo1addresshere' });
};
RPC Client Classesβ
If you want to instantiate a single client, you can generate RPC classes with the rpcClients option;
For any module that has a Msg, Query or Service type, a 
import { osmosis, cosmos } from 'osmojs';
const MsgClient = osmosis.gamm.v1beta1.MsgClientImpl;
const QueryClient = osmosis.gamm.v1beta1.QueryClientImpl;
const ServiceClient = cosmos.base.tendermint.v1beta1.ServiceClientImpl;
Here is an example of making a query if you want to use the RPC client classes manually:
import { osmosis } from "osmojs";
import { createProtobufRpcClient, QueryClient } from "@cosmjs/stargate";
import { Tendermint34Client } from "@cosmjs/tendermint-rpc";
export const main = async () => {
    const tmClient = await Tendermint34Client.connect(RPC_ENDPOINT);
    const QueryClientImpl = osmosis.gamm.v1beta1.QueryClientImpl;
    const client = new QueryClient(tmClient);
    const rpc = createProtobufRpcClient(client);
    const queryService = new QueryClientImpl(rpc);
    const pools = await queryService.pools({})
    console.log(pools);
};
main().then(() => {
    console.log('all done')
})
Manually registering typesβ
This example is with osmosis module in osmojs, but it is the same pattern for any module.
NOTE: this is using @cosmjs/stargate@0.28.4
import {
    AminoTypes,
    SigningStargateClient
} from '@cosmjs/stargate';
import { Registry } from '@cosmjs/proto-signing';
import { defaultRegistryTypes } from '@cosmjs/stargate';
import { OfflineSigner } from '@cosmjs/proto-signing'
import { osmosis } from 'osmojs';
export const getCustomSigningClient = async ({ rpcEndpoint, signer }: { rpcEndpoint: string, signer: OfflineSigner }) => {
  // registry
  const registry = new Registry(defaultRegistryTypes);
  // aminotypes
  const aminoTypes = new AminoTypes({
    ...osmosis.gamm.v1beta1.AminoConverter,
    ...osmosis.lockup.AminoConverter,
    ...osmosis.superfluid.AminoConverter
  });
  // load the 
  osmosis.gamm.v1beta1.load(registry);
  osmosis.lockup.load(registry);
  osmosis.superfluid.load(registry);
  const client = await SigningStargateClient.connectWithSigner(
    rpcEndpoint,
    signer,
    { registry, aminoTypes }
  );
  return client;
};
CosmWasmβ
Generate TypeScript SDKs for your CosmWasm smart contracts by using the cosmwasm option on TelescopeOptions. The cosmwasm option is actually a direct reference to the TSBuilderInput object, for the most up-to-date documentation, visit @cosmwasm/ts-codegen.
import { TSBuilderInput } from '@cosmwasm/ts-codegen';
const options: TelescopeOptions = {
  cosmwasm: { 
    contracts: [
      {
        name: 'SG721',
        dir: './path/to/sg721/schema'
      },
      {
        name: 'Minter',
        dir: './path/to/Minter/schema'
      }
    ],
    outPath: './path/to/code/src/'
  }
};
Dependenciesβ
If you don't use the boilerplate, you will need to manually install
- @cosmjs/amino
- @cosmjs/proto-signing
- @cosmjs/stargate
- @cosmjs/tendermint-rpc
yarn add @cosmjs/amino @cosmjs/proto-signing @cosmjs/stargate @cosmjs/tendermint-rpc
If you use the LCD Client generation, you'll need to add
- @osmonauts/lcd
yarn add @osmonauts/lcd
Troubleshootingβ
Create React Appβ
CRA requires that you update Webpack configurations:
Here is an example of a config-overrides.js:
https://github.com/pyramation/osmosis-ui/blob/main/config-overrides.js
Babelβ
This should not be an issue, but if you experience problems with syntax or are not using preset-env, you may need these babel plugins:
- babel-plugin-proposal-numeric-separator
- babel-plugin-proposal-optional-chaining
- babel-plugin-proposal-nullish-coalescing-operator
Developingβ
Initial setupβ
yarn 
yarn bootstrap
Buildingβ
yarn build
Testsβ
Then cd into a package and run the tests
cd ./packages/ast
yarn test:watch
Generatorsβ
See our plugin generators.
Sponsorsβ
Kudos to our sponsors:
- Osmosis funded the creation of Telescope.
Relatedβ
Checkout these related projects:
- @cosmwasm/ts-codegen for generated CosmWasm contract Typescript classes
Creditsβ
π Built by Cosmology βΒ if you like our tools, please consider delegating to our validator βοΈ
Thanks to these engineers, teams and projects for inspiring Telescope:
Disclaimerβ
AS DESCRIBED IN THE TELESCOPE LICENSES, THE SOFTWARE IS PROVIDED βAS ISβ, AT YOUR OWN RISK, AND WITHOUT WARRANTIES OF ANY KIND.
No developer or entity involved in creating Telescope will be liable for any claims or damages whatsoever associated with your use, inability to use, or your interaction with other users of the Telescope code or Telescope CLI, including any direct, indirect, incidental, special, exemplary, punitive or consequential damages, or loss of profits, cryptocurrencies, tokens, or anything else of value.