Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feat/questions supabase #4053

Merged
merged 43 commits into from
Jan 18, 2025
Merged
Show file tree
Hide file tree
Changes from 37 commits
Commits
Show all changes
43 commits
Select commit Hold shift + click to select a range
e02543f
chore: add tenant_id to cypress environment
mariojsnunes Dec 10, 2024
1d508aa
chore: seed database
mariojsnunes Dec 10, 2024
4df42f0
wip questions
mariojsnunes Dec 10, 2024
e3d8b6a
wip questions
mariojsnunes Dec 17, 2024
eb170c5
Merge branch 'master' into feat/questions-supabase
mariojsnunes Dec 20, 2024
69912aa
feat: questions supabase
mariojsnunes Dec 28, 2024
ee6c71d
fix: discussions with supabase questions
mariojsnunes Dec 29, 2024
e58c5f8
fix: notifications with supabase questions;
mariojsnunes Dec 29, 2024
09139cd
feat: image public url and transforms
mariojsnunes Jan 3, 2025
9790968
fix: questions list
mariojsnunes Jan 4, 2025
0469958
merge
mariojsnunes Jan 4, 2025
c8c306a
feat: added sort by comments
mariojsnunes Jan 6, 2025
f42a0a3
fix import
mariojsnunes Jan 7, 2025
d73ce48
questions migration script
mariojsnunes Jan 7, 2025
758b146
Merge branch 'master' into feat/questions-supabase
mariojsnunes Jan 9, 2025
cee0ed1
merge
mariojsnunes Jan 9, 2025
9a5f00a
remove guidance for files
mariojsnunes Jan 9, 2025
3b78489
remove question store
mariojsnunes Jan 9, 2025
27037b2
remove question store
mariojsnunes Jan 9, 2025
35390e5
feat: duplicate question validation and error message
mariojsnunes Jan 9, 2025
43ae242
fix: spec for file category guidance
mariojsnunes Jan 9, 2025
a9cfe8f
Merge branch 'chore/comments-v2-tests' into feat/questions-supabase
mariojsnunes Jan 9, 2025
9beeb20
chore: updated cypress tests with supabase
mariojsnunes Jan 9, 2025
476f89c
chore: update migration script;
mariojsnunes Jan 9, 2025
f71fc3d
fix
mariojsnunes Jan 10, 2025
c08417e
remove unique check
mariojsnunes Jan 10, 2025
2491f44
chore: fix tests to work with supabase;
mariojsnunes Jan 11, 2025
56e8293
fix: edit question submit button
mariojsnunes Jan 11, 2025
be11d96
fix
mariojsnunes Jan 11, 2025
f38567c
fix: library category
mariojsnunes Jan 11, 2025
7166865
fix: disable image transform
mariojsnunes Jan 13, 2025
88dcbd6
re-add image transforms
mariojsnunes Jan 13, 2025
be13a31
cleanup
mariojsnunes Jan 15, 2025
0c63d87
chore: add reply read test
mariojsnunes Jan 15, 2025
2bdbd86
Merge branch 'master' into feat/questions-supabase
mariojsnunes Jan 16, 2025
c0e93ae
Merge branch 'master' into feat/questions-supabase
mariojsnunes Jan 16, 2025
7cd2a34
Merge branch 'master' into feat/questions-supabase
mariojsnunes Jan 17, 2025
28fc2fb
Merge branch 'master' into feat/questions-supabase
mariojsnunes Jan 18, 2025
313d736
feat: add storage local setup
mariojsnunes Jan 18, 2025
2bdd8a3
fix: question tests
mariojsnunes Jan 18, 2025
cf4a23f
chore: comment out notification tests as they are flaky
mariojsnunes Jan 18, 2025
26184d8
chore: lint
mariojsnunes Jan 18, 2025
53be17b
chore: flaky map test
mariojsnunes Jan 18, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
26 changes: 22 additions & 4 deletions docs/supabase.md
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
### What is Supabase?
# What is Supabase?

Supabase is an open source Firebase alternative, based on Postgres.

Expand All @@ -12,14 +12,22 @@ Make sure you have the docker app open.

Run `supabase start` (Ensure you run it on the project folder root.)

## Running Cypress Tests

