This application is a mockup of the Drift widget. It's a sandbox that mimics the kind of environment you might see on day 1 as an engineer here.
The widget is the #1 value of Drift. It's our flagship product, the first impression, and the most important thing to keep up and running smoothly.
You are an engineer responsible for improving our widget. There are two important improvements we want to make to message sending in our widget:
- Show the "message status" of the sent messages in the conversation view.
- Build retry logic for failed sends of a message.
Someone who has since left Drift built the widget, leaving you to implement the core logic of these features without their guidance.
Work with your interview team (who can help you to navigate the project) to setup the app, and to complete the challenges below.
- Fork this repo
- Install the react devtools from: https://chrome.google.com/webstore/detail/react-developer-tools/fmkadmapgofadopljbjfkapdkoienihi
- Install the Redux devtools from: https://chrome.google.com/webstore/detail/redux-devtools/lmhkpmbekcpmknklioeibfkpmmfibljd
- Install the dependencies with
yarn install
- Start the application with
yarn start
If it starts correctly, your browser should automatically open to a localhost page with the widget.
- Complete as many of the challenges as you can.
- Submit your work by emailing a link to your repo to [email protected] and [email protected] along with anything you think we should know about your submission.
This is your warm-up. 🔥
✅ Inspect the React tree to view a component's props & state.
✅ Messages appear (going top to bottom) oldest to newest
✅ Messages are grouped in 5 minute intervals
✅ Messages have more information in their "context"
React allows you to create components that you can render as if they are HTML tags.
These components can take attributes from what renders them (props) and manage internal attributes (state).
Here is a quick tutorial for React: https://reactjs.org/tutorial/tutorial.html
Redux allows you to create a global application state (it's vague/broad on purpose).
React-Redux allows you to easily hook your React components to the global Redux store.
Here is a typical example of a component that gets props from the global store:
import React from 'react'
import { connect } from 'react-redux'
const GlobalCounter = (props) => (
<div>
<div>Global count: {props.count}</div>
<button onClick={props.globalIncrement}>Increment</button>
</div>
)
// mapStateToProps is a way to map the global "state" to your local component's "props"
const mapStateToProps = state => {
return {
count: store.globalCounter.count,
}
}
// mapDispatchToProps is a way to map the redux store's "dispatcher" to your local component's "props"
const mapDispatchToProps = dispatch => {
return {
globalIncrement: () => dispatch({ type: 'GLOBAL_INCREMENT' })
}
}
export default connect(mapStateToProps, mapDispatchToProps)(GlobalCounter)
To listen to the dispatcher, you'll need a reducer. Here is an example of a reducer that listens for the "GLOBAL_INCREMENT" action:
// reducer.js
const defaultState = {
count: 0,
}
// this is a "reducer" function, which takes its state and an action and is run every time the
// redux dispatcher receives an action. It takes its subtree of the redux state and must return
// the new value of the state
const globalCountReducer = (state = defaultState, action) => {
switch(action.type) {
case 'GLOBAL_INCREMENT':
return { count: state.count + 1 }
default:
return state
}
}
We use RxJS to do "asynchronous" work (such as API calls) triggered by our redux actions.
You can create what RxJS calls an "epic" to listen and respond to redux actions.
Here is an example:
import { ofType, combineEpics } from 'redux-observable';
import { flatMap, startWith, catchError } from 'rxjs/operators';
import { from, of } from 'rxjs';
import CounterAPI from './api';
// these are complicated, but the idea is that you're "streaming" actions and you need to return actions
// back to the stream
const tellServerThatIncrementHappened = (action$, store$) => action$.pipe(
ofType('GLOBAL_INCREMENT'),
// we use flatMap instead of just map here because the inner function returns an Observable, and
// we want that to be "flattened" back to the stream
flatMap(() => {
// important to note that the epics get called after the reducers, so we have
// the updated count value here
const count = store$.getState().globalCount.count
return from(CounterAPI.incrementTo(count))
.pipe(
// we could alternatively use "map" here instead of flatMap, and remove the
// "of" function call inside this
flatMap((res) => {
// each of these are actions that get dispatched back into the redux store
// and can be listened to by reducers
return of({ type: 'GLOBAL_INCREMENT_SUCCESS', response: res })
}),
catchError((err) => {
return of({ type: 'GLOBAL_INCREMENT_FAILED', response: err })
}),
startWith({ type: 'GLOBAL_INCREMENT_PENDING' }),
)
})
)
export default combineEpics(
tellServerThatIncrementHappened,
)