Skip to content

Commit

Permalink
feat(app): Add robot settings toggles to Advanced Settings card
Browse files Browse the repository at this point in the history
Closes #1632
  • Loading branch information
mcous authored and IanLondon committed Jul 2, 2018
1 parent 40b0288 commit f312a1b
Show file tree
Hide file tree
Showing 6 changed files with 573 additions and 15 deletions.
93 changes: 79 additions & 14 deletions app/src/components/RobotSettings/AdvancedSettingsCard.js
Original file line number Diff line number Diff line change
@@ -1,24 +1,89 @@
// @flow
// app info card with version and updated
import * as React from 'react'
import {connect} from 'react-redux'

import {Card} from '@opentrons/components'
import {LabeledButton} from '../controls'
import type {State, Dispatch} from '../../types'
import type {Robot} from '../../robot'
import type {Setting} from '../../http-api-client'
import {fetchSettings, setSettings, makeGetRobotSettings} from '../../http-api-client'

import {RefreshCard} from '@opentrons/components'
import {LabeledButton, LabeledToggle} from '../controls'

type OP = Robot

type SP = {settings: Array<Setting>}

type DP = {
fetch: () => mixed,
set: (id: string, value: boolean) => mixed,
}

type Props = OP & SP & DP

type BooleanSettingProps = {
id: string,
title: string,
description: string,
value: boolean,
set: (id: string, value: boolean) => mixed,
}

const TITLE = 'Advanced Settings'