Create a .env.local file at the packages/cypress folder
SUPABASE_API_URL=your_api_key (probably http://127.0.0.1:54321)
SUPABASE_KEY=your_key

All done! Tests will use your local database. More info about how it works below.

## Migrations

After making schema changes, use the this command to create a migration file:
`supabase db diff --file [migration_name]`

## Technical Decisions

# Multi-tenant
### Multi-tenant

Multi-tenancy is a requirement because:

Expand All @@ -43,7 +51,7 @@ How?
- Each table has a tenant_id column
- On each request, to supabase (via it's sdk) we pass a header 'x-tenant-id' with the process.env.TENANT_ID variable, which is set for each app, via Fly.io secret.

# Comment Counts
### Comment Counts

Currently we can sort questions/research/howtos by the number of comments.
With supabase there are a few ways we can do this:
Expand All @@ -63,7 +71,17 @@ How?
- The function checks the Operation kind (Insert/Delete), the source_type and source_id.
- Fron the source_type it will update the according content total (howtos, research, questions) that matches the source_id

# Local firebase sync testing/debugging
### Cypress with Supabase

Running cypress tests locally will use the local database, while running on CI will use the QA database.
For each test run, a new tenant_id is generated, which has a few benefits:

- ensures no conflicts between parallel test runs
- easier to cleanup
- if the data isn't cleaned for some reason, it won't affect other runs
For each test file, there should be a `before` and `after` block to, respectively, seed and clean the database.

### Local firebase sync testing/debugging

_This is temporary until we fully migrate to supabase!_
We can create and deploy the sync function to the firebase dev environment.
Expand Down
3 changes: 2 additions & 1 deletion packages/cypress/.gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -3,4 +3,5 @@ coverage
screenshots
fixtures/seed/*.rej
build
*.js
*.js
cypress.env.json
12 changes: 12 additions & 0 deletions packages/cypress/scripts/start.mts
Original file line number Diff line number Diff line change
Expand Up @@ -45,9 +45,20 @@ export const generateAlphaNumeric = (length: number) => {
}

const e2eEnv = config()
config({ path: '.env.local' })

const isCi = process.argv.includes('ci')
// const isProduction = process.argv.includes('prod')
const tenantId = generateAlphaNumeric(8)

fs.writeFileSync(
'cypress.env.json',
JSON.stringify({
TENANT_ID: tenantId,
SUPABASE_API_URL: process.env.SUPABASE_API_URL,
SUPABASE_KEY: process.env.SUPABASE_KEY,
}),
)

// Prevent unhandled errors being silently ignored
process.on('unhandledRejection', (err) => {
Expand Down Expand Up @@ -110,6 +121,7 @@ async function startAppServer() {
env: {
...process.env,
VITE_SITE_VARIANT: 'test-ci',
TENANT_ID: tenantId,
},
})

Expand Down
3 changes: 0 additions & 3 deletions packages/cypress/src/integration/library/write.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -177,7 +177,6 @@ describe('[Library]', () => {
]
const categoryGuidanceMain =
'Cover image should show the fully built mould'
const categoryGuidanceFiles = 'Include files to replicate the mould'

cy.signUpNewUser(creator)
cy.get('[data-cy=loader]').should('not.exist')
Expand Down Expand Up @@ -234,10 +233,8 @@ describe('[Library]', () => {

cy.step('Select a category and see further guidance')
cy.contains(categoryGuidanceMain).should('not.exist')
cy.contains(categoryGuidanceFiles).should('not.exist')
selectCategory(category as Category)
cy.contains(categoryGuidanceMain).should('be.visible')
cy.contains(categoryGuidanceFiles).should('be.visible')

selectTimeDuration(time as Duration)
selectDifficultLevel(difficulty_level)
Expand Down
79 changes: 79 additions & 0 deletions packages/cypress/src/integration/questions/discussions.spec.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,79 @@
// This is basically an identical set of steps to the discussion tests for
// how-tos and research. Any changes here should be replicated there.

import { MOCK_DATA } from '../../data'
import { clearDatabase } from '../../support/commands'
import { seedQuestionComments } from '../../support/seedQuestions'
// import { question } from '../../fixtures/question'
// import { generateNewUserDetails } from '../../utils/TestUtils'

describe('[Questions.Discussions]', () => {
let commentId = ''
before(() => {
cy.then(async () => {
const commentData = await seedQuestionComments()
commentId = commentData.comments.data[0].id
})
})

it('can open using deep links', () => {
const question = MOCK_DATA.questions[0]
cy.visit(`/questions/${question.slug}#comment:${commentId}`)
cy.get(`[id="comment:${commentId}"]`).should('be.visible')
cy.get(`[id="comment:${commentId}"]`).contains('Show 1 reply')
cy.get('[data-cy=show-replies]').click()
cy.get(`[data-cy="ReplyItem"]`).contains('First Reply')
})

// it('allows authenticated users to contribute to discussions', () => {
mariojsnunes marked this conversation as resolved.
Show resolved Hide resolved
// const visitor = generateNewUserDetails()
// cy.addQuestion(question, visitor)
// cy.signUpNewUser(visitor)
// const newComment = `An interesting question. The answer must be... ${visitor.username}`
// const updatedNewComment = `An interesting question. The answer must be that when the sky is red, the apocalypse _might_ be on the way. Love, ${visitor.username}`
// const newReply = `Thanks Dave and Ben. What does everyone else think? - ${visitor.username}`
// const updatedNewReply = `Anyone else? Your truly ${visitor.username}`
// const questionPath = `/questions/quick-question-for-${visitor.username}`
// cy.step('Can add comment')
// cy.visit(questionPath)
// cy.contains('Start the discussion')
// cy.contains('0 comments')
// cy.addComment(newComment)
// cy.contains('1 comment')
// cy.step('Can edit their comment')
// cy.editDiscussionItem('CommentItem', newComment, updatedNewComment)
// cy.step('Another user can add reply')
// const secondCommentor = generateNewUserDetails()
// cy.logout()
// cy.signUpNewUser(secondCommentor)
// cy.visit(questionPath)
// cy.addReply(newReply)
// cy.wait(1000)
// cy.contains('2 comments')
// cy.step('Can edit their reply')
// cy.editDiscussionItem('ReplyItem', newReply, updatedNewReply)
// cy.step('Another user can leave a reply')
// const secondReply = `Quick reply. ${visitor.username}`
// cy.step('First commentor can respond')
// cy.logout()
// cy.login(visitor.email, visitor.password)
// cy.visit(questionPath)
// cy.addReply(secondReply)
// cy.step('Can delete their comment')
// cy.deleteDiscussionItem('CommentItem', updatedNewComment)
// cy.step('Replies still show for deleted comments')
// cy.get('[data-cy="deletedComment"]').should('be.visible')
// cy.get('[data-cy=OwnReplyItem]').contains(secondReply)
// cy.step('Can delete their reply')
// cy.deleteDiscussionItem('ReplyItem', secondReply)
// })

after(() => {
const tenantId = Cypress.env('TENANT_ID')
Cypress.log({
displayName: 'Clearing database for tenant',
message: tenantId,
})
clearDatabase(['profiles', 'questions', 'comments'], tenantId)
})
})
49 changes: 23 additions & 26 deletions packages/cypress/src/integration/questions/read.spec.ts
Original file line number Diff line number Diff line change
@@ -1,8 +1,15 @@
import { MOCK_DATA } from '../../data'
import { clearDatabase } from '../../support/commands'
import { seedQuestions } from '../../support/seedQuestions'

const question = Object.values(MOCK_DATA.questions)[0]
const question = MOCK_DATA.questions[0]

describe('[Questions]', () => {
before(() => {
cy.then(async () => {
await seedQuestions()
})
})
describe('[List questions]', () => {
it('[By Everyone]', () => {
cy.visit(`/questions/`)
Expand All @@ -26,18 +33,9 @@ describe('[Questions]', () => {

describe('[Individual questions]', () => {
it('[By Everyone]', () => {
const {
description,
images,
slug,
subscribers,
title,
votedUsefulBy,
questionCategory,
} = question
const { description, slug, title } = question

const pageTitle = `${title} - Question - Precious Plastic`
const image = images[0].downloadUrl

cy.step('Can visit question')
cy.visit(`/questions/${slug}`)
Expand Down Expand Up @@ -65,7 +63,6 @@ describe('[Questions]', () => {
'content',
description,
)
cy.get('meta[property="og:image"]').should('have.attr', 'content', image)

// Twitter
cy.get('meta[name="twitter:title"]').should(
Expand All @@ -78,8 +75,6 @@ describe('[Questions]', () => {
'content',
description,
)
cy.get('meta[name="twitter:image"]').should('have.attr', 'content', image)

cy.step('Links in description are clickable')
cy.contains('a', 'https://www.onearmy.earth/')

Expand All @@ -91,26 +86,28 @@ describe('[Questions]', () => {
.should('have.attr', 'href')
.and('equal', `/questions`)

cy.get('[data-cy=breadcrumbsItem]')
.eq(1)
.should('contain', questionCategory.label)
cy.get('[data-cy=breadcrumbsItem]')
.eq(1)
.children()
.should('have.attr', 'href')
.and('equal', `/questions?category=${questionCategory._id}`)
cy.get('[data-cy=breadcrumbsItem]').eq(1).should('contain', 'Machines')

cy.get('[data-cy=breadcrumbsItem]').eq(2).should('contain', title)

cy.step('Logged in users can complete actions')
cy.login('[email protected]', 'test1234')
cy.visit(`/questions/${slug}`) // Page doesn't reload after login

cy.get('[data-cy=follow-button]').click()
cy.contains(`${subscribers.length + 1} following`)
// cy.get('[data-cy=follow-button]').click()
// cy.contains(`1 following`)

// cy.get('[data-cy=vote-useful]').click()
// cy.contains(`1 useful`)
})
})

cy.get('[data-cy=vote-useful]').click()
cy.contains(`${votedUsefulBy.length + 1} useful`)
after(() => {
const tenantId = Cypress.env('TENANT_ID')
Cypress.log({
displayName: 'Clearing database for tenant',
message: tenantId,
})
clearDatabase(['profiles', 'questions', 'categories'], tenantId)
mariojsnunes marked this conversation as resolved.
Show resolved Hide resolved
})
})
39 changes: 22 additions & 17 deletions packages/cypress/src/integration/questions/search.spec.ts
Original file line number Diff line number Diff line change
@@ -1,9 +1,18 @@
import { clearDatabase } from '../../support/commands'
import { seedQuestions } from '../../support/seedQuestions'

describe('[How To]', () => {
beforeEach(() => {
cy.visit('/questions')
})

describe('[By Everyone]', () => {
before(() => {
cy.then(async () => {
await seedQuestions()
})
})

mariojsnunes marked this conversation as resolved.
Show resolved Hide resolved
it('Searches', () => {
cy.step('Can search for items')
cy.get('[data-cy=questions-search-box]').clear().type(`deal`)
Expand All @@ -28,30 +37,17 @@ describe('[How To]', () => {

it('Filters', () => {
cy.step('Can select a category to limit items displayed')
cy.get('[data-cy=category]').contains('exhibition')
cy.get('[data-cy=CategoryVerticalList]').within(() => {
cy.contains('screening').click()
cy.contains('Machines').click()
})
cy.get('[data-cy=CategoryVerticalList-Item-active]')
cy.url().should('include', 'category=categoryoix4r6grC1mMA0Xz3K')
cy.get('[data-cy=question-list-item]').its('length').should('be.eq', 1)
cy.get('[data-cy=category]').contains('screening')
cy.url().should('include', 'category=')

cy.step('Can remove the category filter by selecting it again')
cy.get('[data-cy=CategoryVerticalList]').within(() => {
cy.contains('screening').click()
cy.contains('Machines').click()
})
cy.url().should('not.include', 'category=categoryoix4r6grC1mMA0Xz3K')
cy.get('[data-cy=category]').contains('exhibition')

cy.step('Going to an item removes the filter on return')
cy.get('[data-cy=CategoryVerticalList]').within(() => {
cy.contains('screening').click()
})
cy.wait(500)
cy.get('[data-cy=question-list-item]').click()
cy.go('back')
cy.url().should('not.include', 'category=categoryoix4r6grC1mMA0Xz3K')
cy.url().should('not.include', 'category=')
})

it('should show question list items after visit a question', () => {
Expand All @@ -66,5 +62,14 @@ describe('[How To]', () => {
cy.get('[data-cy=load-more]').click()
cy.get('[data-cy=question-list-item]:eq(21)').should('exist')
})

after(() => {
const tenantId = Cypress.env('TENANT_ID')
Cypress.log({
displayName: 'Clearing database for tenant',
message: tenantId,
})
clearDatabase(['profiles', 'questions', 'categories'], tenantId)
})
})
})
Loading
Loading