Skip to main content

realtime data with GraphQL subscriptions / apollo-server

using Apollo server:




1. server.js


const {
ApolloServer,
gql,
AuthenticationError,
PubSub
} = require('apollo-server')

const pubsub = new PubSub()


const server = new ApolloServer({
typeDefs,
resolvers,
formatError: error => {
console.log(JSON.stringify(error))
return { ...error }
},

introspection: true,
playground: true,
context: async ({ req, connection }) => {
if (connection) {
return { pubsub }
}

if (req) {
const me = await getMe(req)

return {
me,
secret: process.env.SECRET,
pubsub
}
}
}
})


2. schema: project.js

extend type Subscription {
projectAdded: Project
}
3. resolver: project.js

const PROJECT_CHANNEL = 'PROJECT_CHANNEL'
Mutation: {
addProject: combineResolvers(
isAuthenticated,
async (parent, { projectInput }, { models, me, pubsub }) => {
const { projectName, clientId } = projectInput
const project = await Project.create({
accountId: me.accountId,
projectName: projectName || 'default project',
clientId: clientId || 0
})

console.log(me.accountId)
if (me.accountId === 'sat')
pubsub.publish('PROJECT_CHANNEL', { projectAdded: project })

return project
}
),
Subscription: {
projectAdded: {
subscribe: (parent, args, { pubsub }) => {
return pubsub.asyncIterator(PROJECT_CHANNEL)
}
}
}
4. front end App.js

const wsLink = new WebSocketLink({
uri: REACT_APP_WSURI,
options: {
reconnect: true
}
})
const uploadLink = createUploadLink({ uri: REACT_APP_GRAPHQL_URL })

const terminatingLink = split(
({ query }) => {
const { kind, operation } = getMainDefinition(query)
return kind === 'OperationDefinition' && operation === 'subscription'
},
wsLink,
uploadLink
)

const link = ApolloLink.from([authLink, terminatingLink])

5. Front end: dashboard.js

const { loading, error, data, refetch, subscribeToMore } = useQuery(
GET_PROJECTS
)

<ProjectList
styles={styles}
projects={projects}
navigation={navigation}
onRefresh={onRefresh}
refreshing={refreshing}
onEditProject={onEditProject}
onDeleteProject={onDeleteProject}
subscribeToNewProject={() =>
subscribeToMore({
document: ADD_PROJECT_SUBSCRIPTION,
variables: {},
updateQuery: (prev, { subscriptionData }) => {
if (!subscriptionData.data) return prev
const newProject = subscriptionData.data.projectAdded
console.log(newProject)
return Object.assign({}, prev, {
projects: [newProject, ...prev.projects]
})
}
})
}
/>

6.  project-list.js

componentDidMount () {
this.props.subscribeToNewProject()
}


Comments

Popular posts from this blog

for loop in javascript - promise - .eslintrc for "for of"

the vast majority of cases  map ,  forEach ,  find  etc. can be used.  async function printFiles () { const files = await getFilePaths(); await Promise.all(files. map (async (file) => { const contents = await fs.readFile(file, 'utf8') console.log(contents) })); } const inventory = [ { name : 'apples' , quantity : 2 } , { name : 'bananas' , quantity : 0 } , { name : 'cherries' , quantity : 5 } ] ; const result = inventory . find ( ( { name } ) => name === 'cherries' ) ;   function getFirstMatching(array) { for (let item of array) { const result = heavyTransform(item); if (result) { return result; } } } Specifically this shuts down the whole no-restricted-syntax. If you want to cherry-pick, here is the current definition: 'no-restricted-syntax' : [ 'error' , { selector : 'ForInStatement' , message...

fs.writeFile - permission issue on GCP (Google Cloud Run)

error on this: fs . writeFile ( fileName , pdfBytes , err => { if ( err ) { console . log ( err . message ) reject ( err . message ) } else resolve ( fileName ) }) #1 It looks like when deployed into Cloud Run it also requires the extra permission "Service Account Token Creator" to run  getSignedUrl . Locally for some reason this role is not required. #2 Only the directory  /tmp  is writable in Cloud Run. So, change the default write location to write into this directory. However, you have to be aware of 2 things: Cloud Run is stateless, that means when a new instance is created, the container start from scratch, with an empty  /tmp  directory /tmp  directory is an in-memory file system. The maximum allowed memory on Cloud Run is 2Gb, your app memory footprint included. In addition of your file and Airflow, not sure that you will have a lot of space. A final remark. Cloud Run is active only when it...

add new site to Vagrant Homestead

Homestead.yaml: folders : - map : ~/ freshinup to : / home / vagrant / code sites : - map : smartmotors . local . com to : / home / vagrant / code / smartmotors / public php : "7.3" - map : arbor . test to : / home / vagrant / code / arbor / public php : "7.3" /etc/hosts: 192.168.10.10 arbor.test 192.168.10.10 smartmotors.local.com vagrant up --provision php artisan smartmotors:seed --quickstart