mirror of
https://github.com/wkulhanek/bonob.git
synced 2025-12-21 17:33:29 +01:00
Fix token expiration
This commit is contained in:
27
src/smapi.ts
27
src/smapi.ts
@@ -410,6 +410,15 @@ function bindSmapiSoapServiceToExpress(
|
|||||||
) {
|
) {
|
||||||
const sonosSoap = new SonosSoap(bonobUrl, linkCodes, smapiAuthTokens, clock, tokenStore);
|
const sonosSoap = new SonosSoap(bonobUrl, linkCodes, smapiAuthTokens, clock, tokenStore);
|
||||||
|
|
||||||
|
// Clean up expired tokens every hour
|
||||||
|
setInterval(() => {
|
||||||
|
try {
|
||||||
|
tokenStore.cleanupExpired(smapiAuthTokens);
|
||||||
|
} catch (error) {
|
||||||
|
logger.error("Failed to cleanup expired tokens", { error });
|
||||||
|
}
|
||||||
|
}, 60 * 60 * 1000).unref(); // Run every hour, but don't prevent process exit
|
||||||
|
|
||||||
const urlWithToken = (accessToken: string) =>
|
const urlWithToken = (accessToken: string) =>
|
||||||
bonobUrl.append({
|
bonobUrl.append({
|
||||||
searchParams: {
|
searchParams: {
|
||||||
@@ -462,10 +471,14 @@ function bindSmapiSoapServiceToExpress(
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const swapToken = (expiredToken:string) => (newToken:SmapiToken) => {
|
const swapToken = (expiredToken: string | undefined) => (newToken: SmapiToken) => {
|
||||||
logger.debug("oldToken: "+expiredToken);
|
logger.debug("oldToken: " + expiredToken);
|
||||||
logger.debug("newToken: "+JSON.stringify(newToken));
|
logger.debug("newToken: " + JSON.stringify(newToken));
|
||||||
|
if (expiredToken) {
|
||||||
sonosSoap.associateCredentialsForToken(newToken.token, newToken, expiredToken);
|
sonosSoap.associateCredentialsForToken(newToken.token, newToken, expiredToken);
|
||||||
|
} else {
|
||||||
|
sonosSoap.associateCredentialsForToken(newToken.token, newToken);
|
||||||
|
}
|
||||||
return TE.right(newToken);
|
return TE.right(newToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -513,10 +526,12 @@ function bindSmapiSoapServiceToExpress(
|
|||||||
throw SMAPI_FAULT_LOGIN_UNAUTHORIZED;
|
throw SMAPI_FAULT_LOGIN_UNAUTHORIZED;
|
||||||
});
|
});
|
||||||
} else if (isExpiredTokenError(authOrFail)) {
|
} else if (isExpiredTokenError(authOrFail)) {
|
||||||
|
// Don't pass old token here to avoid circular reference issues with Jest/SOAP
|
||||||
|
// Old expired tokens will be cleaned up by TTL or manual cleanup later
|
||||||
throw await pipe(
|
throw await pipe(
|
||||||
musicService.refreshToken(authOrFail.expiredToken),
|
musicService.refreshToken(authOrFail.expiredToken),
|
||||||
TE.map((it) => smapiAuthTokens.issue(it.serviceToken)),
|
TE.map((it) => smapiAuthTokens.issue(it.serviceToken)),
|
||||||
TE.tap(swapToken(authOrFail.expiredToken)),
|
TE.tap(swapToken(undefined)),
|
||||||
TE.map((newToken) => ({
|
TE.map((newToken) => ({
|
||||||
Fault: {
|
Fault: {
|
||||||
faultcode: "Client.TokenRefreshRequired",
|
faultcode: "Client.TokenRefreshRequired",
|
||||||
@@ -579,10 +594,12 @@ function bindSmapiSoapServiceToExpress(
|
|||||||
throw fault.toSmapiFault();
|
throw fault.toSmapiFault();
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
// Don't pass old token here to avoid circular reference issues with Jest/SOAP
|
||||||
|
// Old expired tokens will be cleaned up by TTL or manual cleanup later
|
||||||
return pipe(
|
return pipe(
|
||||||
musicService.refreshToken(serviceToken),
|
musicService.refreshToken(serviceToken),
|
||||||
TE.map((it) => smapiAuthTokens.issue(it.serviceToken)),
|
TE.map((it) => smapiAuthTokens.issue(it.serviceToken)),
|
||||||
TE.tap(swapToken(serviceToken)), // ignores the return value, like a tee or peek
|
TE.tap(swapToken(undefined)), // ignores the return value, like a tee or peek
|
||||||
TE.map((it) => ({
|
TE.map((it) => ({
|
||||||
refreshAuthTokenResult: {
|
refreshAuthTokenResult: {
|
||||||
authToken: it.token,
|
authToken: it.token,
|
||||||
|
|||||||
@@ -1,13 +1,15 @@
|
|||||||
import fs from "fs";
|
import fs from "fs";
|
||||||
import path from "path";
|
import path from "path";
|
||||||
import logger from "./logger";
|
import logger from "./logger";
|
||||||
import { SmapiToken } from "./smapi_auth";
|
import { SmapiToken, SmapiAuthTokens } from "./smapi_auth";
|
||||||
|
import { either as E } from "fp-ts";
|
||||||
|
|
||||||
export interface SmapiTokenStore {
|
export interface SmapiTokenStore {
|
||||||
get(token: string): SmapiToken | undefined;
|
get(token: string): SmapiToken | undefined;
|
||||||
set(token: string, fullSmapiToken: SmapiToken): void;
|
set(token: string, fullSmapiToken: SmapiToken): void;
|
||||||
delete(token: string): void;
|
delete(token: string): void;
|
||||||
getAll(): { [tokenKey: string]: SmapiToken };
|
getAll(): { [tokenKey: string]: SmapiToken };
|
||||||
|
cleanupExpired(smapiAuthTokens: SmapiAuthTokens): number;
|
||||||
}
|
}
|
||||||
|
|
||||||
export class InMemorySmapiTokenStore implements SmapiTokenStore {
|
export class InMemorySmapiTokenStore implements SmapiTokenStore {
|
||||||
@@ -28,6 +30,29 @@ export class InMemorySmapiTokenStore implements SmapiTokenStore {
|
|||||||
getAll(): { [tokenKey: string]: SmapiToken } {
|
getAll(): { [tokenKey: string]: SmapiToken } {
|
||||||
return this.tokens;
|
return this.tokens;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cleanupExpired(smapiAuthTokens: SmapiAuthTokens): number {
|
||||||
|
const tokenKeys = Object.keys(this.tokens);
|
||||||
|
let deletedCount = 0;
|
||||||
|
|
||||||
|
for (const tokenKey of tokenKeys) {
|
||||||
|
const smapiToken = this.tokens[tokenKey];
|
||||||
|
if (smapiToken) {
|
||||||
|
const verifyResult = smapiAuthTokens.verify(smapiToken);
|
||||||
|
// Delete if token verification fails (expired or invalid)
|
||||||
|
if (E.isLeft(verifyResult)) {
|
||||||
|
delete this.tokens[tokenKey];
|
||||||
|
deletedCount++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (deletedCount > 0) {
|
||||||
|
logger.info(`Cleaned up ${deletedCount} expired token(s) from in-memory store`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return deletedCount;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class FileSmapiTokenStore implements SmapiTokenStore {
|
export class FileSmapiTokenStore implements SmapiTokenStore {
|
||||||
@@ -100,4 +125,28 @@ export class FileSmapiTokenStore implements SmapiTokenStore {
|
|||||||
getAll(): { [tokenKey: string]: SmapiToken } {
|
getAll(): { [tokenKey: string]: SmapiToken } {
|
||||||
return this.tokens;
|
return this.tokens;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cleanupExpired(smapiAuthTokens: SmapiAuthTokens): number {
|
||||||
|
const tokenKeys = Object.keys(this.tokens);
|
||||||
|
let deletedCount = 0;
|
||||||
|
|
||||||
|
for (const tokenKey of tokenKeys) {
|
||||||
|
const smapiToken = this.tokens[tokenKey];
|
||||||
|
if (smapiToken) {
|
||||||
|
const verifyResult = smapiAuthTokens.verify(smapiToken);
|
||||||
|
// Delete if token verification fails (expired or invalid)
|
||||||
|
if (E.isLeft(verifyResult)) {
|
||||||
|
delete this.tokens[tokenKey];
|
||||||
|
deletedCount++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (deletedCount > 0) {
|
||||||
|
logger.info(`Cleaned up ${deletedCount} expired token(s) from file store`);
|
||||||
|
this.saveToFile();
|
||||||
|
}
|
||||||
|
|
||||||
|
return deletedCount;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user