Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
B
baas-ide
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
JIRA
JIRA
Merge Requests
1
Merge Requests
1
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Registry
Registry
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
guxukai
baas-ide
Commits
f1fa6370
Unverified
Commit
f1fa6370
authored
Jul 06, 2018
by
yann300
Committed by
GitHub
Jul 06, 2018
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #1402 from ethereum/pluginAPIWrapper
Plugin api wrapper
parents
bc3f5337
83540ade
Hide whitespace changes
Inline
Side-by-side
Showing
10 changed files
with
402 additions
and
6 deletions
+402
-6
index.html
extensions/etherscan-general/index.html
+43
-0
index.js
extensions/etherscan-general/index.js
+109
-0
index.js
src/app/plugin/index.js
+55
-0
package.json
src/app/plugin/package.json
+71
-0
pluginAPI.js
src/app/plugin/pluginAPI.js
+6
-0
plugins.js
src/app/plugin/plugins.js
+6
-0
settings-tab.js
src/app/tabs/settings-tab.js
+2
-5
index-raw.html
test-browser/plugin/index-raw.html
+49
-0
index.html
test-browser/plugin/index.html
+5
-1
plugin.js
test-browser/plugin/plugin.js
+56
-0
No files found.
extensions/etherscan-general/index.html
0 → 100644
View file @
f1fa6370
<!DOCTYPE html>
<html>
<head>
<meta
charset=
"utf-8"
>
<!--
The MIT License (MIT)
Copyright (c) 2014, 2015, the individual contributors
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
-->
<meta
http-equiv=
"X-UA-Compatible"
content=
"chrome=1"
>
<title></title>
<link
rel=
"stylesheet"
href=
"assets/css/pygment_trac.css"
>
<link
rel=
"stylesheet"
href=
"assets/css/font-awesome.min.css"
>
<script
type=
"text/javascript"
src=
"api.js"
></script>
<script
type=
"text/javascript"
src=
"index.js"
></script>
<meta
name=
"viewport"
content=
"width=device-width, initial-scale=1, user-scalable=no"
>
</head>
<body>
<div>
Etherscan - Network Status
</div>
<div
id=
"network"
></div>
<button
id=
"updateBtn"
>
Update
</button>
<div
id=
"container"
></div>
</body>
</html>
extensions/etherscan-general/index.js
0 → 100644
View file @
f1fa6370
/* global XMLHttpRequest */
var
remix
=
new
window
.
RemixExtension
()
var
container
var
updateBtn
var
network
var
currentNetWork
=
''
var
networks
=
{
'Main'
:
''
,
'Ropsten'
:
'ropsten'
,
'Kovan'
:
'kovan'
,
'Rinkeby'
:
'rinkeby'
}
function
load
()
{
container
=
document
.
getElementById
(
'container'
)
updateBtn
=
document
.
getElementById
(
'updateBtn'
)
network
=
document
.
getElementById
(
'network'
)
var
log
=
function
(
call
,
msg
)
{
container
.
innerHTML
+=
'<div>'
+
call
+
': '
+
msg
+
'</div>'
}
updateBtn
.
addEventListener
(
'click'
,
function
()
{
container
.
innerHTML
=
''
if
(
networks
[
currentNetWork
]
!==
undefined
)
{
getBlockNumber
(
log
)
getLatestBlockInfo
(
log
)
getGasPrice
(
log
)
}
else
{
container
.
innerHTML
=
'current network not available through etherscan API'
}
})
getToken
(
function
(
error
,
result
)
{
if
(
error
)
console
.
log
(
error
)
if
(
!
result
)
{
remix
.
call
(
'config'
,
'setConfig'
,
[
'config.json'
,
'{ apikey: "" }'
],
function
(
error
,
result
)
{
if
(
error
)
return
console
.
log
(
error
)
})
}
})
setInterval
(
function
()
{
remix
.
call
(
'app'
,
'detectNetWork'
,
[],
function
(
error
,
result
)
{
if
(
error
)
console
.
log
(
error
)
if
(
network
.
innerHTML
!==
result
[
0
].
name
+
' - '
+
result
[
0
].
id
)
{
currentNetWork
=
result
[
0
].
name
container
.
innerHTML
=
''
network
.
innerHTML
=
result
[
0
].
name
+
' - '
+
result
[
0
].
id
}
})
},
1000
)
}
function
getToken
(
callback
)
{
remix
.
call
(
'config'
,
'getConfig'
,
[
'config.json'
],
function
(
error
,
result
)
{
if
(
error
)
return
callback
(
error
)
if
(
result
[
0
])
{
try
{
result
=
JSON
.
parse
(
result
[
0
])
}
catch
(
e
)
{
return
callback
(
e
.
message
)
}
callback
(
null
,
result
.
apikey
)
}
else
{
container
.
innerHTML
=
'no api key found'
callback
(
'no api key found'
)
}
})
}
function
httpGetAsync
(
url
,
callback
)
{
var
xmlHttp
=
new
XMLHttpRequest
()
xmlHttp
.
onreadystatechange
=
function
()
{
if
(
xmlHttp
.
readyState
===
4
&&
xmlHttp
.
status
===
200
)
{
callback
(
xmlHttp
.
responseText
)
}
}
xmlHttp
.
open
(
'GET'
,
url
,
true
)
xmlHttp
.
send
(
null
)
}
function
getBlockNumber
(
callback
)
{
getToken
(
function
(
error
,
apikey
)
{
if
(
error
)
console
.
log
(
error
)
httpGetAsync
(
'https://api-'
+
networks
[
currentNetWork
]
+
'.etherscan.io/api?module=proxy&action=eth_blockNumber&apikey='
+
apikey
,
function
(
result
)
{
callback
(
'latest block number'
,
result
)
})
})
}
function
getLatestBlockInfo
(
callback
)
{
getToken
(
function
(
error
,
apikey
)
{
if
(
error
)
console
.
log
(
error
)
httpGetAsync
(
'https://api-'
+
networks
[
currentNetWork
]
+
'.etherscan.io/api?module=proxy&action=eth_getBlockByNumber&tag=latest&boolean=true&apikey='
+
apikey
,
function
(
result
)
{
callback
(
'latest block'
,
result
)
})
})
}
function
getGasPrice
(
callback
)
{
getToken
(
function
(
error
,
apikey
)
{
if
(
error
)
console
.
log
(
error
)
httpGetAsync
(
'https://api-'
+
networks
[
currentNetWork
]
+
'.etherscan.io/api?module=proxy&action=eth_gasPrice&apikey='
+
apikey
,
function
(
result
)
{
callback
(
'current gas price'
,
result
)
})
})
}
setTimeout
(
load
,
1000
)
src/app/plugin/index.js
0 → 100644
View file @
f1fa6370
'use strict'
class
RemixExtension
{
constructor
()
{
this
.
_notifications
=
{}
this
.
_pendingRequests
=
{}
this
.
_id
=
0
window
.
addEventListener
(
'message'
,
(
event
)
=>
this
.
_newMessage
(
event
),
false
)
}
listen
(
key
,
type
,
callback
)
{
if
(
!
this
.
_notifications
[
key
])
this
.
_notifications
[
key
]
=
{}
this
.
_notifications
[
key
][
type
]
=
callback
}
call
(
key
,
type
,
params
,
callback
)
{
this
.
_id
++
this
.
_pendingRequests
[
this
.
_id
]
=
callback
window
.
parent
.
postMessage
(
JSON
.
stringify
({
action
:
'request'
,
key
,
type
,
value
:
params
,
id
:
this
.
_id
}),
'*'
)
}
_newMessage
(
event
)
{
if
(
!
event
.
data
)
return
if
(
typeof
event
.
data
!==
'string'
)
return
var
msg
try
{
msg
=
JSON
.
parse
(
event
.
data
)
}
catch
(
e
)
{
return
console
.
log
(
'unable to parse data'
)
}
const
{
action
,
key
,
type
,
value
}
=
msg
if
(
action
===
'notification'
)
{
if
(
this
.
_notifications
[
key
]
&&
this
.
_notifications
[
key
][
type
])
{
this
.
_notifications
[
key
][
type
](
value
)
}
}
else
if
(
action
===
'response'
)
{
const
{
id
,
error
}
=
msg
if
(
this
.
_pendingRequests
[
id
])
{
this
.
_pendingRequests
[
id
](
error
,
value
)
delete
this
.
_pendingRequests
[
id
]
}
}
}
}
if
(
window
)
window
.
RemixExtension
=
RemixExtension
if
(
module
&&
module
.
exports
)
module
.
exports
=
RemixExtension
src/app/plugin/package.json
0 → 100644
View file @
f1fa6370
{
"name"
:
"remix-extension"
,
"version"
:
"0.0.1"
,
"description"
:
"Ethereum IDE and tools for the web"
,
"contributors"
:
[
{
"name"
:
"Yann Levreau"
,
"email"
:
"yann@ethdev.com"
}
],
"main"
:
"./index.js"
,
"dependencies"
:
{
"babel-eslint"
:
"^7.1.1"
,
"babel-plugin-transform-object-assign"
:
"^6.22.0"
,
"babel-preset-es2015"
:
"^6.24.0"
,
"babelify"
:
"^7.3.0"
,
"standard"
:
"^7.0.1"
,
"tape"
:
"^4.6.0"
},
"scripts"
:
{
"browserify"
:
"browserify index.js -o bundle.js"
},
"standard"
:
{
"ignore"
:
[
"node_modules/*"
],
"parser"
:
"babel-eslint"
},
"repository"
:
{
"type"
:
"git"
,
"url"
:
"git+https://github.com/ethereum/remix-ide.git"
},
"author"
:
"cpp-ethereum team"
,
"license"
:
"MIT"
,
"bugs"
:
{
"url"
:
"https://github.com/ethereum/remix-ide/issues"
},
"homepage"
:
"https://github.com/ethereum/remix-ide#readme"
,
"browserify"
:
{
"transform"
:
[
[
"babelify"
,
{
"plugins"
:
[
[
"fast-async"
,
{
"runtimePatten"
:
null
,
"compiler"
:
{
"promises"
:
true
,
"es7"
:
true
,
"noRuntime"
:
true
,
"wrapAwait"
:
true
}
}
],
"transform-object-assign"
]
}
],
[
"babelify"
,
{
"presets"
:
[
"es2015"
]
}
]
]
}
}
src/app/plugin/pluginAPI.js
View file @
f1fa6370
...
@@ -12,6 +12,12 @@ module.exports = (fileProviders, compiler, udapp, tabbedMenu) => {
...
@@ -12,6 +12,12 @@ module.exports = (fileProviders, compiler, udapp, tabbedMenu) => {
},
},
updateTitle
:
(
mod
,
title
,
cb
)
=>
{
updateTitle
:
(
mod
,
title
,
cb
)
=>
{
tabbedMenu
.
updateTabTitle
(
mod
,
title
)
tabbedMenu
.
updateTabTitle
(
mod
,
title
)
if
(
cb
)
cb
()
},
detectNetWork
:
(
mod
,
cb
)
=>
{
executionContext
.
detectNetwork
((
error
,
network
)
=>
{
cb
(
error
,
network
)
})
}
}
},
},
config
:
{
config
:
{
...
...
src/app/plugin/plugins.js
View file @
f1fa6370
...
@@ -5,4 +5,10 @@ module.exports = {
...
@@ -5,4 +5,10 @@ module.exports = {
url
:
'https://remix-plugin.oraclize.it'
,
url
:
'https://remix-plugin.oraclize.it'
,
title
:
'Oraclize'
title
:
'Oraclize'
}
}
/*
'etherscan-general': {
url: 'http://127.0.0.1:8081',
title: 'Etherscan-general'
}
*/
}
}
src/app/tabs/settings-tab.js
View file @
f1fa6370
...
@@ -150,15 +150,12 @@ module.exports = class SettingsTab {
...
@@ -150,15 +150,12 @@ module.exports = class SettingsTab {
<div class="
${
css
.
info
}
">
<div class="
${
css
.
info
}
">
<div class=
${
css
.
title
}
>Plugin</div>
<div class=
${
css
.
title
}
>Plugin</div>
<div class="
${
css
.
crowNoFlex
}
">
<div class="
${
css
.
crowNoFlex
}
">
<div class=
${
css
.
attention
}
>
<div><input onclick=
${()
=>
{
onLoadPlugin
(
'oraclize'
)
}}
type
=
"button"
value
=
"Oraclize"
class
=
"${css.pluginLoad}"
><
/div
>
<i title="Do not use this feature yet" class="
${
css
.
icon
}
fa fa-exclamation-triangle" aria-hidden="true"></i>
<
div
><
input
onclick
=
$
{()
=>
{
onLoadPlugin
(
'etherscan-general'
)
}}
type
=
"button"
value
=
"Etherscan-general"
class
=
"${css.pluginLoad}"
><
/div
>
<span> Do not use this alpha feature if you are not sure what you are doing!</span>
</div>
<
div
>
<
div
>
$
{
self
.
_view
.
pluginInput
}
$
{
self
.
_view
.
pluginInput
}
<input onclick=
${
onloadPluginJson
}
type="button" value="Load" class="
${
css
.
pluginLoad
}
">
<input onclick=
${
onloadPluginJson
}
type="button" value="Load" class="
${
css
.
pluginLoad
}
">
</div>
</div>
<input onclick=
${()
=>
{
onLoadPlugin
(
'oraclize'
)
}}
type
=
"button"
value
=
"Oraclize"
class
=
"${css.pluginLoad}"
>
</div>
</div>
</div>`
</div>`
self
.
_view
.
config
.
remixd
=
yo
`
self
.
_view
.
config
.
remixd
=
yo
`
...
...
test-browser/plugin/index-raw.html
0 → 100644
View file @
f1fa6370
<!DOCTYPE html>
<html>
<head>
<meta
charset=
"utf-8"
>
<!--
The MIT License (MIT)
Copyright (c) 2014, 2015, the individual contributors
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
-->
<meta
http-equiv=
"X-UA-Compatible"
content=
"chrome=1"
>
<title></title>
<link
rel=
"stylesheet"
href=
"assets/css/pygment_trac.css"
>
<link
rel=
"stylesheet"
href=
"assets/css/font-awesome.min.css"
>
<script
type=
"text/javascript"
src=
"remix.js"
></script>
<meta
name=
"viewport"
content=
"width=device-width, initial-scale=1, user-scalable=no"
>
</head>
<body>
<div>
PLUGIN
</div>
<input
type=
"text"
id=
"filename"
></input>
<input
type=
"text"
id=
"valuetosend"
></input>
<br>
<input
type=
"button"
id=
"testmessageadd"
>
add config
</input>
<br>
<input
type=
"button"
id=
"testmessageremove"
>
remove config
</input>
<br>
<input
type=
"button"
id=
"testmessagerget"
>
get config
</input>
<br>
<input
type=
"button"
id=
"testcontractcreation"
>
oraclize contract creation
</input>
<br>
<input
type=
"button"
id=
"testaccountcreation"
>
account creation
</input>
<br>
<input
type=
"button"
id=
"testchangetitle"
>
change title
</input>
<br>
<br>
<div
id=
'compilationdata'
></div>
</body>
</html>
test-browser/plugin/index.html
View file @
f1fa6370
...
@@ -30,7 +30,8 @@
...
@@ -30,7 +30,8 @@
<title></title>
<title></title>
<link
rel=
"stylesheet"
href=
"assets/css/pygment_trac.css"
>
<link
rel=
"stylesheet"
href=
"assets/css/pygment_trac.css"
>
<link
rel=
"stylesheet"
href=
"assets/css/font-awesome.min.css"
>
<link
rel=
"stylesheet"
href=
"assets/css/font-awesome.min.css"
>
<script
type=
"text/javascript"
src=
"remix.js"
></script>
<script
type=
"text/javascript"
src=
"bundle.js"
></script>
<script
type=
"text/javascript"
src=
"plugin.js"
></script>
<meta
name=
"viewport"
content=
"width=device-width, initial-scale=1, user-scalable=no"
>
<meta
name=
"viewport"
content=
"width=device-width, initial-scale=1, user-scalable=no"
>
</head>
</head>
<body>
<body>
...
@@ -40,6 +41,9 @@
...
@@ -40,6 +41,9 @@
<input
type=
"button"
id=
"testmessageadd"
>
add config
</input>
<br>
<input
type=
"button"
id=
"testmessageadd"
>
add config
</input>
<br>
<input
type=
"button"
id=
"testmessageremove"
>
remove config
</input>
<br>
<input
type=
"button"
id=
"testmessageremove"
>
remove config
</input>
<br>
<input
type=
"button"
id=
"testmessagerget"
>
get config
</input>
<br>
<input
type=
"button"
id=
"testmessagerget"
>
get config
</input>
<br>
<input
type=
"button"
id=
"testcontractcreation"
>
oraclize contract creation
</input>
<br>
<input
type=
"button"
id=
"testaccountcreation"
>
account creation
</input>
<br>
<input
type=
"button"
id=
"testchangetitle"
>
change title
</input>
<br>
<br>
<br>
<div
id=
'compilationdata'
></div>
<div
id=
'compilationdata'
></div>
</body>
</body>
...
...
test-browser/plugin/plugin.js
0 → 100644
View file @
f1fa6370
/*
test contract creation
*/
var
addrResolverByteCode
=
'0x6060604052341561000f57600080fd5b33600160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555061033c8061005f6000396000f300606060405260043610610062576000357c0100000000000000000000000000000000000000000000000000000000900463ffffffff16806338cc483114610067578063767800de146100bc578063a6f9dae114610111578063d1d80fdf1461014a575b600080fd5b341561007257600080fd5b61007a610183565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b34156100c757600080fd5b6100cf6101ac565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b341561011c57600080fd5b610148600480803573ffffffffffffffffffffffffffffffffffffffff169060200190919050506101d1565b005b341561015557600080fd5b610181600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050610271565b005b60008060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b6000809054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b600160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff1614151561022d57600080fd5b80600160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050565b600160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff161415156102cd57600080fd5b806000806101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550505600a165627a7a723058201b23355f578cb9a23c0a43a440ab2631b62df7be0a8e759812a70f01344224da0029'
const
addrResolverTx
=
{
gasLimit
:
'0x2710'
,
from
:
'0xca35b7d915458ef540ade6068dfe2f44e8fa733c'
,
data
:
addrResolverByteCode
,
value
:
'0x00'
,
useCall
:
false
}
var
extension
=
new
window
.
RemixExtension
()
window
.
onload
=
function
()
{
extension
.
listen
(
'compiler'
,
'compilationFinished'
,
function
()
{
console
.
log
(
arguments
)
})
setInterval
(
function
()
{
extension
.
call
(
'app'
,
'detectNetWork'
,
[],
function
(
error
,
result
)
{
console
.
log
(
error
,
result
)
})
},
5000
)
document
.
querySelector
(
'input#testmessageadd'
).
addEventListener
(
'click'
,
function
()
{
extension
.
call
(
'config'
,
'setConfig'
,
[
document
.
getElementById
(
'filename'
).
value
,
document
.
getElementById
(
'valuetosend'
).
value
],
function
(
error
,
result
)
{
console
.
log
(
error
,
result
)
})
})
document
.
querySelector
(
'input#testmessageremove'
).
addEventListener
(
'click'
,
function
()
{
extension
.
call
(
'config'
,
'removeConfig'
,
[
document
.
getElementById
(
'filename'
).
value
],
function
(
error
,
result
)
{
console
.
log
(
error
,
result
)
})
})
document
.
querySelector
(
'input#testmessagerget'
).
addEventListener
(
'click'
,
function
()
{
extension
.
call
(
'config'
,
'getConfig'
,
[
document
.
getElementById
(
'filename'
).
value
],
function
(
error
,
result
)
{
console
.
log
(
error
,
result
)
})
})
document
.
querySelector
(
'input#testcontractcreation'
).
addEventListener
(
'click'
,
function
()
{
extension
.
call
(
'udapp'
,
'runTx'
,
[
addrResolverTx
],
function
(
error
,
result
)
{
console
.
log
(
error
,
result
)
})
})
document
.
querySelector
(
'input#testaccountcreation'
).
addEventListener
(
'click'
,
function
()
{
extension
.
call
(
'udapp'
,
'createVMAccount'
,
[
'71975fbf7fe448e004ac7ae54cad0a383c3906055a75468714156a07385e96ce'
,
'0x56BC75E2D63100000'
],
function
(
error
,
result
)
{
console
.
log
(
error
,
result
)
})
})
var
k
=
0
document
.
querySelector
(
'input#testchangetitle'
).
addEventListener
(
'click'
,
function
()
{
extension
.
call
(
'app'
,
'updateTitle'
,
[
'changed title '
+
k
++
],
function
(
error
,
result
)
{
console
.
log
(
error
,
result
)
})
})
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment