Skip to content

Commit 81168c7

Browse files
authored
feat: Add strategy to post to Discord (#43)
1 parent 7078b9d commit 81168c7

File tree

6 files changed

+311
-5
lines changed

6 files changed

+311
-5
lines changed

README.md

+47-2
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@ The API is split into two parts:
2626
- `MastodonStrategy`
2727
- `TwitterStrategy`
2828
- `LinkedInStrategy`
29+
- `DiscordStrategy`
2930

3031
Each strategy requires its own parameters that are specific to the service. If you only want to post to a particular service, you can just directly use the strategy for that service.
3132

@@ -35,6 +36,8 @@ import {
3536
TwitterStrategy,
3637
MastodonStrategy,
3738
BlueskyStrategy,
39+
LinkedInStrategy,
40+
DiscordStrategy,
3841
} from "@humanwhocodes/crosspost";
3942

4043
// Note: Use an app password, not your login password!
@@ -63,9 +66,15 @@ const linkedin = new LinkedInStrategy({
6366
accessToken: "your-access-token",
6467
});
6568

66-
// create a client that will post to all three
69+
// Note: Bot token and channel ID required
70+
const discord = new DiscordStrategy({
71+
botToken: "your-bot-token",
72+
channelId: "your-channel-id",
73+
});
74+
75+
// create a client that will post to all services
6776
const client = new Client({
68-
strategies: [bluesky, mastodon, twitter, linkedin],
77+
strategies: [bluesky, mastodon, twitter, linkedin, discord],
6978
});
7079

7180
// post to all three
@@ -82,6 +91,7 @@ Usage: crosspost [options] ["Message to post."]
8291
--mastodon, -m Post to Mastodon.
8392
--bluesky, -b Post to Bluesky.
8493
--linkedin, -l Post to LinkedIn.
94+
--discord, -d Post to Discord.
8595
--file, -f The file to read the message from.
8696
--help, -h Show this message.
8797
```
@@ -124,6 +134,9 @@ Each strategy requires a set of environment variables in order to execute:
124134
- `BLUESKY_PASSWORD`
125135
- LinkedIn
126136
- `LINKEDIN_ACCESS_TOKEN`
137+
- Discord
138+
- `DISCORD_BOT_TOKEN`
139+
- `DISCORD_CHANNEL_ID`
127140

128141
Tip: You can also load environment variables from a `.env` file in the current working directory by setting the environment variable `CROSSPOST_DOTENV` to `1`.
129142

@@ -187,6 +200,38 @@ To enable posting to LinkedIn, follow these steps:
187200

188201
**Important:** Tokens automatically expire after two months.
189202

203+
### Discord
204+
205+
To enable posting to Discord, you'll need to create a bot and get its token:
206+
207+
1. Go to the [Discord Developer Portal](https://discord.com/developers/applications).
208+
2. Click "New Application".
209+
3. Give your application a name and click "Create".
210+
4. Click "Installation" in the left sidebar.
211+
5. Under "Install Link" select "None".
212+
6. Click "Save Changes".
213+
7. Click on "Bot" in the left sidebar.
214+
8. Uncheck "Public Bot" to ensure no one else can add this bot.
215+
9. Under "Text Permissions" check "Send Messages".
216+
10. Click "Save Changes".
217+
11. Click "Reset Token" and copy the bot token that appears.
218+
219+
To add the bot to your server:
220+
221+
1. In the Developer Portal, click on "OAuth2" in the left sidebar.
222+
2. Under "OAuth2 URL Generator", check "bot".
223+
3. Under "Bot Permissions", check "Send Messages" under "Text Permissions".
224+
4. Copy the generated URL and open it in your browser.
225+
5. Select your server and authorize the bot.
226+
227+
To get your channel ID:
228+
229+
1. Enable Developer Mode in Discord (User Settings > Advanced > Developer Mode).
230+
2. Right-click the channel you want to post to.
231+
3. Click "Copy Channel ID".
232+
233+
**Note:** By default your application will only be able to send messages to public channels. To send messages to private channels, you'll have to give your application the necessary permissions.
234+
190235
## License
191236

192237
Copyright 2024-2025 Nicholas C. Zakas

release-please-config.json

+2-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,8 @@
99
"type": "json",
1010
"path": "jsr.json",
1111
"jsonpath": "$.version"
12-
}
12+
},
13+
"src/strategies/discord.js"
1314
]
1415
}
1516
}

src/bin.js

+18-2
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ import {
1616
MastodonStrategy,
1717
BlueskyStrategy,
1818
LinkedInStrategy,
19+
DiscordStrategy,
1920
} from "./index.js";
2021
import fs from "node:fs";
2122

@@ -51,6 +52,7 @@ const options = {
5152
mastodon: { type: booleanType, short: "m" },
5253
bluesky: { type: booleanType, short: "b" },
5354
linkedin: { type: booleanType, short: "l" },
55+
discord: { type: booleanType, short: "d" },
5456
file: { type: stringType },
5557
help: { type: booleanType, short: "h" },
5658
};
@@ -63,13 +65,18 @@ const { values: flags, positionals } = parseArgs({
6365
if (
6466
flags.help ||
6567
(positionals.length === 0 && !flags.file) ||
66-
(!flags.twitter && !flags.mastodon && !flags.bluesky && !flags.linkedin)
68+
(!flags.twitter &&
69+
!flags.mastodon &&
70+
!flags.bluesky &&
71+
!flags.linkedin &&
72+
!flags.discord)
6773
) {
6874
console.log('Usage: crosspost [options] ["Message to post."]');
6975
console.log("--twitter, -t Post to Twitter.");
7076
console.log("--mastodon, -m Post to Mastodon.");
7177
console.log("--bluesky, -b Post to Bluesky.");
7278
console.log("--linkedin, -l Post to LinkedIn.");
79+
console.log("--discord, -d Post to Discord.");
7380
console.log("--file The file to read the message from.");
7481
console.log("--help, -h Show this message.");
7582
process.exit(1);
@@ -98,7 +105,7 @@ const env = new Env();
98105
// Determine which strategies to use
99106
//-----------------------------------------------------------------------------
100107

101-
/** @type {Array<TwitterStrategy|MastodonStrategy|BlueskyStrategy|LinkedInStrategy>} */
108+
/** @type {Array<TwitterStrategy|MastodonStrategy|BlueskyStrategy|LinkedInStrategy|DiscordStrategy>} */
102109
const strategies = [];
103110

104111
if (flags.twitter) {
@@ -139,6 +146,15 @@ if (flags.linkedin) {
139146
);
140147
}
141148

149+
if (flags.discord) {
150+
strategies.push(
151+
new DiscordStrategy({
152+
botToken: env.require("DISCORD_BOT_TOKEN"),
153+
channelId: env.require("DISCORD_CHANNEL_ID"),
154+
}),
155+
);
156+
}
157+
142158
//-----------------------------------------------------------------------------
143159
// Main
144160
//-----------------------------------------------------------------------------

src/index.js

+1
Original file line numberDiff line numberDiff line change
@@ -7,4 +7,5 @@ export * from "./strategies/bluesky.js";
77
export * from "./strategies/mastodon.js";
88
export * from "./strategies/twitter.js";
99
export * from "./strategies/linkedin.js";
10+
export * from "./strategies/discord.js";
1011
export * from "./client.js";

src/strategies/discord.js

+114
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,114 @@
1+
/**
2+
* @fileoverview Discord strategy for posting messages.
3+
* @author Nicholas C. Zakas
4+
*/
5+
6+
/* global fetch */
7+
8+
//-----------------------------------------------------------------------------
9+
// Type Definitions
10+
//-----------------------------------------------------------------------------
11+
12+
/**
13+
* @typedef {Object} DiscordOptions
14+
* @property {string} botToken The Discord bot token.
15+
* @property {string} channelId The Discord channel ID to post to.
16+
*/
17+
18+
/**
19+
* @typedef {Object} DiscordMessageResponse
20+
* @property {string} id The ID of the created message.
21+
* @property {string} channel_id The ID of the channel the message was posted to.
22+
* @property {string} content The content of the message.
23+
*/
24+
25+
/**
26+
* @typedef {Object} DiscordErrorResponse
27+
* @property {number} code The error code.
28+
* @property {string} message The error message.
29+
*/
30+
31+
//-----------------------------------------------------------------------------
32+
// Constants
33+
//-----------------------------------------------------------------------------
34+
35+
const API_BASE = "https://discord.com/api/v10";
36+
37+
//-----------------------------------------------------------------------------
38+
// Exports
39+
//-----------------------------------------------------------------------------
40+
41+
/**
42+
* A strategy for posting messages to Discord.
43+
*/
44+
export class DiscordStrategy {
45+
/**
46+
* The name of the strategy.
47+
* @type {string}
48+
* @readonly
49+
*/
50+
name = "discord";
51+
52+
/**
53+
* Options for this instance.
54+
* @type {DiscordOptions}
55+
*/
56+
#options;
57+
58+
/**
59+
* Creates a new instance.
60+
* @param {DiscordOptions} options Options for the instance.
61+
* @throws {Error} When options are missing.
62+
*/
63+
constructor(options) {
64+
const { botToken, channelId } = options;
65+
66+
if (!botToken) {
67+
throw new TypeError("Missing bot token.");
68+
}
69+
70+
if (!channelId) {
71+
throw new TypeError("Missing channel ID.");
72+
}
73+
74+
this.#options = options;
75+
}
76+
77+
/**
78+
* Posts a message to Discord.
79+
* @param {string} message The message to post.
80+
* @returns {Promise<DiscordMessageResponse>} A promise that resolves with the message data.
81+
* @throws {Error} When the message fails to post.
82+
*/
83+
async post(message) {
84+
if (!message) {
85+
throw new TypeError("Missing message to post.");
86+
}
87+
88+
const url = `${API_BASE}/channels/${this.#options.channelId}/messages`;
89+
const response = await fetch(url, {
90+
method: "POST",
91+
headers: {
92+
Authorization: `Bot ${this.#options.botToken}`,
93+
"Content-Type": "application/json",
94+
"User-Agent":
95+
"Crosspost CLI (https://github.com/humanwhocodes/crosspost, v0.5.0)", // x-release-please-version
96+
},
97+
body: JSON.stringify({
98+
content: message,
99+
}),
100+
});
101+
102+
if (!response.ok) {
103+
const errorResponse = /** @type {DiscordErrorResponse} */ (
104+
await response.json()
105+
);
106+
107+
throw new Error(
108+
`${response.status} Failed to post message: ${response.statusText}\n${errorResponse.message} (code: ${errorResponse.code})`,
109+
);
110+
}
111+
112+
return /** @type {Promise<DiscordMessageResponse>} */ (response.json());
113+
}
114+
}

0 commit comments

Comments
 (0)