mirror of
https://github.com/whekin/household-bot.git
synced 2026-03-31 13:54:02 +00:00
fix(bot): restrict setup admin bootstrap
This commit is contained in:
@@ -148,6 +148,7 @@ function createHouseholdConfigurationRepository(): HouseholdConfigurationReposit
|
|||||||
isAdmin: input.isAdmin === true
|
isAdmin: input.isAdmin === true
|
||||||
}),
|
}),
|
||||||
getHouseholdMember: async () => null,
|
getHouseholdMember: async () => null,
|
||||||
|
listHouseholdMembers: async () => [],
|
||||||
listHouseholdMembersByTelegramUserId: async () => [
|
listHouseholdMembersByTelegramUserId: async () => [
|
||||||
{
|
{
|
||||||
id: 'member-123456',
|
id: 'member-123456',
|
||||||
|
|||||||
@@ -64,6 +64,7 @@ function onboardingRepository(): HouseholdConfigurationRepository {
|
|||||||
isAdmin: input.isAdmin === true
|
isAdmin: input.isAdmin === true
|
||||||
}),
|
}),
|
||||||
getHouseholdMember: async () => null,
|
getHouseholdMember: async () => null,
|
||||||
|
listHouseholdMembers: async () => [],
|
||||||
listHouseholdMembersByTelegramUserId: async () => [],
|
listHouseholdMembersByTelegramUserId: async () => [],
|
||||||
listPendingHouseholdMembers: async () => [
|
listPendingHouseholdMembers: async () => [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -95,6 +95,8 @@ function onboardingRepository(): HouseholdConfigurationRepository {
|
|||||||
return member
|
return member
|
||||||
},
|
},
|
||||||
getHouseholdMember: async (_householdId, telegramUserId) => members.get(telegramUserId) ?? null,
|
getHouseholdMember: async (_householdId, telegramUserId) => members.get(telegramUserId) ?? null,
|
||||||
|
listHouseholdMembers: async (householdId) =>
|
||||||
|
[...members.values()].filter((member) => member.householdId === householdId),
|
||||||
listHouseholdMembersByTelegramUserId: async (telegramUserId) => {
|
listHouseholdMembersByTelegramUserId: async (telegramUserId) => {
|
||||||
const member = members.get(telegramUserId)
|
const member = members.get(telegramUserId)
|
||||||
return member ? [member] : []
|
return member ? [member] : []
|
||||||
|
|||||||
@@ -121,6 +121,7 @@ function onboardingRepository(): HouseholdConfigurationRepository {
|
|||||||
isAdmin: input.isAdmin === true
|
isAdmin: input.isAdmin === true
|
||||||
}),
|
}),
|
||||||
getHouseholdMember: async () => null,
|
getHouseholdMember: async () => null,
|
||||||
|
listHouseholdMembers: async () => [],
|
||||||
listHouseholdMembersByTelegramUserId: async () => [],
|
listHouseholdMembersByTelegramUserId: async () => [],
|
||||||
listPendingHouseholdMembers: async () => [],
|
listPendingHouseholdMembers: async () => [],
|
||||||
approvePendingHouseholdMember: async () => null
|
approvePendingHouseholdMember: async () => null
|
||||||
|
|||||||
@@ -595,6 +595,22 @@ export function createDbHouseholdConfigurationRepository(databaseUrl: string): {
|
|||||||
return row ? toHouseholdMemberRecord(row) : null
|
return row ? toHouseholdMemberRecord(row) : null
|
||||||
},
|
},
|
||||||
|
|
||||||
|
async listHouseholdMembers(householdId) {
|
||||||
|
const rows = await db
|
||||||
|
.select({
|
||||||
|
id: schema.members.id,
|
||||||
|
householdId: schema.members.householdId,
|
||||||
|
telegramUserId: schema.members.telegramUserId,
|
||||||
|
displayName: schema.members.displayName,
|
||||||
|
isAdmin: schema.members.isAdmin
|
||||||
|
})
|
||||||
|
.from(schema.members)
|
||||||
|
.where(eq(schema.members.householdId, householdId))
|
||||||
|
.orderBy(schema.members.displayName, schema.members.telegramUserId)
|
||||||
|
|
||||||
|
return rows.map(toHouseholdMemberRecord)
|
||||||
|
},
|
||||||
|
|
||||||
async listHouseholdMembersByTelegramUserId(telegramUserId) {
|
async listHouseholdMembersByTelegramUserId(telegramUserId) {
|
||||||
const rows = await db
|
const rows = await db
|
||||||
.select({
|
.select({
|
||||||
|
|||||||
@@ -92,6 +92,8 @@ function createRepositoryStub() {
|
|||||||
return record
|
return record
|
||||||
},
|
},
|
||||||
getHouseholdMember: async (_householdId, telegramUserId) => members.get(telegramUserId) ?? null,
|
getHouseholdMember: async (_householdId, telegramUserId) => members.get(telegramUserId) ?? null,
|
||||||
|
listHouseholdMembers: async (householdId) =>
|
||||||
|
[...members.values()].filter((member) => member.householdId === householdId),
|
||||||
listHouseholdMembersByTelegramUserId: async (telegramUserId) =>
|
listHouseholdMembersByTelegramUserId: async (telegramUserId) =>
|
||||||
[...members.values()].filter((member) => member.telegramUserId === telegramUserId),
|
[...members.values()].filter((member) => member.telegramUserId === telegramUserId),
|
||||||
listPendingHouseholdMembers: async () => [...pendingMembers.values()],
|
listPendingHouseholdMembers: async () => [...pendingMembers.values()],
|
||||||
|
|||||||
@@ -101,6 +101,9 @@ function createRepositoryStub() {
|
|||||||
async getHouseholdMember(_householdId, telegramUserId) {
|
async getHouseholdMember(_householdId, telegramUserId) {
|
||||||
return members.get(telegramUserId) ?? null
|
return members.get(telegramUserId) ?? null
|
||||||
},
|
},
|
||||||
|
async listHouseholdMembers(householdId) {
|
||||||
|
return [...members.values()].filter((member) => member.householdId === householdId)
|
||||||
|
},
|
||||||
async listHouseholdMembersByTelegramUserId(telegramUserId) {
|
async listHouseholdMembersByTelegramUserId(telegramUserId) {
|
||||||
const member = members.get(telegramUserId)
|
const member = members.get(telegramUserId)
|
||||||
return member ? [member] : []
|
return member ? [member] : []
|
||||||
|
|||||||
@@ -176,6 +176,10 @@ function createRepositoryStub() {
|
|||||||
return members.get(`${householdId}:${telegramUserId}`) ?? null
|
return members.get(`${householdId}:${telegramUserId}`) ?? null
|
||||||
},
|
},
|
||||||
|
|
||||||
|
async listHouseholdMembers(householdId) {
|
||||||
|
return [...members.values()].filter((member) => member.householdId === householdId)
|
||||||
|
},
|
||||||
|
|
||||||
async listHouseholdMembersByTelegramUserId(telegramUserId) {
|
async listHouseholdMembersByTelegramUserId(telegramUserId) {
|
||||||
return [...members.values()].filter((member) => member.telegramUserId === telegramUserId)
|
return [...members.values()].filter((member) => member.telegramUserId === telegramUserId)
|
||||||
},
|
},
|
||||||
@@ -240,6 +244,77 @@ describe('createHouseholdSetupService', () => {
|
|||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
test('ensures the actor becomes admin when rerunning setup for an existing household with no admins', async () => {
|
||||||
|
const { repository } = createRepositoryStub()
|
||||||
|
const service = createHouseholdSetupService(repository)
|
||||||
|
|
||||||
|
await service.setupGroupChat({
|
||||||
|
actorIsAdmin: true,
|
||||||
|
telegramChatId: '-100123',
|
||||||
|
telegramChatType: 'supergroup',
|
||||||
|
title: 'Kojori House'
|
||||||
|
})
|
||||||
|
|
||||||
|
const result = await service.setupGroupChat({
|
||||||
|
actorIsAdmin: true,
|
||||||
|
actorTelegramUserId: '77',
|
||||||
|
actorDisplayName: 'Mia',
|
||||||
|
telegramChatId: '-100123',
|
||||||
|
telegramChatType: 'supergroup',
|
||||||
|
title: 'Kojori House'
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(result.status).toBe('existing')
|
||||||
|
if (result.status !== 'existing') {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const admin = await repository.getHouseholdMember(result.household.householdId, '77')
|
||||||
|
expect(admin).toEqual({
|
||||||
|
id: 'member-77',
|
||||||
|
householdId: result.household.householdId,
|
||||||
|
telegramUserId: '77',
|
||||||
|
displayName: 'Mia',
|
||||||
|
isAdmin: true
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
test('does not grant admin when rerunning setup for an existing household that already has admins', async () => {
|
||||||
|
const { repository } = createRepositoryStub()
|
||||||
|
const service = createHouseholdSetupService(repository)
|
||||||
|
|
||||||
|
const created = await service.setupGroupChat({
|
||||||
|
actorIsAdmin: true,
|
||||||
|
actorTelegramUserId: '42',
|
||||||
|
actorDisplayName: 'Stan',
|
||||||
|
telegramChatId: '-100123',
|
||||||
|
telegramChatType: 'supergroup',
|
||||||
|
title: 'Kojori House'
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(created.status).toBe('created')
|
||||||
|
if (created.status !== 'created') {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const result = await service.setupGroupChat({
|
||||||
|
actorIsAdmin: true,
|
||||||
|
actorTelegramUserId: '77',
|
||||||
|
actorDisplayName: 'Mia',
|
||||||
|
telegramChatId: '-100123',
|
||||||
|
telegramChatType: 'supergroup',
|
||||||
|
title: 'Kojori House'
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(result.status).toBe('existing')
|
||||||
|
if (result.status !== 'existing') {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const admin = await repository.getHouseholdMember(result.household.householdId, '77')
|
||||||
|
expect(admin).toBeNull()
|
||||||
|
})
|
||||||
|
|
||||||
test('rejects setup when the actor is not a group admin', async () => {
|
test('rejects setup when the actor is not a group admin', async () => {
|
||||||
const { repository } = createRepositoryStub()
|
const { repository } = createRepositoryStub()
|
||||||
const service = createHouseholdSetupService(repository)
|
const service = createHouseholdSetupService(repository)
|
||||||
|
|||||||
@@ -85,13 +85,20 @@ export function createHouseholdSetupService(
|
|||||||
: {})
|
: {})
|
||||||
})
|
})
|
||||||
|
|
||||||
if (registered.status === 'created' && input.actorTelegramUserId && input.actorDisplayName) {
|
if (input.actorTelegramUserId && input.actorDisplayName) {
|
||||||
await repository.ensureHouseholdMember({
|
const existingMembers = await repository.listHouseholdMembers(
|
||||||
householdId: registered.household.householdId,
|
registered.household.householdId
|
||||||
telegramUserId: input.actorTelegramUserId,
|
)
|
||||||
displayName: input.actorDisplayName,
|
const hasAdmin = existingMembers.some((member) => member.isAdmin)
|
||||||
isAdmin: true
|
|
||||||
})
|
if (registered.status === 'created' || !hasAdmin) {
|
||||||
|
await repository.ensureHouseholdMember({
|
||||||
|
householdId: registered.household.householdId,
|
||||||
|
telegramUserId: input.actorTelegramUserId,
|
||||||
|
displayName: input.actorDisplayName,
|
||||||
|
isAdmin: true
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
|||||||
@@ -53,6 +53,7 @@ function repository(): HouseholdConfigurationRepository {
|
|||||||
isAdmin: input.isAdmin === true
|
isAdmin: input.isAdmin === true
|
||||||
}),
|
}),
|
||||||
getHouseholdMember: async () => null,
|
getHouseholdMember: async () => null,
|
||||||
|
listHouseholdMembers: async () => [],
|
||||||
listHouseholdMembersByTelegramUserId: async () => [],
|
listHouseholdMembersByTelegramUserId: async () => [],
|
||||||
listPendingHouseholdMembers: async () => [
|
listPendingHouseholdMembers: async () => [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -105,6 +105,7 @@ export interface HouseholdConfigurationRepository {
|
|||||||
householdId: string,
|
householdId: string,
|
||||||
telegramUserId: string
|
telegramUserId: string
|
||||||
): Promise<HouseholdMemberRecord | null>
|
): Promise<HouseholdMemberRecord | null>
|
||||||
|
listHouseholdMembers(householdId: string): Promise<readonly HouseholdMemberRecord[]>
|
||||||
listHouseholdMembersByTelegramUserId(
|
listHouseholdMembersByTelegramUserId(
|
||||||
telegramUserId: string
|
telegramUserId: string
|
||||||
): Promise<readonly HouseholdMemberRecord[]>
|
): Promise<readonly HouseholdMemberRecord[]>
|
||||||
|
|||||||
Reference in New Issue
Block a user