diff --git a/api/resolvers/user.js b/api/resolvers/user.js
index 91c13968e..90e9c22c1 100644
--- a/api/resolvers/user.js
+++ b/api/resolvers/user.js
@@ -1,3 +1,5 @@
+import { readFile } from 'fs/promises'
+import { join, resolve } from 'path'
import { GraphQLError } from 'graphql'
import { decodeCursor, LIMIT, nextCursorEncoded } from '../../lib/cursor'
import { msatsToSats } from '../../lib/format'
@@ -5,6 +7,20 @@ import { bioSchema, emailSchema, settingsSchema, ssValidate, userSchema } from '
import { getItem, updateItem, filterClause, createItem } from './item'
import { datePivot } from '../../lib/time'
+const contributors = new Set()
+
+const loadContributors = async (set) => {
+ try {
+ const fileContent = await readFile(resolve(join(process.cwd(), 'contributors.txt')), 'utf-8')
+ fileContent.split('\n')
+ .map(line => line.trim())
+ .filter(line => !!line)
+ .forEach(name => set.add(name))
+ } catch (err) {
+ console.error('Error loading contributors', err)
+ }
+}
+
export function within (table, within) {
let interval = ' AND "' + table + '".created_at >= $1 - INTERVAL '
switch (within) {
@@ -817,6 +833,16 @@ export default {
})
return !!subscription?.commentsSubscribedAt
+ },
+ isContributor: async (user, args, { me }) => {
+ // lazy init contributors only once
+ if (contributors.size === 0) {
+ await loadContributors(contributors)
+ }
+ if (me?.id === user.id) {
+ return contributors.has(user.name)
+ }
+ return !user.hideIsContributor && contributors.has(user.name)
}
}
}
diff --git a/api/typeDefs/user.js b/api/typeDefs/user.js
index 95b102e60..3cb97a9f6 100644
--- a/api/typeDefs/user.js
+++ b/api/typeDefs/user.js
@@ -25,7 +25,7 @@ export default gql`
noteInvites: Boolean!, noteJobIndicator: Boolean!, noteCowboyHat: Boolean!, hideInvoiceDesc: Boolean!,
hideFromTopUsers: Boolean!, hideCowboyHat: Boolean!, clickToLoadImg: Boolean!,
wildWestMode: Boolean!, greeterMode: Boolean!, nostrPubkey: String, nostrRelays: [String!], hideBookmarks: Boolean!,
- noteForwardedSats: Boolean!, hideWalletBalance: Boolean!): User
+ noteForwardedSats: Boolean!, hideWalletBalance: Boolean!, hideIsContributor: Boolean!): User
setPhoto(photoId: ID!): Int!
upsertBio(bio: String!): User!
setWalkthrough(tipPopover: Boolean, upvotePopover: Boolean): Boolean
@@ -92,6 +92,8 @@ export default gql`
greeterMode: Boolean!
lastCheckedJobs: String
authMethods: AuthMethods!
+ isContributor: Boolean!
+ hideIsContributor: Boolean!
meSubscriptionPosts: Boolean!
meSubscriptionComments: Boolean!
}
diff --git a/components/user-header.js b/components/user-header.js
index aaa0f1bf5..6bd88da6a 100644
--- a/components/user-header.js
+++ b/components/user-header.js
@@ -212,6 +212,7 @@ function HeaderHeader ({ user }) {
: never}
longest cowboy streak: {user.maxStreak !== null ? user.maxStreak : 'none'}
+ {user.isContributor && 🧑💻✅ verified stacker.news contributor}
diff --git a/contributors.txt b/contributors.txt
new file mode 100644
index 000000000..5f59b5215
--- /dev/null
+++ b/contributors.txt
@@ -0,0 +1,4 @@
+k00b
+kr
+ekzyis
+WeAreAllSatoshi
diff --git a/fragments/users.js b/fragments/users.js
index d0d17e54d..6871ceac5 100644
--- a/fragments/users.js
+++ b/fragments/users.js
@@ -36,6 +36,8 @@ export const ME = gql`
lastCheckedJobs
hideWelcomeBanner
hideWalletBalance
+ isContributor
+ hideIsContributor
}
}`
@@ -57,6 +59,7 @@ export const SETTINGS_FIELDS = gql`
hideFromTopUsers
hideCowboyHat
hideBookmarks
+ hideIsContributor
clickToLoadImg
hideWalletBalance
nostrPubkey
@@ -88,14 +91,14 @@ mutation setSettings($tipDefault: Int!, $turboTipping: Boolean!, $fiatCurrency:
$noteInvites: Boolean!, $noteJobIndicator: Boolean!, $noteCowboyHat: Boolean!, $hideInvoiceDesc: Boolean!,
$hideFromTopUsers: Boolean!, $hideCowboyHat: Boolean!, $clickToLoadImg: Boolean!,
$wildWestMode: Boolean!, $greeterMode: Boolean!, $nostrPubkey: String, $nostrRelays: [String!], $hideBookmarks: Boolean!,
- $noteForwardedSats: Boolean!, $hideWalletBalance: Boolean!) {
+ $noteForwardedSats: Boolean!, $hideWalletBalance: Boolean!, $hideIsContributor: Boolean!) {
setSettings(tipDefault: $tipDefault, turboTipping: $turboTipping, fiatCurrency: $fiatCurrency,
noteItemSats: $noteItemSats, noteEarning: $noteEarning, noteAllDescendants: $noteAllDescendants,
noteMentions: $noteMentions, noteDeposits: $noteDeposits, noteInvites: $noteInvites,
noteJobIndicator: $noteJobIndicator, noteCowboyHat: $noteCowboyHat, hideInvoiceDesc: $hideInvoiceDesc,
hideFromTopUsers: $hideFromTopUsers, hideCowboyHat: $hideCowboyHat, clickToLoadImg: $clickToLoadImg,
wildWestMode: $wildWestMode, greeterMode: $greeterMode, nostrPubkey: $nostrPubkey, nostrRelays: $nostrRelays, hideBookmarks: $hideBookmarks,
- noteForwardedSats: $noteForwardedSats, hideWalletBalance: $hideWalletBalance) {
+ noteForwardedSats: $noteForwardedSats, hideWalletBalance: $hideWalletBalance, hideIsContributor: $hideIsContributor) {
...SettingsFields
}
}
@@ -150,6 +153,7 @@ export const USER_FIELDS = gql`
stacked
since
photoId
+ isContributor
meSubscriptionPosts
meSubscriptionComments
}`
diff --git a/lib/validate.js b/lib/validate.js
index 0062f2f2a..db5e8c28e 100644
--- a/lib/validate.js
+++ b/lib/validate.js
@@ -224,7 +224,8 @@ export const settingsSchema = object({
).max(NOSTR_MAX_RELAY_NUM,
({ max, value }) => `${Math.abs(max - value.length)} too many`),
hideBookmarks: boolean(),
- hideWalletBalance: boolean()
+ hideWalletBalance: boolean(),
+ hideIsContributor: boolean()
})
const warningMessage = 'If I logout, even accidentally, I will never be able to access my account again'
diff --git a/pages/settings.js b/pages/settings.js
index eb8d8de23..872061982 100644
--- a/pages/settings.js
+++ b/pages/settings.js
@@ -23,6 +23,7 @@ import { useShowModal } from '../components/modal'
import { authErrorMessage } from '../components/login'
import { NostrAuth } from '../components/nostr-auth'
import { useToast } from '../components/toast'
+import { useMe } from '../components/me'
export const getServerSideProps = getGetServerSideProps({ query: SETTINGS, authRequired: true })
@@ -32,6 +33,7 @@ function bech32encode (hexString) {
export default function Settings ({ ssrData }) {
const toaster = useToast()
+ const me = useMe()
const [setSettings] = useMutation(SET_SETTINGS, {
update (cache, { data: { setSettings } }) {
cache.modify({
@@ -77,7 +79,8 @@ export default function Settings ({ ssrData }) {
nostrPubkey: settings?.nostrPubkey ? bech32encode(settings.nostrPubkey) : '',
nostrRelays: settings?.nostrRelays?.length ? settings?.nostrRelays : [''],
hideBookmarks: settings?.hideBookmarks,
- hideWalletBalance: settings?.hideWalletBalance
+ hideWalletBalance: settings?.hideWalletBalance,
+ hideIsContributor: settings?.hideIsContributor
}}
schema={settingsSchema}
onSubmit={async ({ tipDefault, nostrPubkey, nostrRelays, ...values }) => {
@@ -241,6 +244,12 @@ export default function Settings ({ ssrData }) {
name='clickToLoadImg'
groupClassName='mb-0'
/>
+ {me.isContributor &&
+ hide that I'm a stacker.news contributor>}
+ name='hideIsContributor'
+ groupClassName='mb-0'
+ />}
hide my bookmarks from other stackers>}
name='hideBookmarks'
diff --git a/prisma/migrations/20230906010648_verified_contributors/migration.sql b/prisma/migrations/20230906010648_verified_contributors/migration.sql
new file mode 100644
index 000000000..3ce4fb00a
--- /dev/null
+++ b/prisma/migrations/20230906010648_verified_contributors/migration.sql
@@ -0,0 +1,2 @@
+-- AlterTable
+ALTER TABLE "users" ADD COLUMN "hideIsContributor" BOOLEAN NOT NULL DEFAULT false;
diff --git a/prisma/schema.prisma b/prisma/schema.prisma
index c3fceb8d9..5f6f3280a 100644
--- a/prisma/schema.prisma
+++ b/prisma/schema.prisma
@@ -90,6 +90,7 @@ model User {
followers UserSubscription[] @relation("follower")
followees UserSubscription[] @relation("followee")
hideWelcomeBanner Boolean @default(false)
+ hideIsContributor Boolean @default(false)
@@index([createdAt], map: "users.created_at_index")
@@index([inviteId], map: "users.inviteId_index")