WARNING: This is not ready for use in production yet. Still a WIP with likely breaking changes. Use at your own risk!
Unofficial node API wrapper for Canny's API (https://canny.io/) with native typescript support.
This API is not intended to be used on the front-end, as it would expose your Canny API Key.
npm i canny-api-js
Instantiate a new canny object by passing your API key, then use it to access all available entities and methods.
Javascript Example
const CannyAPI = require("canny-api-js").default;
const canny = new CannyAPI({
    apiKey: <yourApiKeyHere>,
});
const votesResponse = await canny.votes.list();
console.log({ votesResponse });Typescript Example
import CannyAPI from 'canny-api-js';
const canny = new CannyAPI({
  apiKey: <yourApiKeyHere>,
});
const votesResponse = await canny.votes.list();
console.log({ votesResponse });These are the available entities inside canny
- boards- retrieve
- list
 
- changelogEntries- list
 
- comments- retrieve
- list
- create
- delete
 
- companies- delete
 
- opportunities- list
 
- posts- retrieve
- list
- listAll(non-native, makes successive API calls)
- create
- changePostStatus
- addTag
- removeTag
 
- statusChanges- list
 
- tags- retrieve
- list
- create
 
- users- list
- retrieve
- findOrCreate
- delete
 
- votes- retrieve
- list
- create
- delete
 
e.g canny.boards.list():
The method arguments try to mimic the arguments listed in the official Canny API docs, so check it when in doubt.
A work in progress documentation of all arguments is also available below.
Example implementation
const { boards } = await canny.boards.list();
console.log({ boards });
const board = await canny.boards.retrieve('my-board-id');
console.log({ board });Available Methods
no parameters
Returns a list of boards.
id: string
Returns the board with a certain id.
Example implementation
const { hasMore, entries } = await canny.changelogEntries.list({
  labelIDs: ["label-1", "label-2"]
  sort: "lastSaved",
});
console.log({ hasMore, entries });Available Methods
args?: {
  /** Fetch only entries with at least one of the labels in the array. */
  labelIDs?: string[];
  /** The number of entries you'd like to fetch. Defaults to 10 if not specified. */
  limit?: number;
  /** The number of entries you'd like to skip before starting to fetch. Defaults to 0 if not specified. */
  skip?: number;
  /** The order in which the entries should be fetched. Options include: "created", "lastSaved", "nonPublishedFirst", "publishedAt". Defaults to "nonPublishedFirst" if not specified. */
  sort?: CannyChangelogEntrySortOptions;
  /** The type of entries to fetch. Value can be "new", "improved", or "fixed". */
  type?: CannyChangelogEntryType;
}Returns a list of changelog entries.
Example implementation
const comment = await canny.comments.retrieve('my-comment-id');
console.log({ comment });
const { hasMore, comments } = await canny.comments.list({
  postID: 'a-certain-post-id',
  limit: 20,
});
console.log({ hasMore, comments });
await canny.comments.create({
  authorID: 'id-of-an-user',
  postID: 'id-of-a-post',
  value: 'Yay! I would love that feature!',
});
await canny.comments.delete('certain-comment-id');Available Methods
id: string
Returns the comment with a certain id.
args: {
  /** The id of the author you'd like to fetch comments for. */
  authorID?: string;
  /** The id of the board you'd like to fetch comments for. */
  boardID?: string;
  /** The number of comments you'd like to fetch. Defaults to 10 if not specified. */
  limit?: number;
  /** The id of the post you'd like to fetch comments for. */
  postID?: string;
  /** The number of comments you'd like to skip before starting to fetch. Defaults to 0 if not specified. */
  skip?: number;
}Returns a list of comments.
args: {
  /** The unique identifier of the comment's author. */
  authorID: string;
  /** The unique identifier of the comment's post. */
  postID: string;
  /** The comment value. */
  value: string;
  /** An array of the URLs of comment's images. */
  imageURLs?: string[];
  /** Whether this comment is only available for internal usage. Default is false. */
  internal?: boolean;
  /** The unique identifier of the comment's parent, if this comment is a reply. */
  parentID?: string;
  /** Whether this comment should be allowed to trigger email notifications. Default is false. */
  shouldNotifyVoters?: boolean;
}Create a new comment given an author's id, post id and a text value.
id: string
Delete a comment with a certain id.
(Documentation still a work in progress. Contributions are welcome!)
- 
Add better typechecking for webhooks. The objectattribute on webhook events has inconsistent typing compared to other API requests to the same resource (see Undocumented and Inconsistencies)
- 
Add more typechecking for subqueries that dont have all attributes (e.g Posts.authordoesn't have the useravatarURL, butUserhas)
- 
Ability to automagically list ALL paginated queries by calling multiple times 
- 
Built-in caching 
A list of current undocumented attributes and inconsistent behavior of Canny's API is written here.