Skip to content

Commit 98edb3e

Browse files
authored
Merge pull request #763 from podverse/develop
Release v4.16.22
2 parents 5006229 + 1cbd6c8 commit 98edb3e

File tree

6 files changed

+113
-39
lines changed

6 files changed

+113
-39
lines changed

.example.env

+1-1
Original file line numberDiff line numberDiff line change
@@ -127,7 +127,7 @@ RATE_LIMITER_MAX_OVERRIDE=50
127127
PODPING_HIVE_ACCOUNT=
128128
PODPING_HIVE_POSTING_KEY=
129129

130-
FCM_GOOGLE_API_AUTH_TOKEN=
130+
FCM_GOOGLE_API_PATH_TO_AUTH_JSON=
131131

132132
MAINTENANCE_MODE_ENABLED= # lowercase true to enable
133133
MAINTENANCE_MODE_DOWNTIME_EXPECTED= # in minutes

package.json

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,8 @@
11
{
22
"name": "podverse-api",
3-
"version": "4.16.21",
3+
"version": "4.16.22",
44
"description": "Data API, database migration scripts, and backend services for all Podverse models.",
5+
"private": true,
56
"contributors": [
67
"Mitch Downey"
78
],
@@ -176,6 +177,7 @@
176177
"eslint-plugin-react": "7.x",
177178
"eslint-plugin-react-hooks": "1.x",
178179
"follow-redirects": "^1.15.6",
180+
"google-auth-library": "^9.14.1",
179181
"googleapis": "45.0.0",
180182
"http-errors": "1.7.3",
181183
"husky": "3.1.0",

src/config/index.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -69,7 +69,7 @@ export interface Config {
6969
rateLimiterMaxOverride: any
7070
manticore: any
7171
twitterAPIBearerToken: string
72-
fcmGoogleApiAuthToken: string
72+
fcmGoogleApiPathToAuthJson: string
7373
minimumMobileVersion: string
7474
podping: {
7575
hiveAccount: string
@@ -124,7 +124,7 @@ const podcastIndexBaseUrl = process.env.PODCAST_INDEX_BASE_URL || ''
124124
// default 1 hour (3600000 milliseconds)
125125
const podcastIndexRecentlyUpdatedSinceTime = process.env.PODCAST_INDEX_RECENTLY_UPDATED_SINCE_TIME || '3600000' // 1 hour
126126
const podcastIndexNewFeedsSinceTime = process.env.PODCAST_INDEX_NEW_FEEDS_SINCE_TIME || '43200000' // half a day
127-
const fcmGoogleApiAuthToken = process.env.FCM_GOOGLE_API_AUTH_TOKEN || ''
127+
const fcmGoogleApiPathToAuthJson = process.env.FCM_GOOGLE_PATH_TO_AUTH_JSON || ''
128128

129129
const bitpayConfig = {
130130
apiKeyPath: process.env.BITPAY_API_KEY_PATH || '/',
@@ -249,7 +249,7 @@ const config: Config = {
249249
},
250250
twitterAPIBearerToken,
251251
minimumMobileVersion,
252-
fcmGoogleApiAuthToken,
252+
fcmGoogleApiPathToAuthJson,
253253
podping,
254254
maintenanceMode: {
255255
isEnabled: process.env.MAINTENANCE_MODE_ENABLED === 'true' || false,

src/lib/notifications/fcmGoogleApi.ts

+27-34
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,15 @@
11
/* eslint-disable @typescript-eslint/camelcase */
22
import { request } from '../request'
3-
import { config } from '~/config'
43
import { getFCMTokensForPodcastId } from '~/controllers/fcmDevice'
4+
import { generateAccessToken } from './firebaseGenerateAccessToken'
55
import { SendNotificationOptions } from './sendNotificationOptions'
6-
const { fcmGoogleApiAuthToken } = config
6+
import { config } from '~/config';
7+
const fs = require('fs');
78

8-
const fcmGoogleApiPath = 'https://fcm.googleapis.com/fcm/send'
9+
const keyFilePath = config.fcmGoogleApiPathToAuthJson;
10+
const key = JSON.parse(fs.readFileSync(keyFilePath, 'utf8'));
11+
12+
const fcmGoogleApiPath = `https://fcm.googleapis.com/v1/projects/${key.project_id}/messages:send`
913

1014
export const sendFcmNewEpisodeDetectedNotification = async (options: SendNotificationOptions) => {
1115
const { podcastId, podcastShrunkImageUrl, podcastFullImageUrl, episodeFullImageUrl, episodeId } = options
@@ -69,36 +73,25 @@ export const sendFCMGoogleApiNotification = async (
6973
episodeImage?: string,
7074
episodeId?: string
7175
) => {
72-
if (!fcmTokens || fcmTokens.length === 0) return
76+
const accessToken = await generateAccessToken()
7377

74-
const fcmTokenBatches: any[] = []
75-
const size = 1000
76-
for (let i = 0; i < fcmTokens.length; i += size) {
77-
fcmTokenBatches.push(fcmTokens.slice(i, i + size))
78-
}
79-
80-
for (const fcmTokenBatch of fcmTokenBatches) {
81-
if (fcmTokenBatch?.length > 0) {
82-
const imageUrl = episodeImage || podcastImage
78+
if (!fcmTokens || fcmTokens.length === 0) return
8379

84-
try {
85-
await request(fcmGoogleApiPath, {
86-
method: 'POST',
87-
headers: {
88-
Authorization: `key=${fcmGoogleApiAuthToken}`,
89-
'Content-Type': 'application/json'
90-
},
91-
body: {
92-
registration_ids: fcmTokenBatch || [],
80+
for (const fcmToken of fcmTokens) {
81+
const imageUrl = episodeImage || podcastImage
82+
try {
83+
await request(fcmGoogleApiPath, {
84+
method: 'POST',
85+
headers: {
86+
Authorization: `Bearer ${accessToken}`,
87+
'Content-Type': 'application/json'
88+
},
89+
body: {
90+
message: {
91+
token: fcmToken,
9392
notification: {
9493
body,
9594
title,
96-
podcastId,
97-
episodeId,
98-
podcastTitle: podcastTitle,
99-
episodeTitle: episodeTitle,
100-
notificationType,
101-
timeSent: new Date(),
10295
image: imageUrl
10396
},
10497
data: {
@@ -131,12 +124,12 @@ export const sendFCMGoogleApiNotification = async (
131124
image: imageUrl
132125
}
133126
}
134-
},
135-
json: true
136-
})
137-
} catch (error) {
138-
console.log('sendFCMGoogleApiNotification error', error)
139-
}
127+
}
128+
},
129+
json: true
130+
})
131+
} catch (error) {
132+
console.log('sendFCMGoogleApiNotification error', error)
140133
}
141134
}
142135
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
import { config } from '~/config';
2+
import { JWT } from 'google-auth-library';
3+
const fs = require('fs');
4+
5+
const keyFilePath = config.fcmGoogleApiPathToAuthJson;
6+
const key = JSON.parse(fs.readFileSync(keyFilePath, 'utf8'));
7+
8+
const client = new JWT(
9+
key.client_email,
10+
'',
11+
key.private_key,
12+
['https://www.googleapis.com/auth/firebase.messaging'],
13+
);
14+
15+
export async function generateAccessToken() {
16+
const token = await client.authorize();
17+
return token.access_token;
18+
}

yarn.lock

+61
Original file line numberDiff line numberDiff line change
@@ -4514,6 +4514,17 @@ gaxios@^2.0.1, gaxios@^2.1.0:
45144514
is-stream "^2.0.0"
45154515
node-fetch "^2.3.0"
45164516

4517+
gaxios@^6.0.0, gaxios@^6.1.1:
4518+
version "6.7.1"
4519+
resolved "https://registry.yarnpkg.com/gaxios/-/gaxios-6.7.1.tgz#ebd9f7093ede3ba502685e73390248bb5b7f71fb"
4520+
integrity sha512-LDODD4TMYx7XXdpwxAVRAIAuB0bzv0s+ywFonY46k126qzQHT9ygyoa9tncmOiQmmDrik65UYsEkv3lbfqQ3yQ==
4521+
dependencies:
4522+
extend "^3.0.2"
4523+
https-proxy-agent "^7.0.1"
4524+
is-stream "^2.0.0"
4525+
node-fetch "^2.6.9"
4526+
uuid "^9.0.1"
4527+
45174528
gcp-metadata@^3.4.0:
45184529
version "3.5.0"
45194530
resolved "https://registry.yarnpkg.com/gcp-metadata/-/gcp-metadata-3.5.0.tgz#6d28343f65a6bbf8449886a0c0e4a71c77577055"
@@ -4522,6 +4533,14 @@ gcp-metadata@^3.4.0:
45224533
gaxios "^2.1.0"
45234534
json-bigint "^0.3.0"
45244535

4536+
gcp-metadata@^6.1.0:
4537+
version "6.1.0"
4538+
resolved "https://registry.yarnpkg.com/gcp-metadata/-/gcp-metadata-6.1.0.tgz#9b0dd2b2445258e7597f2024332d20611cbd6b8c"
4539+
integrity sha512-Jh/AIwwgaxan+7ZUUmRLCjtchyDiqh4KjBJ5tW3plBZb5iL/BPcso8A5DlzeD9qlw0duCamnNdpFjxwaT0KyKg==
4540+
dependencies:
4541+
gaxios "^6.0.0"
4542+
json-bigint "^1.0.0"
4543+
45254544
45264545
version "3.8.2"
45274546
resolved "https://registry.yarnpkg.com/generic-pool/-/generic-pool-3.8.2.tgz#aab4f280adb522fdfbdc5e5b64d718d3683f04e9"
@@ -4732,6 +4751,18 @@ google-auth-library@^5.2.0, google-auth-library@^5.6.1:
47324751
jws "^4.0.0"
47334752
lru-cache "^5.0.0"
47344753

4754+
google-auth-library@^9.14.1:
4755+
version "9.14.1"
4756+
resolved "https://registry.yarnpkg.com/google-auth-library/-/google-auth-library-9.14.1.tgz#4c6f535f474b01847ea1a60ef1d56dbd6a0aad2f"
4757+
integrity sha512-Rj+PMjoNFGFTmtItH7gHfbHpGVSb3vmnGK3nwNBqxQF9NoBpttSZI/rc0WiM63ma2uGDQtYEkMHkK9U6937NiA==
4758+
dependencies:
4759+
base64-js "^1.3.0"
4760+
ecdsa-sig-formatter "^1.0.11"
4761+
gaxios "^6.1.1"
4762+
gcp-metadata "^6.1.0"
4763+
gtoken "^7.0.0"
4764+
jws "^4.0.0"
4765+
47354766
google-p12-pem@^2.0.0:
47364767
version "2.0.5"
47374768
resolved "https://registry.yarnpkg.com/google-p12-pem/-/google-p12-pem-2.0.5.tgz#b1c44164d567ae894f7a19b4ff362a06be5b793b"
@@ -4796,6 +4827,14 @@ gtoken@^4.1.0:
47964827
jws "^4.0.0"
47974828
mime "^2.2.0"
47984829

4830+
gtoken@^7.0.0:
4831+
version "7.1.0"
4832+
resolved "https://registry.yarnpkg.com/gtoken/-/gtoken-7.1.0.tgz#d61b4ebd10132222817f7222b1e6064bd463fc26"
4833+
integrity sha512-pCcEwRi+TKpMlxAQObHDQ56KawURgyAf6jtIY046fJ5tIv3zDe/LEIubckAO8fj6JnAxLdmWkUfNyulQ2iKdEw==
4834+
dependencies:
4835+
gaxios "^6.0.0"
4836+
jws "^4.0.0"
4837+
47994838
handlebars@^4.4.3:
48004839
version "4.7.7"
48014840
resolved "https://registry.yarnpkg.com/handlebars/-/handlebars-4.7.7.tgz#9ce33416aad02dbd6c8fafa8240d5d98004945a1"
@@ -5121,6 +5160,14 @@ https-proxy-agent@^7.0.0:
51215160
agent-base "^7.0.2"
51225161
debug "4"
51235162

5163+
https-proxy-agent@^7.0.1:
5164+
version "7.0.5"
5165+
resolved "https://registry.yarnpkg.com/https-proxy-agent/-/https-proxy-agent-7.0.5.tgz#9e8b5013873299e11fab6fd548405da2d6c602b2"
5166+
integrity sha512-1e4Wqeblerz+tMKPIq2EMGiiWW1dIjZOksyHWSUm1rmuvw/how9hBHZ38lAGj5ID4Ik6EdkOw7NmWPy6LAwalw==
5167+
dependencies:
5168+
agent-base "^7.0.2"
5169+
debug "4"
5170+
51245171
human-signals@^1.1.1:
51255172
version "1.1.1"
51265173
resolved "https://registry.yarnpkg.com/human-signals/-/human-signals-1.1.1.tgz#c5b1cd14f50aeae09ab6c59fe63ba3395fe4dfa3"
@@ -6301,6 +6348,13 @@ json-bigint@^0.3.0:
63016348
dependencies:
63026349
bignumber.js "^9.0.0"
63036350

6351+
json-bigint@^1.0.0:
6352+
version "1.0.0"
6353+
resolved "https://registry.yarnpkg.com/json-bigint/-/json-bigint-1.0.0.tgz#ae547823ac0cad8398667f8cd9ef4730f5b01ff1"
6354+
integrity sha512-SiPv/8VpZuWbvLSMtTDU8hEfrZWg/mH/nV/b4o0CYbSxu1UIQPLdwKOCIyLQX+VIPO5vrLX3i8qtqFyhdPSUSQ==
6355+
dependencies:
6356+
bignumber.js "^9.0.0"
6357+
63046358
json-parse-better-errors@^1.0.1, json-parse-better-errors@^1.0.2:
63056359
version "1.0.2"
63066360
resolved "https://registry.yarnpkg.com/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz#bb867cfb3450e69107c131d1c514bab3dc8bcaa9"
@@ -7412,6 +7466,13 @@ [email protected], node-fetch@^2.3.0, node-fetch@^2.6.1:
74127466
dependencies:
74137467
whatwg-url "^5.0.0"
74147468

7469+
node-fetch@^2.6.9:
7470+
version "2.7.0"
7471+
resolved "https://registry.yarnpkg.com/node-fetch/-/node-fetch-2.7.0.tgz#d0f0fa6e3e2dc1d27efcd8ad99d550bda94d187d"
7472+
integrity sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==
7473+
dependencies:
7474+
whatwg-url "^5.0.0"
7475+
74157476
node-forge@^0.10.0:
74167477
version "0.10.0"
74177478
resolved "https://registry.yarnpkg.com/node-forge/-/node-forge-0.10.0.tgz#32dea2afb3e9926f02ee5ce8794902691a676bf3"

0 commit comments

Comments
 (0)