export default function AdvancedSettingsCard () {
export default connect(
makeMapStateToProps,
mapDispatchToProps
)(AdvancedSettingsCard)

class BooleanSettingToggle extends React.Component<BooleanSettingProps> {
toggle = (value) => this.props.set(this.props.id, !this.props.value)

render () {
return (
<LabeledToggle
label={this.props.title}
onClick={this.toggle}
toggledOn={this.props.value}
>
<p>{this.props.description}</p>
</LabeledToggle>
)
}
}

function AdvancedSettingsCard (props: Props) {
const {name, settings, set, fetch} = props

return (
<Card title={TITLE} column>
<LabeledButton
label='Download Logs'
buttonProps={{
disabled: true,
children: 'Download'
}}
>
<p>Access logs from this robot.</p>
</LabeledButton>
</Card>
<RefreshCard watch={name} refresh={fetch} title={TITLE} column>
{settings.map(s => (
<BooleanSettingToggle {...s} key={s.id} set={set} />
))}
<LabeledButton
label='Download Logs'
buttonProps={{
disabled: true,
children: 'Download'
}}
>
<p>Access logs from this robot.</p>
</LabeledButton>
</RefreshCard>
)
}

function makeMapStateToProps (): (state: State, ownProps: OP) => SP {
const getRobotSettings = makeGetRobotSettings()

return (state, ownProps) =>
getRobotSettings(state, ownProps).response || {settings: []}
}

function mapDispatchToProps (dispatch: Dispatch, ownProps: OP): DP {
return {
fetch: () => dispatch(fetchSettings(ownProps)),
set: (id, value) => dispatch(setSettings(ownProps, id, value))
}
}
249 changes: 249 additions & 0 deletions app/src/http-api-client/__tests__/settings.test.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,249 @@
// http api /settings tests
import configureMockStore from 'redux-mock-store'
import thunk from 'redux-thunk'

import client from '../client'
import {
reducer,
fetchSettings,
setSettings,
makeGetRobotSettings
} from '..'

jest.mock('../client')

const middlewares = [thunk]
const mockStore = configureMockStore(middlewares)

const NAME = 'opentrons-dev'

describe('/settings', () => {
let robot
let state
let store

beforeEach(() => {
client.__clearMock()

robot = {name: NAME, ip: '1.2.3.4', port: '1234'}
state = {api: {settings: {}}}
store = mockStore(state)
})

describe('fetchSettings action creator', () => {
const path = 'settings'
const response = {
settings: [{id: 'i', title: 't', description: 'd', value: true}]
}

test('calls GET /settings', () => {
client.__setMockResponse(response)

return store.dispatch(fetchSettings(robot))
.then(() =>
expect(client).toHaveBeenCalledWith(robot, 'GET', 'settings'))
})

test('dispatches api:REQUEST and api:SUCCESS', () => {
const request = null
const expectedActions = [
{type: 'api:REQUEST', payload: {robot, request, path}},
{type: 'api:SUCCESS', payload: {robot, response, path}}
]

client.__setMockResponse(response)

return store.dispatch(fetchSettings(robot))
.then(() => expect(store.getActions()).toEqual(expectedActions))
})

test('dispatches api:REQUEST and api:FAILURE', () => {
const request = null
const error = {name: 'ResponseError', status: 500, message: ''}
const expectedActions = [
{type: 'api:REQUEST', payload: {robot, request, path}},
{type: 'api:FAILURE', payload: {robot, error, path}}
]

client.__setMockError(error)

return store.dispatch(fetchSettings(robot))
.then(() => expect(store.getActions()).toEqual(expectedActions))
})
})

describe('setSettings action creator', () => {
const path = 'settings'
const response = {
settings: [{id: 'i', title: 't', description: 'd', value: true}]
}

test('calls GET /settings', () => {
const request = {id: 'i', value: true}

client.__setMockResponse(response)

return store.dispatch(setSettings(robot, 'i', true))
.then(() =>
expect(client).toHaveBeenCalledWith(robot, 'POST', 'settings', request))
})

test('dispatches api:REQUEST and api:SUCCESS', () => {
const request = {id: 'i', value: true}
const expectedActions = [
{type: 'api:REQUEST', payload: {robot, request, path}},
{type: 'api:SUCCESS', payload: {robot, response, path}}
]

client.__setMockResponse(response)

return store.dispatch(setSettings(robot, 'i', true))
.then(() => expect(store.getActions()).toEqual(expectedActions))
})

test('dispatches api:REQUEST and api:FAILURE', () => {
const request = {id: 'i', value: true}
const error = {name: 'ResponseError', status: 500, message: ''}
const expectedActions = [
{type: 'api:REQUEST', payload: {robot, request, path}},
{type: 'api:FAILURE', payload: {robot, error, path}}
]

client.__setMockError(error)

return store.dispatch(setSettings(robot, 'i', true))
.then(() => expect(store.getActions()).toEqual(expectedActions))
})
})

describe('reducer', () => {
beforeEach(() => {
state = state.api
})

const REDUCER_REQUEST_RESPONSE_TESTS = [
{
method: 'GET',
path: 'settings',
request: null,
response: {
settings: [{id: 'i', title: 't', description: 'd', value: true}]
}
},
{
method: 'POST',
path: 'settings',
request: {id: 'i', value: false},
response: {
settings: [{id: 'i', title: 't', description: 'd', value: false}]
}
}
]

REDUCER_REQUEST_RESPONSE_TESTS.forEach((spec) => {
const {method, path, request, response} = spec

describe(`reducer with ${method} /${path}`, () => {
test('handles api:REQUEST', () => {
const action = {
type: 'api:REQUEST',
payload: {path, robot, request}
}

expect(reducer(state, action).settings).toEqual({
[NAME]: {
[path]: {
request,
inProgress: true,
error: null
}
}
})
})

test('handles api:SUCCESS', () => {
const action = {
type: 'api:SUCCESS',
payload: {path, robot, response}
}

state.settings[NAME] = {
[path]: {
request,
inProgress: true,
error: null,
response: null
}
}

expect(reducer(state, action).settings).toEqual({
[NAME]: {
[path]: {
request,
response,
inProgress: false,
error: null
}
}
})
})

test('handles api:FAILURE', () => {
const error = {message: 'we did not do it!'}
const action = {
type: 'api:FAILURE',
payload: {path, robot, error}
}

state.settings[NAME] = {
[path]: {
request,
inProgress: true,
error: null,
response: null
}
}

expect(reducer(state, action).settings).toEqual({
[NAME]: {
[path]: {
request,
error,
response: null,
inProgress: false
}
}
})
})
})
})
})

describe('selectors', () => {
beforeEach(() => {
state.api.settings[NAME] = {
settings: {inProgress: true}
}
})

test('makeGetRobotSettings', () => {
const getSettings = makeGetRobotSettings()

expect(getSettings(state, robot))
.toEqual(state.api.settings[NAME].settings)

expect(getSettings(state, {name: 'foo'})).toEqual({inProgress: false})
})

test('makeGetRobotSettings with bad response', () => {
const getSettings = makeGetRobotSettings()

state.api.settings[NAME].settings.response = {foo: 'bar'}

expect(getSettings(state, robot)).toEqual({
...state.api.settings[NAME].settings,
response: {settings: []}
})
})
})
})
Loading

0 comments on commit f312a1b

Please sign in to comment.