2017-07-25 13:34:48 +08:00
|
|
|
|
import React, { Component } from 'react'
|
2017-07-25 21:24:12 +08:00
|
|
|
|
import PropTypes from 'prop-types'
|
|
|
|
|
import { connect } from 'react-redux'
|
2017-07-30 11:10:33 +08:00
|
|
|
|
import { Button, Input, Select, Card, Alert, Spin, Icon } from 'antd'
|
2017-07-25 13:34:48 +08:00
|
|
|
|
import { autobind } from 'core-decorators';
|
|
|
|
|
import crossRequest from 'cross-request';
|
2017-07-25 21:24:12 +08:00
|
|
|
|
import { withRouter } from 'react-router';
|
2017-07-30 11:10:33 +08:00
|
|
|
|
import axios from 'axios';
|
2017-07-25 21:24:12 +08:00
|
|
|
|
import URL from 'url';
|
2017-07-25 13:34:48 +08:00
|
|
|
|
|
|
|
|
|
import {
|
|
|
|
|
} from '../../../actions/group.js'
|
|
|
|
|
|
|
|
|
|
import './InterfaceTest.scss'
|
|
|
|
|
|
2017-07-26 14:54:14 +08:00
|
|
|
|
const { TextArea } = Input;
|
2017-07-26 17:53:36 +08:00
|
|
|
|
const InputGroup = Input.Group;
|
|
|
|
|
const Option = Select.Option;
|
2017-07-26 14:54:14 +08:00
|
|
|
|
|
2017-07-25 21:24:12 +08:00
|
|
|
|
@connect(
|
|
|
|
|
state => ({
|
|
|
|
|
reqParams: state.addInterface.reqParams,
|
|
|
|
|
method: state.addInterface.method,
|
|
|
|
|
url: state.addInterface.url,
|
|
|
|
|
seqGroup: state.addInterface.seqGroup,
|
|
|
|
|
interfaceName: state.addInterface.interfaceName,
|
|
|
|
|
interfaceProject: state.addInterface.project
|
|
|
|
|
}),
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
@withRouter
|
2017-07-25 13:34:48 +08:00
|
|
|
|
export default class InterfaceTest extends Component {
|
|
|
|
|
|
|
|
|
|
static propTypes = {
|
2017-07-25 21:24:12 +08:00
|
|
|
|
reqParams: PropTypes.string,
|
|
|
|
|
method: PropTypes.string,
|
|
|
|
|
url: PropTypes.string,
|
|
|
|
|
interfaceName: PropTypes.string,
|
|
|
|
|
seqGroup: PropTypes.array,
|
|
|
|
|
match: PropTypes.object,
|
|
|
|
|
interfaceProject: PropTypes.object
|
2017-07-25 13:34:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
state = {
|
2017-07-26 17:53:36 +08:00
|
|
|
|
res: '',
|
2017-07-26 22:01:30 +08:00
|
|
|
|
method: 'GET',
|
|
|
|
|
domains: [],
|
|
|
|
|
pathname: '',
|
|
|
|
|
query: {},
|
|
|
|
|
params: {},
|
|
|
|
|
paramsNotJson: false,
|
|
|
|
|
headers: {},
|
|
|
|
|
currDomain: ''
|
2017-07-25 13:34:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
constructor(props) {
|
|
|
|
|
super(props)
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-25 21:24:12 +08:00
|
|
|
|
componentWillMount() {
|
2017-07-28 17:28:32 +08:00
|
|
|
|
this.getInterfaceState()
|
2017-07-26 22:01:30 +08:00
|
|
|
|
}
|
2017-07-25 21:24:12 +08:00
|
|
|
|
|
2017-07-26 22:01:30 +08:00
|
|
|
|
componentWillReceiveProps(nextProps) {
|
2017-07-28 17:28:32 +08:00
|
|
|
|
this.getInterfaceState(nextProps)
|
2017-07-25 21:24:12 +08:00
|
|
|
|
}
|
|
|
|
|
|
2017-07-25 13:34:48 +08:00
|
|
|
|
@autobind
|
2017-07-28 17:28:32 +08:00
|
|
|
|
getInterfaceState(nextProps) {
|
2017-07-26 22:01:30 +08:00
|
|
|
|
const props = nextProps || this.props;
|
|
|
|
|
const { method, url, seqGroup, interfaceProject } = props;
|
|
|
|
|
const { prd_host, basepath, protocol, env } = interfaceProject;
|
|
|
|
|
const pathname = (basepath + url).replace(/\/+/g, '/');
|
|
|
|
|
|
|
|
|
|
const domains = {prd: protocol + '://' + prd_host};
|
|
|
|
|
env.forEach(item => {
|
|
|
|
|
domains[item.name] = item.domain;
|
|
|
|
|
})
|
2017-07-26 14:54:14 +08:00
|
|
|
|
|
2017-07-30 11:10:33 +08:00
|
|
|
|
const query = [];
|
2017-07-26 22:01:30 +08:00
|
|
|
|
let params = {};
|
|
|
|
|
let reqParams = this.props.reqParams ? this.props.reqParams : '{}';
|
|
|
|
|
let paramsNotJson = false;
|
|
|
|
|
try {
|
|
|
|
|
reqParams = JSON.parse(reqParams)
|
|
|
|
|
paramsNotJson = false;
|
|
|
|
|
} catch (e) {
|
|
|
|
|
paramsNotJson = true;
|
|
|
|
|
}
|
2017-07-26 14:54:14 +08:00
|
|
|
|
if (method === 'GET') {
|
|
|
|
|
Object.keys(reqParams).forEach(key => {
|
2017-07-26 22:01:30 +08:00
|
|
|
|
const value = typeof reqParams[key] === 'object' ? JSON.stringify(reqParams[key]) : reqParams[key].toString();
|
2017-07-30 11:10:33 +08:00
|
|
|
|
query.push({key, value})
|
2017-07-26 14:54:14 +08:00
|
|
|
|
})
|
2017-07-26 22:01:30 +08:00
|
|
|
|
} else if (method === 'POST') {
|
|
|
|
|
params = reqParams;
|
2017-07-26 14:54:14 +08:00
|
|
|
|
}
|
|
|
|
|
|
2017-07-26 17:53:36 +08:00
|
|
|
|
const headers = {}
|
2017-07-26 14:54:14 +08:00
|
|
|
|
seqGroup.forEach((headerItem) => {
|
2017-07-26 17:53:36 +08:00
|
|
|
|
if (headerItem.name) {
|
|
|
|
|
headers[headerItem.name] = headerItem.value;
|
|
|
|
|
}
|
2017-07-26 14:54:14 +08:00
|
|
|
|
})
|
|
|
|
|
|
2017-07-26 22:01:30 +08:00
|
|
|
|
this.setState({
|
2017-07-28 17:28:32 +08:00
|
|
|
|
method,
|
2017-07-26 22:01:30 +08:00
|
|
|
|
domains,
|
|
|
|
|
pathname,
|
|
|
|
|
query,
|
|
|
|
|
params,
|
|
|
|
|
paramsNotJson,
|
|
|
|
|
headers,
|
2017-07-27 17:16:23 +08:00
|
|
|
|
currDomain: domains.prd,
|
|
|
|
|
loading: false
|
2017-07-26 22:01:30 +08:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@autobind
|
2017-07-30 11:10:33 +08:00
|
|
|
|
requestInterface() {
|
2017-07-28 17:28:32 +08:00
|
|
|
|
const { headers, params, currDomain, method, pathname, query } = this.state;
|
2017-07-26 22:01:30 +08:00
|
|
|
|
const urlObj = URL.parse(currDomain);
|
|
|
|
|
|
2017-07-26 14:54:14 +08:00
|
|
|
|
const href = URL.format({
|
2017-07-26 22:01:30 +08:00
|
|
|
|
protocol: urlObj.protocol || 'http',
|
|
|
|
|
host: urlObj.host,
|
|
|
|
|
pathname,
|
2017-07-30 11:10:33 +08:00
|
|
|
|
query: this.getQueryObj(query)
|
2017-07-26 14:54:14 +08:00
|
|
|
|
});
|
|
|
|
|
|
2017-07-27 17:16:23 +08:00
|
|
|
|
this.setState({ loading: true })
|
|
|
|
|
|
2017-07-25 13:34:48 +08:00
|
|
|
|
crossRequest({
|
2017-07-26 14:54:14 +08:00
|
|
|
|
url: href,
|
|
|
|
|
method,
|
|
|
|
|
headers,
|
2017-07-26 22:01:30 +08:00
|
|
|
|
data: params,
|
2017-07-25 13:34:48 +08:00
|
|
|
|
success: (res, header) => {
|
|
|
|
|
console.log(header)
|
2017-07-27 17:34:46 +08:00
|
|
|
|
try {
|
|
|
|
|
res = JSON.parse(res);
|
|
|
|
|
} catch (e) {
|
|
|
|
|
null;
|
|
|
|
|
}
|
2017-07-26 17:22:12 +08:00
|
|
|
|
this.setState({res})
|
2017-07-27 17:16:23 +08:00
|
|
|
|
this.setState({ loading: false })
|
2017-07-27 17:34:46 +08:00
|
|
|
|
},
|
|
|
|
|
error: (err, header) => {
|
|
|
|
|
console.log(header)
|
|
|
|
|
this.setState({res: err || '请求失败'})
|
|
|
|
|
this.setState({ loading: false })
|
2017-07-25 13:34:48 +08:00
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-26 22:01:30 +08:00
|
|
|
|
@autobind
|
|
|
|
|
changeDomain(value) {
|
2017-07-28 17:28:32 +08:00
|
|
|
|
this.setState({ currDomain: value });
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@autobind
|
|
|
|
|
selectDomain(value) {
|
|
|
|
|
this.setState({ currDomain: value });
|
2017-07-26 22:01:30 +08:00
|
|
|
|
}
|
2017-07-25 13:34:48 +08:00
|
|
|
|
|
2017-07-26 22:01:30 +08:00
|
|
|
|
@autobind
|
|
|
|
|
changeHeader(e, key) {
|
|
|
|
|
const headers = JSON.parse(JSON.stringify(this.state.headers));
|
|
|
|
|
headers[key] = e.target.value;
|
|
|
|
|
this.setState({ headers });
|
|
|
|
|
}
|
2017-07-25 21:24:12 +08:00
|
|
|
|
|
2017-07-26 22:01:30 +08:00
|
|
|
|
@autobind
|
2017-07-30 11:10:33 +08:00
|
|
|
|
changeQuery(e, index, isKey) {
|
2017-07-26 22:01:30 +08:00
|
|
|
|
const query = JSON.parse(JSON.stringify(this.state.query));
|
2017-07-30 11:10:33 +08:00
|
|
|
|
const v = e.target.value;
|
|
|
|
|
if (isKey) {
|
|
|
|
|
query[index].key = v;
|
|
|
|
|
} else {
|
|
|
|
|
query[index].value = v;
|
|
|
|
|
}
|
2017-07-26 22:01:30 +08:00
|
|
|
|
this.setState({ query });
|
|
|
|
|
}
|
2017-07-30 11:10:33 +08:00
|
|
|
|
@autobind
|
|
|
|
|
addQuery() {
|
|
|
|
|
const { query } = this.state;
|
|
|
|
|
this.setState({query: query.concat([{key: '', value: ''}])})
|
|
|
|
|
}
|
|
|
|
|
@autobind
|
|
|
|
|
deleteQuery(index) {
|
|
|
|
|
const { query } = this.state;
|
|
|
|
|
this.setState({query: query.filter((item, i) => +index !== +i)});
|
|
|
|
|
}
|
2017-07-26 17:53:36 +08:00
|
|
|
|
|
2017-07-26 22:01:30 +08:00
|
|
|
|
@autobind
|
|
|
|
|
changeParams(e, key) {
|
|
|
|
|
const params = JSON.parse(JSON.stringify(this.state.params));
|
|
|
|
|
params[key] = e.target.value;
|
|
|
|
|
this.setState({ params });
|
|
|
|
|
}
|
2017-07-25 21:24:12 +08:00
|
|
|
|
|
2017-07-28 17:28:32 +08:00
|
|
|
|
@autobind
|
|
|
|
|
changeMethod(value) {
|
|
|
|
|
this.setState({ method: value });
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@autobind
|
|
|
|
|
changePath(e) {
|
|
|
|
|
const path = e.target.value;
|
|
|
|
|
const urlObj = URL.parse(path, true);
|
|
|
|
|
this.setState({
|
|
|
|
|
query: urlObj.query,
|
|
|
|
|
pathname: urlObj.pathname
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-27 10:12:50 +08:00
|
|
|
|
hasCrossRequestPlugin() {
|
|
|
|
|
const dom = document.getElementById('y-request');
|
|
|
|
|
return dom.getAttribute('key') === 'yapi';
|
|
|
|
|
}
|
2017-07-26 17:53:36 +08:00
|
|
|
|
|
2017-07-30 11:10:33 +08:00
|
|
|
|
getQueryObj(query) {
|
|
|
|
|
const queryObj = {};
|
|
|
|
|
query.forEach(item => {
|
|
|
|
|
if (item.key) {
|
|
|
|
|
queryObj[item.key] = item.value || '';
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
return queryObj;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-26 22:01:30 +08:00
|
|
|
|
render () {
|
|
|
|
|
|
2017-07-28 17:28:32 +08:00
|
|
|
|
const { interfaceName } = this.props;
|
|
|
|
|
const { method, domains, pathname, query, headers, params, paramsNotJson, currDomain } = this.state;
|
2017-07-27 10:12:50 +08:00
|
|
|
|
const hasPlugin = this.hasCrossRequestPlugin();
|
2017-07-30 11:10:33 +08:00
|
|
|
|
const search = decodeURIComponent(URL.format({query: this.getQueryObj(query)}));
|
|
|
|
|
|
|
|
|
|
console.log(axios)
|
|
|
|
|
window.axios = axios
|
2017-07-25 21:24:12 +08:00
|
|
|
|
|
2017-07-25 13:34:48 +08:00
|
|
|
|
|
|
|
|
|
return (
|
2017-07-26 14:54:14 +08:00
|
|
|
|
<div className="interface-test">
|
2017-07-27 10:12:50 +08:00
|
|
|
|
<div style={{padding: '0 20%'}}>
|
|
|
|
|
{ hasPlugin ? '' :
|
|
|
|
|
<Alert
|
|
|
|
|
message={
|
|
|
|
|
<div>
|
|
|
|
|
温馨提示:当前正在使用接口测试服务,请安装我们为您免费提供的
|
|
|
|
|
<a
|
|
|
|
|
target="blank"
|
|
|
|
|
href="https://chrome.google.com/webstore/detail/cross-request/cmnlfmgbjmaciiopcgodlhpiklaghbok?hl=en-US"
|
|
|
|
|
>
|
|
|
|
|
测试增强插件 [点击获取]!
|
|
|
|
|
</a>
|
|
|
|
|
</div>
|
|
|
|
|
}
|
|
|
|
|
type="warning"
|
|
|
|
|
/>
|
|
|
|
|
}
|
|
|
|
|
</div>
|
2017-07-26 14:54:14 +08:00
|
|
|
|
<div className="interface-name">{interfaceName}</div>
|
2017-07-28 17:28:32 +08:00
|
|
|
|
|
|
|
|
|
{/* url */}
|
2017-07-26 14:54:14 +08:00
|
|
|
|
<div className="req-part">
|
2017-07-26 17:53:36 +08:00
|
|
|
|
<div className="req-row href">
|
2017-07-27 12:13:18 +08:00
|
|
|
|
<InputGroup compact style={{display: 'inline-block', width: 680, border: 0, background: '#fff', marginBottom: -4}}>
|
2017-07-30 11:10:33 +08:00
|
|
|
|
<Input value="Method" disabled style={{display: 'inline-block', width: 80, border: 0, background: '#fff'}} />
|
|
|
|
|
<Input value="Domain" disabled style={{display: 'inline-block', width: 300, border: 0, background: '#fff'}} />
|
|
|
|
|
<Input value="Basepath + Url + [Query]" disabled style={{display: 'inline-block', width: 300, border: 0, background: '#fff'}} />
|
2017-07-27 12:13:18 +08:00
|
|
|
|
</InputGroup>
|
2017-07-26 17:53:36 +08:00
|
|
|
|
<InputGroup compact style={{display: 'inline-block', width: 680}}>
|
2017-07-28 17:28:32 +08:00
|
|
|
|
<Select value={method} style={{display: 'inline-block', width: 80}} onChange={this.changeMethod} >
|
|
|
|
|
<Option value="GET">GET</Option>
|
|
|
|
|
<Option value="POST">POST</Option>
|
|
|
|
|
</Select>
|
|
|
|
|
<Select value={currDomain} mode="combobox" filterOption={() => true} style={{display: 'inline-block', width: 300}} onChange={this.changeDomain} onSelect={this.selectDomain}>
|
2017-07-26 17:53:36 +08:00
|
|
|
|
{
|
2017-07-28 17:28:32 +08:00
|
|
|
|
Object.keys(domains).map((key, index) => (<Option value={domains[key]} key={index}>{key + ':' + domains[key]}</Option>))
|
2017-07-26 17:53:36 +08:00
|
|
|
|
}
|
|
|
|
|
</Select>
|
2017-07-28 17:28:32 +08:00
|
|
|
|
<Input value={pathname + search} onChange={this.changePath} spellCheck="false" style={{display: 'inline-block', width: 300}} />
|
2017-07-26 17:53:36 +08:00
|
|
|
|
</InputGroup>
|
2017-07-27 17:16:23 +08:00
|
|
|
|
<Button
|
2017-07-30 11:10:33 +08:00
|
|
|
|
onClick={this.requestInterface}
|
2017-07-27 17:16:23 +08:00
|
|
|
|
type="primary"
|
|
|
|
|
style={{marginLeft: 10}}
|
|
|
|
|
loading={this.state.loading}
|
|
|
|
|
>发送</Button>
|
2017-07-27 20:59:54 +08:00
|
|
|
|
<span style={{fontSize: 12, color: 'rgba(0, 0, 0, 0.25)'}}>(请求测试真实接口)</span>
|
2017-07-26 14:54:14 +08:00
|
|
|
|
</div>
|
2017-07-28 17:28:32 +08:00
|
|
|
|
|
2017-07-30 11:10:33 +08:00
|
|
|
|
<Card title="Query" noHovering style={{marginTop: 10}}>
|
|
|
|
|
{
|
|
|
|
|
query.map((item, index) => {
|
|
|
|
|
return (
|
|
|
|
|
<div key={index}>
|
|
|
|
|
<Input value={item.key} onChange={e => this.changeQuery(e, index, true)} style={{display: 'inline-block', width: 200, margin: 10}} />{' = '}
|
|
|
|
|
<Input value={item.value} onChange={e => this.changeQuery(e, index)} style={{display: 'inline-block', width: 200, margin: 10}} />
|
|
|
|
|
<Icon type="close" className="icon-btn" onClick={() => this.deleteQuery(index)} />
|
|
|
|
|
</div>
|
|
|
|
|
)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
<Button type="primary" icon="plus" onClick={this.addQuery}>Add query parameter</Button>
|
2017-07-26 18:24:06 +08:00
|
|
|
|
</Card>
|
2017-07-28 17:28:32 +08:00
|
|
|
|
<Card title="HEADERS" noHovering style={{marginTop: 10}} className={Object.keys(headers).length ? '' : 'hidden'}>
|
|
|
|
|
<div className="req-row headers">
|
2017-07-26 18:24:06 +08:00
|
|
|
|
{
|
2017-07-28 17:28:32 +08:00
|
|
|
|
Object.keys(headers).map((key, index) => {
|
2017-07-26 22:01:30 +08:00
|
|
|
|
return (
|
|
|
|
|
<div key={index}>
|
2017-07-28 17:29:42 +08:00
|
|
|
|
<Input value={key} style={{display: 'inline-block', width: 200, margin: 10}} />{' = '}
|
2017-07-28 17:28:32 +08:00
|
|
|
|
<Input value={headers[key]} onChange={e => this.changeHeader(e, key)} style={{display: 'inline-block', width: 200, margin: 10}} />
|
2017-07-26 22:01:30 +08:00
|
|
|
|
</div>
|
|
|
|
|
)
|
|
|
|
|
})
|
|
|
|
|
}
|
2017-07-30 11:10:33 +08:00
|
|
|
|
<Button type="primary" icon="plus">Add header</Button>
|
2017-07-26 22:01:30 +08:00
|
|
|
|
</div>
|
|
|
|
|
</Card>
|
2017-07-30 11:10:33 +08:00
|
|
|
|
<Card title="Body" noHovering style={{marginTop: 10}} className={method === 'POST' ? '' : 'hidden'}>
|
2017-07-26 22:01:30 +08:00
|
|
|
|
<div className="req-row params">
|
|
|
|
|
{ paramsNotJson ?
|
|
|
|
|
<TextArea
|
|
|
|
|
value={params}
|
|
|
|
|
style={{margin: 10}}
|
|
|
|
|
autosize={{ minRows: 2, maxRows: 6 }}
|
|
|
|
|
></TextArea> :
|
|
|
|
|
Object.keys(params).map((key, index) => {
|
|
|
|
|
const value = typeof params[key] === 'object' ? JSON.stringify(params[key]) : params[key].toString();
|
2017-07-26 18:24:06 +08:00
|
|
|
|
return (
|
|
|
|
|
<div key={index}>
|
2017-07-28 17:29:42 +08:00
|
|
|
|
<Input value={key} style={{display: 'inline-block', width: 200, margin: 10}} />{' = '}
|
2017-07-26 22:01:30 +08:00
|
|
|
|
<Input value={value} onChange={e => this.changeParams(e, key)} style={{display: 'inline-block', width: 200, margin: 10}} />
|
2017-07-26 18:24:06 +08:00
|
|
|
|
</div>
|
|
|
|
|
)
|
|
|
|
|
})
|
|
|
|
|
}
|
2017-07-30 11:10:33 +08:00
|
|
|
|
<Button type="primary" icon="plus">Add form parameter</Button>
|
2017-07-26 18:24:06 +08:00
|
|
|
|
</div>
|
|
|
|
|
</Card>
|
2017-07-25 21:24:12 +08:00
|
|
|
|
</div>
|
2017-07-26 22:01:30 +08:00
|
|
|
|
<Card title="返回结果" noHovering style={{marginTop: 10}}>
|
2017-07-27 17:16:23 +08:00
|
|
|
|
<Spin spinning={this.state.loading}>
|
|
|
|
|
<div className="res-part">
|
2017-07-27 20:59:54 +08:00
|
|
|
|
<div style={{padding: 10}}>
|
2017-07-27 17:16:23 +08:00
|
|
|
|
<TextArea
|
2017-07-27 17:34:46 +08:00
|
|
|
|
value={typeof this.state.res === 'object' ? JSON.stringify(this.state.res, null, 2) : this.state.res.toString()}
|
2017-07-27 17:16:23 +08:00
|
|
|
|
autosize={{ minRows: 2, maxRows: 6 }}
|
|
|
|
|
></TextArea>
|
|
|
|
|
</div>
|
2017-07-26 18:24:06 +08:00
|
|
|
|
</div>
|
2017-07-27 17:16:23 +08:00
|
|
|
|
</Spin>
|
2017-07-26 18:24:06 +08:00
|
|
|
|
</Card>
|
2017-07-25 13:34:48 +08:00
|
|
|
|
</div>
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}
|