yapi/client/components/Postman/Postman.js

721 lines
23 KiB
JavaScript
Raw Normal View History

2017-08-22 09:43:33 +08:00
import React, { Component } from 'react'
import PropTypes from 'prop-types'
2017-08-31 15:18:37 +08:00
import Mock from 'mockjs'
2017-09-08 10:40:28 +08:00
import { Button, Input, Select, Card, Alert, Spin, Icon, Collapse, Tooltip, message } from 'antd'
2017-08-22 09:43:33 +08:00
import { autobind } from 'core-decorators';
2017-09-08 10:40:28 +08:00
import constants from '../../constants/variable.js'
2017-08-24 12:13:44 +08:00
import mockEditor from '../../containers/Project/Interface/InterfaceList/mockEditor'
2017-08-22 09:43:33 +08:00
import URL from 'url';
2017-08-31 18:04:45 +08:00
const MockExtra = require('common/mock-extra.js')
import './Postman.scss';
2017-09-07 20:39:58 +08:00
import json5 from 'json5'
function json_parse(data) {
try {
return json5.parse(data)
} catch (e) {
return data
2017-09-07 20:39:58 +08:00
}
}
2017-08-22 09:43:33 +08:00
2017-09-08 15:21:41 +08:00
function isValidJson(json){
if(!json) return false;
if(typeof json === 'object') return true;
try{
if(typeof json === 'string'){
json5.parse(json);
return true;
}
}catch(e){
return false;
}
}
function isJsonData(headers, res){
if(isValidJson(res)){
return true;
}
if(!headers || typeof headers !== 'object') return false;
let isResJson = false;
Object.keys(headers).map(key => {
if (/content-type/i.test(key) && /application\/json/i.test(headers[key])) {
isResJson = true;
}
})
return isResJson;
}
2017-08-24 21:07:49 +08:00
const { TextArea } = Input;
2017-08-22 09:43:33 +08:00
const InputGroup = Input.Group;
const Option = Select.Option;
const Panel = Collapse.Panel;
2017-09-08 10:40:28 +08:00
const HTTP_METHOD = constants.HTTP_METHOD;
2017-08-22 09:43:33 +08:00
export default class Run extends Component {
static propTypes = {
2017-08-23 11:41:46 +08:00
data: PropTypes.object,
2017-08-24 10:44:02 +08:00
save: PropTypes.func,
saveTip: PropTypes.string,
type: PropTypes.string
2017-08-22 09:43:33 +08:00
}
state = {
2017-08-31 16:00:41 +08:00
res: null,
resHeader: null,
2017-08-22 09:43:33 +08:00
method: 'GET',
domains: [],
pathname: '',
query: [],
bodyForm: [],
headers: [],
2017-08-29 10:08:55 +08:00
caseEnv: '',
2017-08-22 09:43:33 +08:00
bodyType: '',
2017-08-24 17:02:39 +08:00
bodyOther: '',
2017-08-31 15:18:37 +08:00
loading: false,
validRes: null,
hasPlugin: true
2017-08-22 09:43:33 +08:00
}
constructor(props) {
super(props)
}
componentWillMount() {
let startTime = 0;
2017-09-07 20:39:58 +08:00
this.interval = setInterval(() => {
startTime += 500;
2017-09-07 20:39:58 +08:00
if (startTime > 5000) {
clearInterval(this.interval);
}
2017-09-07 20:39:58 +08:00
if (window.crossRequest) {
clearInterval(this.interval);
this.setState({
hasPlugin: true
})
2017-09-07 20:39:58 +08:00
} else {
this.setState({
hasPlugin: false
})
}
}, 500)
2017-08-22 09:43:33 +08:00
this.getInterfaceState()
}
2017-09-07 20:39:58 +08:00
componentWillUnmount() {
clearInterval(this.interval)
}
2017-08-22 09:43:33 +08:00
componentWillReceiveProps(nextProps) {
2017-08-23 14:59:24 +08:00
if (nextProps.data._id !== this.props.data._id) {
this.getInterfaceState(nextProps)
}
2017-08-22 09:43:33 +08:00
}
2017-08-24 17:02:39 +08:00
componentDidMount() {
2017-08-24 17:15:58 +08:00
const { bodyType } = this.state;
2017-09-07 20:39:58 +08:00
if (bodyType && bodyType !== 'file' && bodyType !== 'form') {
2017-08-24 17:15:58 +08:00
this.loadBodyEditor()
}
2017-08-24 17:02:39 +08:00
}
2017-08-22 09:43:33 +08:00
@autobind
getInterfaceState(nextProps) {
const props = nextProps || this.props;
2017-08-24 10:44:02 +08:00
const { data, type } = props;
2017-08-22 09:43:33 +08:00
const {
method = '',
path: url = '',
req_headers = [],
req_body_type,
req_query = [],
req_params = [],
req_body_other = '',
req_body_form = [],
basepath = '',
2017-08-23 11:41:46 +08:00
env = [],
2017-08-29 10:08:55 +08:00
case_env = ''
2017-08-22 09:43:33 +08:00
} = data;
2017-08-24 10:44:02 +08:00
// case 任意编辑 pathname不管项目的 basepath
const pathname = (type === 'inter' ? (basepath + url) : url).replace(/\/+/g, '/');
2017-08-22 09:43:33 +08:00
2017-08-29 10:08:55 +08:00
// let hasContentType = false;
// req_headers.forEach(headerItem => {
// // TODO 'Content-Type' 排除大小写不同格式影响
// if (headerItem.name === 'Content-Type'){
// hasContentType = true;
// headerItem.value = headerItem.value || 'application/x-www-form-urlencoded';
// }
// })
// if (!hasContentType) {
// req_headers.push({name: 'Content-Type', value: 'application/x-www-form-urlencoded'});
// }
// const domains = env.concat();
// if (domain && !env.find(item => item.domain === domain)) {
// domains.push({name: 'default', domain})
// }
2017-08-22 09:43:33 +08:00
this.setState({
method,
2017-08-29 10:08:55 +08:00
domains: env.concat(),
2017-08-22 09:43:33 +08:00
pathParam: req_params.concat(),
pathname,
query: req_query.concat(),
bodyForm: req_body_form.concat(),
headers: req_headers.concat(),
bodyOther: req_body_other,
2017-08-29 10:08:55 +08:00
caseEnv: case_env || (env[0] && env[0].name),
2017-08-22 09:43:33 +08:00
bodyType: req_body_type || 'form',
loading: false
2017-08-24 17:15:58 +08:00
}, () => {
2017-09-07 20:39:58 +08:00
if (req_body_type && req_body_type !== 'file' && req_body_type !== 'form') {
2017-08-24 17:15:58 +08:00
this.loadBodyEditor()
}
2017-08-22 09:43:33 +08:00
});
}
@autobind
reqRealInterface() {
2017-08-29 11:28:28 +08:00
if (this.state.loading) {
this.setState({ loading: false })
2017-09-07 20:39:58 +08:00
return;
2017-08-29 11:28:28 +08:00
}
2017-08-30 14:28:07 +08:00
const { headers, bodyForm, pathParam, bodyOther, caseEnv, domains, method, pathname, query, bodyType } = this.state;
2017-08-29 11:28:28 +08:00
const urlObj = URL.parse(domains.find(item => item.name === caseEnv).domain);
2017-08-30 14:28:07 +08:00
let path = pathname
pathParam.forEach(item => {
path = path.replace(`:${item.name}`, item.value || `:${item.name}`);
});
2017-08-22 09:43:33 +08:00
const href = URL.format({
protocol: urlObj.protocol || 'http',
host: urlObj.host,
2017-09-14 13:53:22 +08:00
pathname: urlObj.pathname? urlObj.pathname + path : path,
2017-08-22 09:43:33 +08:00
query: this.getQueryObj(query)
});
this.setState({ loading: true })
2017-09-07 20:39:58 +08:00
let that = this;
window.crossRequest({
2017-08-22 09:43:33 +08:00
url: href,
method,
headers: this.getHeadersObj(headers),
data: bodyType === 'form' ? this.arrToObj(bodyForm) : bodyOther,
files: bodyType === 'form' ? this.getFiles(bodyForm) : {},
2017-09-09 09:11:08 +08:00
file: bodyType === 'file' ? 'single-file' : null,
2017-08-24 10:44:02 +08:00
success: (res, header) => {
2017-08-24 12:13:44 +08:00
try {
2017-09-08 15:21:41 +08:00
if(isJsonData(header)){
res = json_parse(res);
2017-08-31 16:00:41 +08:00
}
2017-09-07 21:15:53 +08:00
const { res_body, res_body_type } = that.props.data;
let validRes = '';
let query = {};
that.state.query.forEach(item => {
query[item.name] = item.value;
2017-08-31 18:04:45 +08:00
})
2017-09-07 21:15:53 +08:00
let body = {};
if (that.state.bodyType === 'form') {
that.state.bodyForm.forEach(item => {
body[item.name] = item.value;
})
} else if (that.state.bodyType === 'json') {
2017-09-08 15:21:41 +08:00
body = json_parse(that.state.bodyOther);
2017-08-31 18:04:45 +08:00
}
2017-09-07 21:15:53 +08:00
if (res_body && res_body_type === 'json' && typeof res === 'object') {
let tpl = MockExtra(json_parse(res_body), {
query: query,
body: body
})
validRes = Mock.valid(tpl, res)
2017-08-31 18:04:45 +08:00
}
2017-09-07 20:39:58 +08:00
2017-09-07 21:15:53 +08:00
message.success('请求完成')
that.setState({ res, resHeader: header, validRes })
that.setState({ loading: false })
that.bindAceEditor()
} catch (e) {
console.error(e.message)
2017-08-31 18:04:45 +08:00
}
2017-08-22 09:43:33 +08:00
},
2017-08-24 10:44:02 +08:00
error: (err, header) => {
2017-08-24 12:13:44 +08:00
try {
2017-09-08 15:21:41 +08:00
if(isJsonData(header)){
err = json_parse(err);
}
2017-08-24 12:13:44 +08:00
} catch (e) {
message.error(e.message)
}
2017-09-08 15:21:41 +08:00
message.error('请求异常')
2017-09-07 20:39:58 +08:00
that.setState({ res: err || '请求失败', resHeader: header, validRes: null })
that.setState({ loading: false })
that.bindAceEditor()
2017-08-22 09:43:33 +08:00
}
})
}
2017-08-29 11:28:28 +08:00
// @autobind
// changeDomain(value) {
// this.setState({ currDomain: value });
// }
2017-08-22 09:43:33 +08:00
@autobind
selectDomain(value) {
2017-08-29 10:08:55 +08:00
this.setState({ caseEnv: value });
2017-08-22 09:43:33 +08:00
}
@autobind
changeHeader(e, index, isName) {
2017-09-07 20:39:58 +08:00
const headers = json_parse(JSON.stringify(this.state.headers));
2017-08-22 09:43:33 +08:00
const v = e.target.value;
if (isName) {
headers[index].name = v;
} else {
headers[index].value = v;
}
this.setState({ headers });
}
@autobind
addHeader() {
const { headers } = this.state;
2017-09-07 20:39:58 +08:00
this.setState({ headers: headers.concat([{ name: '', value: '' }]) })
2017-08-22 09:43:33 +08:00
}
@autobind
deleteHeader(index) {
const { headers } = this.state;
2017-09-07 20:39:58 +08:00
this.setState({ headers: headers.filter((item, i) => +index !== +i) });
2017-08-22 09:43:33 +08:00
}
@autobind
2017-09-08 15:21:41 +08:00
setContentType() {
// const headersObj = this.getHeadersObj(this.state.headers);
// headersObj['Content-Type'] = type;
// this.setState({ headers: this.objToArr(headersObj) })
2017-08-22 09:43:33 +08:00
}
@autobind
changeQuery(e, index, isKey) {
2017-09-07 20:39:58 +08:00
const query = json_parse(JSON.stringify(this.state.query));
2017-08-22 09:43:33 +08:00
const v = e.target.value;
if (isKey) {
query[index].name = v;
} else {
query[index].value = v;
}
this.setState({ query });
}
@autobind
addQuery() {
const { query } = this.state;
2017-09-07 20:39:58 +08:00
this.setState({ query: query.concat([{ name: '', value: '' }]) })
2017-08-22 09:43:33 +08:00
}
@autobind
deleteQuery(index) {
const { query } = this.state;
2017-09-07 20:39:58 +08:00
this.setState({ query: query.filter((item, i) => +index !== +i) });
2017-08-22 09:43:33 +08:00
}
@autobind
changePathParam(e, index, isKey) {
const pathParam = JSON.parse(JSON.stringify(this.state.pathParam));
const v = e.target.value;
const name = pathParam[index].name;
let newPathname = this.state.pathname;
if (isKey) {
if (!name && v) {
newPathname += `/:${v}`;
} else {
newPathname = newPathname.replace(`/:${name}`, v ? `/:${v}` : '')
}
pathParam[index].name = v;
} else {
pathParam[index].value = v;
}
this.setState({ pathParam, pathname: newPathname });
}
@autobind
addPathParam() {
const { pathParam } = this.state;
2017-09-07 20:39:58 +08:00
this.setState({ pathParam: pathParam.concat([{ name: '', value: '' }]) })
2017-08-22 09:43:33 +08:00
}
@autobind
deletePathParam(index) {
const { pathParam } = this.state;
const name = pathParam[index].name;
const newPathname = this.state.pathname.replace(`/:${name}`, '');
2017-09-07 20:39:58 +08:00
this.setState({ pathParam: pathParam.filter((item, i) => +index !== +i), pathname: newPathname });
2017-08-22 09:43:33 +08:00
}
@autobind
changeBody(e, index, type) {
2017-09-07 20:39:58 +08:00
const bodyForm = json_parse(JSON.stringify(this.state.bodyForm));
2017-08-22 09:43:33 +08:00
switch (type) {
case 'key':
bodyForm[index].name = e.target.value
break;
case 'type':
bodyForm[index].type = e
break;
case 'value':
if (bodyForm[index].type === 'file') {
bodyForm[index].value = e.target.id
} else {
bodyForm[index].value = e.target.value
}
break;
default:
break;
}
if (type === 'type' && e === 'file') {
2017-09-08 15:21:41 +08:00
//this.setContentType('multipart/form-data')
2017-08-22 09:43:33 +08:00
}
this.setState({ bodyForm });
}
@autobind
addBody() {
const { bodyForm } = this.state;
2017-09-07 20:39:58 +08:00
this.setState({ bodyForm: bodyForm.concat([{ name: '', value: '', type: 'text' }]) })
2017-08-22 09:43:33 +08:00
}
@autobind
deleteBody(index) {
const { bodyForm } = this.state;
2017-09-07 20:39:58 +08:00
this.setState({ bodyForm: bodyForm.filter((item, i) => +index !== +i) });
2017-08-22 09:43:33 +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: this.objToArr(urlObj.query),
pathname: urlObj.pathname
})
}
@autobind
changeBodyType(value) {
2017-09-07 20:39:58 +08:00
this.setState({ bodyType: value }, () => {
if (value !== 'file' && value !== 'form') {
2017-08-24 17:02:39 +08:00
this.loadBodyEditor()
}
})
2017-08-22 09:43:33 +08:00
}
// hasCrossRequestPlugin() {
// const dom = document.getElementById('y-request');
// return dom.getAttribute('key') === 'yapi';
// }
2017-08-22 09:43:33 +08:00
objToArr(obj, key, value) {
const keyName = key || 'name';
const valueName = value || 'value';
const arr = []
Object.keys(obj).forEach((_key) => {
if (_key) {
2017-09-07 20:39:58 +08:00
arr.push({ [keyName]: _key, [valueName]: obj[_key] });
2017-08-22 09:43:33 +08:00
}
})
return arr;
}
arrToObj(arr) {
const obj = {};
arr.forEach(item => {
if (item.name && item.type !== 'file') {
obj[item.name] = item.value || '';
}
})
return obj;
}
getFiles(bodyForm) {
const files = {};
bodyForm.forEach(item => {
if (item.name && item.type === 'file') {
files[item.name] = item.value
}
})
return files;
}
getQueryObj(query) {
const queryObj = {};
query.forEach(item => {
if (item.name) {
queryObj[item.name] = item.value || '';
}
})
return queryObj;
}
getHeadersObj(headers) {
const headersObj = {};
headers.forEach(item => {
if (item.name && item.value) {
headersObj[item.name] = item.value;
}
})
return headersObj;
}
2017-08-24 12:13:44 +08:00
bindAceEditor = () => {
mockEditor({
container: 'res-body-pretty',
2017-09-08 15:21:41 +08:00
data: this.state.res,
2017-09-07 20:39:58 +08:00
readOnly: true,
onChange: function () { }
2017-08-24 12:13:44 +08:00
})
mockEditor({
container: 'res-headers-pretty',
2017-09-08 15:21:41 +08:00
data: this.state.resHeader,
2017-09-07 20:39:58 +08:00
readOnly: true,
onChange: function () { }
2017-08-24 12:13:44 +08:00
})
}
2017-08-24 17:02:39 +08:00
loadBodyEditor = () => {
const that = this;
2017-09-07 20:39:58 +08:00
setTimeout(function () {
2017-08-24 17:02:39 +08:00
mockEditor({
container: 'body-other-edit',
data: that.state.bodyOther,
onChange: function (d) {
if (d.format !== true) return false;
that.setState({
bodyOther: d.text
})
}
})
}, 0);
}
2017-08-24 12:13:44 +08:00
2017-08-22 09:43:33 +08:00
@autobind
fileChange(e, index) {
console.log(e)
console.log(index)
}
2017-09-07 20:39:58 +08:00
render() {
2017-09-08 15:21:41 +08:00
const { method, domains, pathParam, pathname, query, headers, bodyForm, caseEnv, bodyType, resHeader, loading, validRes, res } = this.state;
2017-09-08 11:09:17 +08:00
HTTP_METHOD[method] = HTTP_METHOD[method] || {}
const hasPlugin = this.state.hasPlugin;
2017-09-08 15:21:41 +08:00
let isResJson = isJsonData(resHeader, res);
2017-08-22 09:43:33 +08:00
let path = pathname;
pathParam.forEach(item => {
path = path.replace(`:${item.name}`, item.value || `:${item.name}`);
});
2017-09-07 20:39:58 +08:00
const search = decodeURIComponent(URL.format({ query: this.getQueryObj(query) }));
let validResView;
if (!validRes) {
validResView = '请定义返回json'
}
if (Array.isArray(validRes) && validRes.length > 0) {
2017-09-08 10:40:28 +08:00
validResView = validRes.map((item, index) => {
2017-09-07 21:15:53 +08:00
return <div key={index}>{item.message}</div>
2017-09-07 20:39:58 +08:00
})
} else if (Array.isArray(validRes)) {
validResView = <p>验证通过</p>
}
2017-08-22 09:43:33 +08:00
return (
2017-08-24 12:13:44 +08:00
<div className="interface-test postman">
2017-09-07 20:39:58 +08:00
<div className="has-plugin">
{hasPlugin ? '' : <Alert
message={
<div>
{/* 温馨提示:当前正在使用接口测试服务,请安装我们为您免费提供的测试增强插件&nbsp;(该插件可支持任何 chrome 内核的浏览器) */}
重要当前的接口测试服务需安装免费测试增强插件 支持所有 webkit 内核选择下面任意一种安装方式
2017-08-22 09:43:33 +08:00
<div>
2017-09-07 20:39:58 +08:00
<a
target="blank"
href="https://chrome.google.com/webstore/detail/cross-request/cmnlfmgbjmaciiopcgodlhpiklaghbok?hl=en-US"
> [Google 商店获取需翻墙]</a>
2017-08-22 09:43:33 +08:00
</div>
2017-09-07 20:39:58 +08:00
<div>
<a
target="blank"
2017-09-10 21:47:38 +08:00
href="/attachment/cross-request-v2.0.1.zip"
2017-09-07 20:39:58 +08:00
> [手动下载] </a>
<span> zip 文件解压后将 crx 文件拖入到 chrome://extensions/ </span>
<a
target="blank"
href="http://www.jianshu.com/p/12ca04c61fc6"
> [详细安装教程] </a>
</div>
</div>
}
type="warning"
/>
2017-08-22 09:43:33 +08:00
}
</div>
<Card title="请求部分" noHovering className="req-part">
<div className="url">
2017-09-07 20:39:58 +08:00
<InputGroup compact style={{ display: 'flex' }}>
<Select disabled value={method} style={{ flexBasis: 60 }} onChange={this.changeMethod} >
2017-08-22 09:43:33 +08:00
<Option value="GET">GET</Option>
<Option value="POST">POST</Option>
</Select>
2017-09-07 20:39:58 +08:00
<Select value={caseEnv} style={{ flexBasis: 180, flexGrow: 1 }} onSelect={this.selectDomain}>
2017-08-22 09:43:33 +08:00
{
2017-08-29 10:08:55 +08:00
domains.map((item, index) => (<Option value={item.name} key={index}>{item.name + '' + item.domain}</Option>))
2017-08-22 09:43:33 +08:00
}
</Select>
2017-09-07 20:39:58 +08:00
<Input disabled value={path + search} onChange={this.changePath} spellCheck="false" style={{ flexBasis: 180, flexGrow: 1 }} />
2017-08-22 09:43:33 +08:00
</InputGroup>
<Tooltip placement="bottom" title="请求真实接口">
<Button
2017-08-25 11:29:43 +08:00
disabled={!hasPlugin}
2017-08-22 09:43:33 +08:00
onClick={this.reqRealInterface}
type="primary"
2017-09-07 20:39:58 +08:00
style={{ marginLeft: 10 }}
2017-08-29 11:28:28 +08:00
icon={loading ? 'loading' : ''}
>{loading ? '取消' : '发送'}</Button>
2017-08-22 09:43:33 +08:00
</Tooltip>
2017-08-23 15:50:07 +08:00
<Tooltip placement="bottom" title={this.props.saveTip}>
2017-08-22 09:43:33 +08:00
<Button
2017-08-23 11:41:46 +08:00
onClick={this.props.save}
2017-08-22 09:43:33 +08:00
type="primary"
2017-09-07 20:39:58 +08:00
style={{ marginLeft: 10 }}
2017-08-24 10:44:02 +08:00
>{this.props.type === 'inter' ? '保存' : '更新'}</Button>
2017-08-22 09:43:33 +08:00
</Tooltip>
</div>
<Collapse defaultActiveKey={['0', '1', '2', '3']} bordered={true}>
2017-08-28 16:12:57 +08:00
<Panel header="PATH PARAMETERS" key="0" className={pathParam.length === 0 ? 'hidden' : ''}>
2017-08-22 09:43:33 +08:00
{
pathParam.map((item, index) => {
return (
<div key={index} className="key-value-wrap">
2017-08-28 15:47:57 +08:00
<Input disabled value={item.name} onChange={e => this.changePathParam(e, index, true)} className="key" />
2017-08-22 09:43:33 +08:00
<span className="eq-symbol">=</span>
<Input value={item.value} onChange={e => this.changePathParam(e, index)} className="value" />
2017-09-07 20:39:58 +08:00
<Icon style={{ display: 'none' }} type="delete" className="icon-btn" onClick={() => this.deletePathParam(index)} />
2017-08-22 09:43:33 +08:00
</div>
)
})
}
2017-09-07 20:39:58 +08:00
<Button style={{ display: 'none' }} type="primary" icon="plus" onClick={this.addPathParam}>添加Path参数</Button>
2017-08-22 09:43:33 +08:00
</Panel>
2017-08-28 16:12:57 +08:00
<Panel header="QUERY PARAMETERS" key="1" className={query.length === 0 ? 'hidden' : ''}>
2017-08-22 09:43:33 +08:00
{
query.map((item, index) => {
return (
<div key={index} className="key-value-wrap">
2017-08-28 15:47:57 +08:00
<Input disabled value={item.name} onChange={e => this.changeQuery(e, index, true)} className="key" />
2017-08-22 09:43:33 +08:00
<span className="eq-symbol">=</span>
<Input value={item.value} onChange={e => this.changeQuery(e, index)} className="value" />
2017-09-07 20:39:58 +08:00
<Icon style={{ display: 'none' }} type="delete" className="icon-btn" onClick={() => this.deleteQuery(index)} />
2017-08-22 09:43:33 +08:00
</div>
)
})
}
2017-09-07 20:39:58 +08:00
<Button style={{ display: 'none' }} type="primary" icon="plus" onClick={this.addQuery}>添加Query参数</Button>
2017-08-22 09:43:33 +08:00
</Panel>
2017-08-28 16:12:57 +08:00
<Panel header="HEADERS" key="2" className={headers.length === 0 ? 'hidden' : ''}>
2017-08-22 09:43:33 +08:00
{
headers.map((item, index) => {
return (
<div key={index} className="key-value-wrap">
2017-08-28 15:47:57 +08:00
<Input disabled value={item.name} onChange={e => this.changeHeader(e, index, true)} className="key" />
2017-08-22 09:43:33 +08:00
<span className="eq-symbol">=</span>
<Input value={item.value} onChange={e => this.changeHeader(e, index)} className="value" />
2017-09-07 20:39:58 +08:00
<Icon style={{ display: 'none' }} type="delete" className="icon-btn" onClick={() => this.deleteHeader(index)} />
2017-08-22 09:43:33 +08:00
</div>
)
})
}
2017-09-07 20:39:58 +08:00
<Button style={{ display: 'none' }} type="primary" icon="plus" onClick={this.addHeader}>添加Header</Button>
2017-08-22 09:43:33 +08:00
</Panel>
<Panel
header={
2017-09-07 20:39:58 +08:00
<div style={{ display: 'flex', justifyContent: 'space-between' }}>
2017-09-08 10:40:28 +08:00
<div>BODY</div>
2017-08-22 09:43:33 +08:00
</div>
}
key="3"
2017-09-08 10:40:28 +08:00
className={HTTP_METHOD[method].request_body?'POST':'hidden'}
2017-08-22 09:43:33 +08:00
>
2017-09-08 10:40:28 +08:00
<div style={{ display: HTTP_METHOD[method].request_body && bodyType !== 'form' && bodyType !== 'file'? 'block': 'none' }}>
<div id="body-other-edit" style={{ marginTop: 10, minHeight:150 }} className="pretty-editor"></div>
2017-09-08 10:40:28 +08:00
</div>
2017-08-22 09:43:33 +08:00
{
2017-09-08 10:40:28 +08:00
HTTP_METHOD[method].request_body && bodyType === 'form' &&
2017-08-22 09:43:33 +08:00
<div>
2017-09-08 10:40:28 +08:00
{
2017-08-22 09:43:33 +08:00
bodyForm.map((item, index) => {
return (
<div key={index} className="key-value-wrap">
2017-08-28 15:47:57 +08:00
<Input disabled value={item.name} onChange={e => this.changeBody(e, index, 'key')} className="key" />
2017-08-22 09:43:33 +08:00
<span>[</span>
2017-08-28 15:47:57 +08:00
<Select disabled value={item.type} onChange={e => this.changeBody(e, index, 'type')}>
2017-08-22 09:43:33 +08:00
<Option value="file">File</Option>
<Option value="text">Text</Option>
</Select>
<span>]</span>
<span className="eq-symbol">=</span>
2017-09-07 20:39:58 +08:00
{item.type === 'file' ?
<Input type="file" id={'file_' + index} onChange={e => this.changeBody(e, index, 'value')} multiple className="value" /> :
2017-08-22 09:43:33 +08:00
<Input value={item.value} onChange={e => this.changeBody(e, index, 'value')} className="value" />
}
2017-09-07 20:39:58 +08:00
<Icon style={{ display: 'none' }} type="delete" className="icon-btn" onClick={() => this.deleteBody(index)} />
2017-08-22 09:43:33 +08:00
</div>
)
})
}
2017-09-07 20:39:58 +08:00
<Button style={{ display: 'none' }} type="primary" icon="plus" onClick={this.addBody}>添加Form参数</Button>
2017-08-22 09:43:33 +08:00
</div>
}
{
2017-09-09 09:11:08 +08:00
HTTP_METHOD[method].request_body && bodyType === 'file' &&
2017-08-22 09:43:33 +08:00
<div>
2017-09-09 09:11:08 +08:00
<Input type="file" id="single-file"></Input>
2017-08-22 09:43:33 +08:00
</div>
}
2017-08-28 16:12:57 +08:00
{/*
2017-08-22 09:43:33 +08:00
method !== 'POST' &&
<div>GET 请求没有 BODY</div>
2017-08-28 16:12:57 +08:00
*/}
2017-08-22 09:43:33 +08:00
</Panel>
</Collapse>
</Card>
<Card title="返回结果" noHovering className="resp-part">
<Spin spinning={this.state.loading}>
2017-08-24 10:44:02 +08:00
<div className="res-code"></div>
<Collapse defaultActiveKey={['0', '1']} bordered={true}>
<Panel header="BODY" key="0" >
2017-09-07 21:15:53 +08:00
<div id="res-body-pretty" className="pretty-editor-body" style={{ display: isResJson ? '' : 'none' }}></div>
2017-08-24 21:07:49 +08:00
<TextArea
2017-09-07 20:39:58 +08:00
style={{ display: isResJson ? 'none' : '' }}
2017-08-31 16:00:41 +08:00
value={this.state.res && this.state.res.toString()}
2017-09-07 21:15:53 +08:00
autosize={{ minRows: 10, maxRows: 20 }}
2017-08-31 15:18:37 +08:00
></TextArea>
2017-09-08 10:40:28 +08:00
<h3 style={{ marginTop: '15px', display: isResJson ? '' : 'none' }}>返回 Body 验证结果</h3>
2017-09-07 21:15:53 +08:00
<div style={{ display: isResJson ? '' : 'none' }}>
2017-09-07 20:39:58 +08:00
{validResView}
2017-09-07 21:15:53 +08:00
</div>
2017-08-24 10:44:02 +08:00
</Panel>
<Panel header="HEADERS" key="1" >
2017-08-24 12:13:44 +08:00
{/*<TextArea
2017-08-24 10:44:02 +08:00
value={typeof this.state.resHeader === 'object' ? JSON.stringify(this.state.resHeader, null, 2) : this.state.resHeader.toString()}
autosize={{ minRows: 2, maxRows: 10 }}
2017-08-24 12:13:44 +08:00
></TextArea>*/}
2017-09-07 21:15:53 +08:00
<div id="res-headers-pretty" className="pretty-editor-header"></div>
2017-08-24 10:44:02 +08:00
</Panel>
</Collapse>
2017-08-22 09:43:33 +08:00
</Spin>
</Card>
</div>
)
}
}