Test mutation.
This commit is contained in:
parent
ead184c3a9
commit
5476d9fce8
14
README.md
14
README.md
@ -6,17 +6,19 @@ Apollo GraphQL client and server using express.
|
||||
|
||||
### POC
|
||||
|
||||
- [ ] Trigger server-side wire commands (separate express path?)
|
||||
- [ ] Trigger server-side wire commands (mutation or separate express path?)
|
||||
|
||||
### Next
|
||||
|
||||
- [ ] Routes.
|
||||
- [ ] Fix JsonTree (yarn link).
|
||||
- [ ] Client/server API abstraction (error handler, etc.)
|
||||
- [ ] Config routes for services (test).
|
||||
- [ ] Webpack config (remove dynamic config?)
|
||||
|
||||
- [ ] Client/server API abstraction (error handler, etc.)
|
||||
- [ ] Port dashboard API calls (resolve config first).
|
||||
- [ ] Port dashboard react modules with dummy resolvers.
|
||||
|
||||
- [ ] Fix JsonTree (yarn link).
|
||||
- [ ] https://github.com/standard/standardx (JSX)
|
||||
- [ ] Shared config.
|
||||
- [ ] Port dashboard modules with dummy resolvers.
|
||||
|
||||
### Done
|
||||
|
||||
|
@ -29,7 +29,7 @@ This creates the following folders:
|
||||
NOTE: GQL and Production files and exported and may be used by the server.
|
||||
|
||||
```javascript
|
||||
import QUERY_STATUS from '@dxos/console-client/gql/status.graphql';
|
||||
import QUERY_STATUS from '@dxos/console-client/gql/system_status.graphql';
|
||||
import config from '@dxos/console-client/config.json';
|
||||
|
||||
...
|
||||
|
@ -6,7 +6,7 @@
|
||||
app:
|
||||
title: 'Console'
|
||||
org': 'DxOS'
|
||||
theme: 'dark'
|
||||
theme: 'light'
|
||||
website: 'https://dxos.org'
|
||||
publicUrl: '/console'
|
||||
|
||||
|
@ -3,7 +3,7 @@
|
||||
#
|
||||
|
||||
{
|
||||
ipfs {
|
||||
ipfs_status {
|
||||
json
|
||||
}
|
||||
}
|
@ -3,7 +3,7 @@
|
||||
#
|
||||
|
||||
{
|
||||
status {
|
||||
system_status {
|
||||
timestamp
|
||||
version
|
||||
}
|
10
packages/console-client/gql/wns_action.graphql
Normal file
10
packages/console-client/gql/wns_action.graphql
Normal file
@ -0,0 +1,10 @@
|
||||
#
|
||||
# Copyright 2020 DxOS
|
||||
#
|
||||
|
||||
mutation Action($command: String!) {
|
||||
wns_action(command: $command) {
|
||||
timestamp
|
||||
code
|
||||
}
|
||||
}
|
@ -3,7 +3,7 @@
|
||||
#
|
||||
|
||||
{
|
||||
wns {
|
||||
json @client
|
||||
wns_status @client {
|
||||
json
|
||||
}
|
||||
}
|
@ -27,6 +27,7 @@
|
||||
"testEnvironment": "node"
|
||||
},
|
||||
"dependencies": {
|
||||
"@apollo/react-components": "^3.1.5",
|
||||
"@apollo/react-hooks": "^3.1.5",
|
||||
"@babel/runtime": "^7.8.7",
|
||||
"@dxos/gem-core": "^1.0.0-beta.11",
|
||||
|
@ -3,6 +3,7 @@
|
||||
//
|
||||
|
||||
import React, { Component } from 'react';
|
||||
import Typography from '@material-ui/core/Typography';
|
||||
|
||||
/**
|
||||
* Root-level error boundary.
|
||||
@ -34,7 +35,10 @@ class ErrorBoundary extends Component {
|
||||
|
||||
if (error) {
|
||||
return (
|
||||
<div>
|
||||
<Typography>Error</Typography>
|
||||
<pre>{String(error)}</pre>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
|
@ -16,9 +16,16 @@ const useStyles = makeStyles(() => ({
|
||||
}
|
||||
}));
|
||||
|
||||
/**
|
||||
* Remove Apollo __typename directive.
|
||||
* @param {Object} data
|
||||
* @returns {Object}
|
||||
*/
|
||||
const removeTypename = data => transform(data, (result, value, key) => {
|
||||
result[key] = isObject(value) && '__typename' in value ? omit(value, '__typename') : value;
|
||||
});
|
||||
if (key !== '__typename') {
|
||||
result[key] = isObject(value) ? ('__typename' in value ? omit(value, '__typename') : value) : value;
|
||||
}
|
||||
}, {});
|
||||
|
||||
const Json = ({ data }) => {
|
||||
const classes = useStyles();
|
||||
|
@ -17,7 +17,8 @@ const useStyles = makeStyles((theme) => ({
|
||||
root: {
|
||||
display: 'flex',
|
||||
flexDirection: 'column',
|
||||
flex: 1
|
||||
flex: 1,
|
||||
overflow: 'hidden'
|
||||
},
|
||||
container: {
|
||||
display: 'flex',
|
||||
@ -25,19 +26,19 @@ const useStyles = makeStyles((theme) => ({
|
||||
flex: 1,
|
||||
overflow: 'hidden'
|
||||
},
|
||||
sidebar: {
|
||||
display: 'flex',
|
||||
flexDirection: 'column',
|
||||
flexShrink: 0,
|
||||
width: 180,
|
||||
borderRight: `1px solid ${theme.palette.primary.dark}`
|
||||
},
|
||||
main: {
|
||||
display: 'flex',
|
||||
flex: 1,
|
||||
overflow: 'hidden'
|
||||
},
|
||||
cooter: {
|
||||
sidebar: {
|
||||
display: 'flex',
|
||||
flexDirection: 'column',
|
||||
flexShrink: 0,
|
||||
width: 200,
|
||||
borderRight: `1px solid ${theme.palette.primary.dark}`
|
||||
},
|
||||
footer: {
|
||||
display: 'flex',
|
||||
flexShrink: 0
|
||||
}
|
||||
|
@ -2,25 +2,23 @@
|
||||
// Copyright 2020 DxOS
|
||||
//
|
||||
|
||||
import React, { useContext } from 'react';
|
||||
import React from 'react';
|
||||
import { useQuery } from '@apollo/react-hooks';
|
||||
|
||||
import Json from '../components/Json';
|
||||
|
||||
import { ConsoleContext, useQueryStatusReducer } from '../hooks';
|
||||
import { useQueryStatusReducer } from '../hooks';
|
||||
|
||||
import QUERY from '../../gql/ipfs.graphql';
|
||||
import IPFS_STATUS from '../../gql/ipfs_status.graphql';
|
||||
|
||||
const IPFS = () => {
|
||||
const { config } = useContext(ConsoleContext);
|
||||
const data = useQueryStatusReducer(useQuery(QUERY, { pollInterval: config.api.pollInterval }));
|
||||
const data = useQueryStatusReducer(useQuery(IPFS_STATUS));
|
||||
if (!data) {
|
||||
return null;
|
||||
}
|
||||
|
||||
// TODO(burdon): Return structured GraphQL.
|
||||
return (
|
||||
<Json data={{ ipfs: JSON.parse(data.ipfs.json) }} />
|
||||
<Json data={JSON.parse(data.ipfs_status.json)} />
|
||||
);
|
||||
};
|
||||
|
||||
|
@ -9,17 +9,17 @@ import Json from '../components/Json';
|
||||
|
||||
import { ConsoleContext, useQueryStatusReducer } from '../hooks';
|
||||
|
||||
import QUERY from '../../gql/status.graphql';
|
||||
import SYSTEM_STATUS from '../../gql/system_status.graphql';
|
||||
|
||||
const Status = () => {
|
||||
const { config } = useContext(ConsoleContext);
|
||||
const data = useQueryStatusReducer(useQuery(QUERY, { pollInterval: config.api.pollInterval }));
|
||||
const data = useQueryStatusReducer(useQuery(SYSTEM_STATUS, { pollInterval: config.api.pollInterval }));
|
||||
if (!data) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return (
|
||||
<Json data={data} />
|
||||
<Json data={data.system_status} />
|
||||
);
|
||||
};
|
||||
|
||||
|
@ -4,13 +4,16 @@
|
||||
|
||||
import React, { useContext } from 'react';
|
||||
import { useQuery } from '@apollo/react-hooks';
|
||||
import { Mutation } from '@apollo/react-components';
|
||||
import { makeStyles } from '@material-ui/core';
|
||||
import Button from '@material-ui/core/Button';
|
||||
|
||||
import Json from '../components/Json';
|
||||
|
||||
import { ConsoleContext, useQueryStatusReducer } from '../hooks';
|
||||
|
||||
import QUERY from '../../gql/wns.graphql';
|
||||
import WNS_STATUS from '../../gql/wns_status.graphql';
|
||||
import WNS_ACTION from '../../gql/wns_action.graphql';
|
||||
|
||||
const useStyles = makeStyles((theme) => ({
|
||||
root: {
|
||||
@ -24,20 +27,30 @@ const useStyles = makeStyles((theme) => ({
|
||||
const WNS = () => {
|
||||
const classes = useStyles();
|
||||
const { config } = useContext(ConsoleContext);
|
||||
const data = useQueryStatusReducer(useQuery(QUERY, { pollInterval: config.api.pollInterval }));
|
||||
const data = useQueryStatusReducer(useQuery(WNS_STATUS, { pollInterval: config.api.pollInterval }));
|
||||
if (!data) {
|
||||
return null;
|
||||
}
|
||||
|
||||
// TODO(burdon): peers causes issues.
|
||||
// Warning: Failed prop type: Invalid prop `children` supplied to `ForwardRef(Typography)`, expected a ReactNode.
|
||||
const d = JSON.parse(data.wns.json);
|
||||
d.peers = [];
|
||||
|
||||
// TODO(burdon): Return structured GraphQL.
|
||||
return (
|
||||
<div className={classes.root}>
|
||||
<Json data={{ wns: d }} />
|
||||
<Mutation mutation={WNS_ACTION}>
|
||||
{(action, { data }) => (
|
||||
<div>
|
||||
<Button
|
||||
onClick={() => {
|
||||
action({ variables: { command: 'test' } });
|
||||
}}
|
||||
>
|
||||
Test
|
||||
</Button>
|
||||
|
||||
<pre>Result: {JSON.stringify(data)}</pre>
|
||||
</div>
|
||||
)}
|
||||
</Mutation>
|
||||
|
||||
<Json data={JSON.parse(data.wns_status.json)} />
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
@ -21,7 +21,7 @@ export const createResolvers = config => {
|
||||
|
||||
return {
|
||||
Query: {
|
||||
wns: async () => {
|
||||
wns_status: async () => {
|
||||
log('Querying WNS...');
|
||||
|
||||
const status = await registry.getStatus();
|
||||
|
@ -1,7 +1,7 @@
|
||||
{
|
||||
"build": {
|
||||
"name": "@dxos/console-client",
|
||||
"buildDate": "2020-05-24T02:00:10.452Z",
|
||||
"buildDate": "2020-05-24T13:13:49.317Z",
|
||||
"version": "1.0.0-beta.0"
|
||||
}
|
||||
}
|
||||
|
@ -6,17 +6,36 @@ type JSONResult {
|
||||
json: String!
|
||||
}
|
||||
|
||||
type Log {
|
||||
log: [String]!
|
||||
}
|
||||
|
||||
type Status {
|
||||
timestamp: String!
|
||||
version: String!
|
||||
}
|
||||
|
||||
type Result {
|
||||
timestamp: String!
|
||||
code: Int!
|
||||
}
|
||||
|
||||
#
|
||||
# Schema
|
||||
#
|
||||
|
||||
type Query {
|
||||
status: Status
|
||||
ipfs: JSONResult
|
||||
wns: JSONResult
|
||||
system_status: Status
|
||||
ipfs_status: JSONResult
|
||||
wns_status: JSONResult
|
||||
wns_log: Log
|
||||
}
|
||||
|
||||
type Mutation {
|
||||
wns_action(command: String!): Result
|
||||
}
|
||||
|
||||
schema {
|
||||
mutation: Mutation
|
||||
query: Query
|
||||
}
|
||||
|
@ -10,9 +10,9 @@ import yaml from 'js-yaml';
|
||||
import { ApolloServer, gql } from 'apollo-server-express';
|
||||
import { print } from 'graphql/language';
|
||||
|
||||
import QUERY_STATUS from '@dxos/console-client/gql/status.graphql';
|
||||
import SYSTEM_STATUS from '@dxos/console-client/gql/system_status.graphql';
|
||||
|
||||
import { resolvers } from './resolvers';
|
||||
import { createResolvers } from './resolvers';
|
||||
|
||||
import SCHEMA from './gql/api.graphql';
|
||||
|
||||
@ -60,7 +60,8 @@ app.get(`${publicUrl}(/:filePath)?`, (req, res) => {
|
||||
|
||||
const server = new ApolloServer({
|
||||
typeDefs: SCHEMA,
|
||||
resolvers,
|
||||
|
||||
resolvers: createResolvers(config),
|
||||
|
||||
// https://www.apollographql.com/docs/apollo-server/testing/graphql-playground
|
||||
// https://github.com/prisma-labs/graphql-playground#usage
|
||||
@ -73,7 +74,7 @@ const server = new ApolloServer({
|
||||
{
|
||||
name: 'Status',
|
||||
endpoint: config.api.path,
|
||||
query: print(gql(QUERY_STATUS))
|
||||
query: print(gql(SYSTEM_STATUS))
|
||||
}
|
||||
]
|
||||
}
|
||||
|
@ -13,13 +13,31 @@ const log = debug('dxos:console:server:resolvers');
|
||||
// Resolvers
|
||||
//
|
||||
|
||||
export const resolvers = {
|
||||
const timestamp = () => new Date().toUTCString();
|
||||
|
||||
export const createResolvers = config => ({
|
||||
Mutation: {
|
||||
//
|
||||
// WNS
|
||||
//
|
||||
|
||||
wns_action: async (_, __, { action }) => {
|
||||
log(`WNS action: ${action}`);
|
||||
|
||||
return {
|
||||
timestamp: timestamp(),
|
||||
code: 0
|
||||
};
|
||||
}
|
||||
},
|
||||
|
||||
Query: {
|
||||
//
|
||||
// Status
|
||||
// System
|
||||
//
|
||||
status: () => ({
|
||||
timestamp: new Date().toUTCString(),
|
||||
|
||||
system_status: () => ({
|
||||
timestamp: timestamp(),
|
||||
version
|
||||
}),
|
||||
|
||||
@ -29,15 +47,20 @@ export const resolvers = {
|
||||
// https://github.com/ipfs/js-ipfs
|
||||
// https://github.com/ipfs/js-ipfs/tree/master/packages/ipfs-http-client#api
|
||||
//
|
||||
ipfs: async () => {
|
||||
|
||||
ipfs_status: async () => {
|
||||
log('Calling IPFS...');
|
||||
|
||||
// TODO(burdon): Config.
|
||||
// NOTE: Hangs if server not running.
|
||||
const ipfs = new IpfsHttpClient('/ip4/127.0.0.1/tcp/5001');
|
||||
|
||||
const version = await ipfs.version();
|
||||
const status = await ipfs.id();
|
||||
|
||||
console.log(version);
|
||||
log('Done');
|
||||
|
||||
return {
|
||||
json: JSON.stringify({
|
||||
version,
|
||||
@ -46,4 +69,4 @@ export const resolvers = {
|
||||
};
|
||||
}
|
||||
}
|
||||
};
|
||||
});
|
||||
|
11
yarn.lock
11
yarn.lock
@ -29,6 +29,17 @@
|
||||
ts-invariant "^0.4.4"
|
||||
tslib "^1.10.0"
|
||||
|
||||
"@apollo/react-components@^3.1.5":
|
||||
version "3.1.5"
|
||||
resolved "https://registry.yarnpkg.com/@apollo/react-components/-/react-components-3.1.5.tgz#040d2f35ce4947747efe16f76d59dcbd797ffdaf"
|
||||
integrity sha512-c82VyUuE9VBnJB7bnX+3dmwpIPMhyjMwyoSLyQWPHxz8jK4ak30XszJtqFf4eC4hwvvLYa+Ou6X73Q8V8e2/jg==
|
||||
dependencies:
|
||||
"@apollo/react-common" "^3.1.4"
|
||||
"@apollo/react-hooks" "^3.1.5"
|
||||
prop-types "^15.7.2"
|
||||
ts-invariant "^0.4.4"
|
||||
tslib "^1.10.0"
|
||||
|
||||
"@apollo/react-hooks@^3.1.5":
|
||||
version "3.1.5"
|
||||
resolved "https://registry.yarnpkg.com/@apollo/react-hooks/-/react-hooks-3.1.5.tgz#7e710be52461255ae7fc0b3b9c2ece64299c10e6"
|
||||
|
Loading…
Reference in New Issue
Block a user