Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
M
MetaProtocol
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Nebula
MetaProtocol
Commits
c5023e6c
Commit
c5023e6c
authored
Nov 30, 2022
by
Ubuntu
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
fix any type
parent
f2a034dd
Changes
4
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
290 additions
and
15 deletions
+290
-15
resource.proto
baseapi/base/v1/resource.proto
+7
-1
buf.yaml
baseapi/buf.yaml
+3
-0
grcp-tx_test.go
grcp-tx_test.go
+279
-14
main.go
main.go
+1
-0
No files found.
baseapi/base/v1/resource.proto
View file @
c5023e6c
...
@@ -4,6 +4,12 @@ package base.v1;
...
@@ -4,6 +4,12 @@ package base.v1;
import
"google/protobuf/timestamp.proto"
;
import
"google/protobuf/timestamp.proto"
;
import
"google/protobuf/any.proto"
;
import
"google/protobuf/any.proto"
;
import
"google/protobuf/descriptor.proto"
;
import
"github.com/gogo/protobuf/gogoproto/gogo.proto"
;
// import "base/v1/options.proto";
// import "base/v1/options.proto";
message
Bytes32
{
message
Bytes32
{
...
@@ -86,7 +92,7 @@ message TxProof{
...
@@ -86,7 +92,7 @@ message TxProof{
message
Transaction
{
message
Transaction
{
TxProof
tx_proof
=
1
;
TxProof
tx_proof
=
1
;
int64
timeout_block_num
=
2
;
int64
timeout_block_num
=
2
;
google.protobuf.Any
tx
=
3
;
// EthTx StdTx
google.protobuf.Any
tx
=
3
[(
gogoproto.customtype
)
=
"InterfaceType"
]
;
// EthTx StdTx
}
}
//2. eth std: proto3 eth tx --> grpc --->proto3 eth tx
//2. eth std: proto3 eth tx --> grpc --->proto3 eth tx
...
...
baseapi/buf.yaml
View file @
c5023e6c
version
:
v1
version
:
v1
deps
:
-
buf.build/googleapis/googleapis
-
buf.build/acme/paymentapis
breaking
:
breaking
:
use
:
use
:
-
FILE
-
FILE
...
...
grcp-tx_test.go
View file @
c5023e6c
...
@@ -14,7 +14,6 @@ import (
...
@@ -14,7 +14,6 @@ import (
"google.golang.org/grpc"
"google.golang.org/grpc"
"google.golang.org/grpc/credentials/insecure"
"google.golang.org/grpc/credentials/insecure"
"google.golang.org/protobuf/proto"
"google.golang.org/protobuf/types/known/anypb"
"google.golang.org/protobuf/types/known/anypb"
base
"github.com/CaduceusMetaverseProtocol/metaprotocol/gen/proto/go/base/v1"
base
"github.com/CaduceusMetaverseProtocol/metaprotocol/gen/proto/go/base/v1"
...
@@ -34,17 +33,100 @@ type RingServer struct {
...
@@ -34,17 +33,100 @@ type RingServer struct {
ring
.
UnimplementedRingServiceServer
ring
.
UnimplementedRingServiceServer
}
}
func
(
*
RingServer
)
SendTxAsEth
(
ctx
context
.
Context
,
req
*
base
.
TransactionEth
)
(
*
ring
.
SendRawTransactionResponse
,
error
)
{
func
(
*
RingServer
)
SendTxAsEth
(
ctx
context
.
Context
,
in
*
base
.
TransactionEth
)
(
*
ring
.
SendRawTransactionResponse
,
error
)
{
return
&
ring
.
SendRawTransactionResponse
{},
nil
//fmt.Println(in.Tx.Inner.Recipient)
addr
:=
common
.
Address
{}
copy
(
addr
[
:
],
in
.
Tx
.
Inner
.
Recipient
.
Address
[
:
common
.
AddressLength
])
ethTx
:=
types
.
NewTx
(
&
types
.
LegacyTx
{
Nonce
:
in
.
Tx
.
Inner
.
AccountNonce
,
To
:
&
addr
,
Value
:
new
(
big
.
Int
)
.
SetBytes
(
in
.
Tx
.
Inner
.
Amount
),
Gas
:
in
.
Tx
.
Inner
.
GasLimit
,
GasPrice
:
new
(
big
.
Int
)
.
SetBytes
(
in
.
Tx
.
Inner
.
Price
),
Data
:
in
.
Tx
.
Inner
.
Payload
,
V
:
new
(
big
.
Int
)
.
SetBytes
(
in
.
Tx
.
Inner
.
V
),
R
:
new
(
big
.
Int
)
.
SetBytes
(
in
.
Tx
.
Inner
.
R
),
S
:
new
(
big
.
Int
)
.
SetBytes
(
in
.
Tx
.
Inner
.
S
),
})
return
&
ring
.
SendRawTransactionResponse
{
TxHash
:
ethTx
.
Hash
()
.
Bytes
()},
nil
}
}
func
(
*
RingServer
)
SendTxAsStd
(
ctx
context
.
Context
,
req
*
base
.
TransactionStd
)
(
*
ring
.
SendRawTransactionResponse
,
error
)
{
func
(
*
RingServer
)
SendTxAsStd
(
ctx
context
.
Context
,
in
*
base
.
TransactionStd
)
(
*
ring
.
SendRawTransactionResponse
,
error
)
{
return
&
ring
.
SendRawTransactionResponse
{},
nil
addr
:=
common
.
Address
{}
copy
(
addr
[
:
],
in
.
Tx
.
Inner
.
Recipient
.
Address
[
:
common
.
AddressLength
])
ethTx
:=
types
.
NewTx
(
&
types
.
LegacyTx
{
Nonce
:
in
.
Tx
.
Inner
.
AccountNonce
,
To
:
&
addr
,
Value
:
new
(
big
.
Int
)
.
SetBytes
(
in
.
Tx
.
Inner
.
Amount
),
Gas
:
in
.
Tx
.
Inner
.
GasLimit
,
GasPrice
:
new
(
big
.
Int
)
.
SetBytes
(
in
.
Tx
.
Inner
.
Price
),
Data
:
in
.
Tx
.
Inner
.
Payload
,
V
:
new
(
big
.
Int
)
.
SetBytes
(
in
.
Tx
.
Inner
.
V
),
R
:
new
(
big
.
Int
)
.
SetBytes
(
in
.
Tx
.
Inner
.
R
),
S
:
new
(
big
.
Int
)
.
SetBytes
(
in
.
Tx
.
Inner
.
S
),
})
return
&
ring
.
SendRawTransactionResponse
{
TxHash
:
ethTx
.
Hash
()
.
Bytes
()},
nil
}
}
func
(
*
RingServer
)
SendTxAsAny
(
ctx
context
.
Context
,
req
*
base
.
Transaction
)
(
*
ring
.
SendRawTransactionResponse
,
error
)
{
func
(
*
RingServer
)
SendTxAsAny
(
ctx
context
.
Context
,
in
*
base
.
Transaction
)
(
*
ring
.
SendRawTransactionResponse
,
error
)
{
msg
,
err
:=
in
.
Tx
.
UnmarshalNew
()
if
err
!=
nil
{
return
nil
,
err
}
switch
m
:=
msg
.
(
type
)
{
case
*
base
.
EthTx
:
addr
:=
common
.
Address
{}
copy
(
addr
[
:
],
m
.
Inner
.
Recipient
.
Address
[
:
common
.
AddressLength
])
ethTx
:=
types
.
NewTx
(
&
types
.
LegacyTx
{
Nonce
:
m
.
Inner
.
AccountNonce
,
To
:
&
addr
,
Value
:
new
(
big
.
Int
)
.
SetBytes
(
m
.
Inner
.
Amount
),
Gas
:
m
.
Inner
.
GasLimit
,
GasPrice
:
new
(
big
.
Int
)
.
SetBytes
(
m
.
Inner
.
Price
),
Data
:
m
.
Inner
.
Payload
,
V
:
new
(
big
.
Int
)
.
SetBytes
(
m
.
Inner
.
V
),
R
:
new
(
big
.
Int
)
.
SetBytes
(
m
.
Inner
.
R
),
S
:
new
(
big
.
Int
)
.
SetBytes
(
m
.
Inner
.
S
),
})
return
&
ring
.
SendRawTransactionResponse
{
TxHash
:
ethTx
.
Hash
()
.
Bytes
()},
nil
case
*
base
.
StdTx
:
addr
:=
common
.
Address
{}
copy
(
addr
[
:
],
m
.
Inner
.
Recipient
.
Address
[
:
common
.
AddressLength
])
ethTx
:=
types
.
NewTx
(
&
types
.
LegacyTx
{
Nonce
:
m
.
Inner
.
AccountNonce
,
To
:
&
addr
,
Value
:
new
(
big
.
Int
)
.
SetBytes
(
m
.
Inner
.
Amount
),
Gas
:
m
.
Inner
.
GasLimit
,
GasPrice
:
new
(
big
.
Int
)
.
SetBytes
(
m
.
Inner
.
Price
),
Data
:
m
.
Inner
.
Payload
,
V
:
new
(
big
.
Int
)
.
SetBytes
(
m
.
Inner
.
V
),
R
:
new
(
big
.
Int
)
.
SetBytes
(
m
.
Inner
.
R
),
S
:
new
(
big
.
Int
)
.
SetBytes
(
m
.
Inner
.
S
),
})
return
&
ring
.
SendRawTransactionResponse
{
TxHash
:
ethTx
.
Hash
()
.
Bytes
()},
nil
}
return
&
ring
.
SendRawTransactionResponse
{},
nil
return
&
ring
.
SendRawTransactionResponse
{},
nil
}
}
...
@@ -77,6 +159,11 @@ func pricedTransaction(to common.Address, nonce uint64, gaslimit uint64, gaspric
...
@@ -77,6 +159,11 @@ func pricedTransaction(to common.Address, nonce uint64, gaslimit uint64, gaspric
}
}
// go test -v -run EthTx -bench=. -benchtime=3s
// go test -v -run EthTx -bench=. -benchtime=3s
// go test -v -run BenchmarkEthTx -bench BenchmarkEthTx -benchtime=3s
// go test -v -run BenchmarkStdTx -bench BenchmarkStdTx -benchtime=3s
// go test -v -run BenchmarkAnyTx -bench BenchmarkAnyTx -benchtime=3s
// go test -v -run BenchmarkBytesEth -bench BenchmarkBytesEth -benchtime=3s
// go test -v -run TestGrpcServer -timeout 0
// go test -v -run TestGrpcServer -timeout 0
//BenchmarkAny
//BenchmarkAny
...
@@ -96,22 +183,46 @@ func int() {
...
@@ -96,22 +183,46 @@ func int() {
publicKey
:=
local
.
Public
()
publicKey
:=
local
.
Public
()
publicKeyECDSA
,
_
:=
publicKey
.
(
*
ecdsa
.
PublicKey
)
publicKeyECDSA
,
_
:=
publicKey
.
(
*
ecdsa
.
PublicKey
)
fromAddr
=
crypto
.
PubkeyToAddress
(
*
publicKeyECDSA
)
fromAddr
=
crypto
.
PubkeyToAddress
(
*
publicKeyECDSA
)
//b.Log(fromAddress)
remote
,
_
:=
crypto
.
GenerateKey
()
remote
,
_
:=
crypto
.
GenerateKey
()
tx
=
pricedTransaction
(
crypto
.
PubkeyToAddress
(
remote
.
PublicKey
),
0
,
100000
,
big
.
NewInt
(
1
),
local
)
tx
=
pricedTransaction
(
crypto
.
PubkeyToAddress
(
remote
.
PublicKey
),
0
,
100000
,
big
.
NewInt
(
1
),
local
)
}
}
var
count
int64
var
countParallel
int64
func
BenchmarkEthTx
(
b
*
testing
.
B
)
{
func
BenchmarkEthTx
(
b
*
testing
.
B
)
{
// count++
// defer fmt.Println("defer countParallel", countParallel)
// defer fmt.Println("defer count", count)
// defer fmt.Println("defer b.N", b.N)
onceFunc
:=
func
()
{
//fmt.Println("once b.N", b.N)
local
,
_
:=
crypto
.
HexToECDSA
(
"FD5CC6F5E7E2805E920AC5DC83D5AF1106F9C92F0C04F9D5E1FD4261B4B4464A"
)
publicKey
:=
local
.
Public
()
publicKeyECDSA
,
_
:=
publicKey
.
(
*
ecdsa
.
PublicKey
)
fromAddr
=
crypto
.
PubkeyToAddress
(
*
publicKeyECDSA
)
remote
,
_
:=
crypto
.
GenerateKey
()
tx
=
pricedTransaction
(
crypto
.
PubkeyToAddress
(
remote
.
PublicKey
),
0
,
100000
,
big
.
NewInt
(
1
),
local
)
}
once
.
Do
(
onceFunc
)
b
.
ReportAllocs
()
b
.
ReportAllocs
()
b
.
RunParallel
(
func
(
pb
*
testing
.
PB
)
{
b
.
RunParallel
(
func
(
pb
*
testing
.
PB
)
{
//countParallel++
for
pb
.
Next
()
{
for
pb
.
Next
()
{
//for i := 0; i < b.N; i++ { //串行
conn
,
err
:=
grpc
.
Dial
(
"127.0.0.1:9006"
,
grpc
.
WithTransportCredentials
(
insecure
.
NewCredentials
()))
conn
,
err
:=
grpc
.
Dial
(
"127.0.0.1:9006"
,
grpc
.
WithTransportCredentials
(
insecure
.
NewCredentials
()))
if
err
!=
nil
{
if
err
!=
nil
{
b
.
Fatal
(
err
)
b
.
Fatal
(
err
)
...
@@ -150,12 +261,28 @@ func BenchmarkEthTx(b *testing.B) {
...
@@ -150,12 +261,28 @@ func BenchmarkEthTx(b *testing.B) {
_
=
res
_
=
res
//fmt.Printf("%x \n", res.TxHash)
}
}
})
})
//}
}
}
func
BenchmarkStdTx
(
b
*
testing
.
B
)
{
func
BenchmarkStdTx
(
b
*
testing
.
B
)
{
onceFunc
:=
func
()
{
local
,
_
:=
crypto
.
HexToECDSA
(
"FD5CC6F5E7E2805E920AC5DC83D5AF1106F9C92F0C04F9D5E1FD4261B4B4464A"
)
publicKey
:=
local
.
Public
()
publicKeyECDSA
,
_
:=
publicKey
.
(
*
ecdsa
.
PublicKey
)
fromAddr
=
crypto
.
PubkeyToAddress
(
*
publicKeyECDSA
)
remote
,
_
:=
crypto
.
GenerateKey
()
tx
=
pricedTransaction
(
crypto
.
PubkeyToAddress
(
remote
.
PublicKey
),
0
,
100000
,
big
.
NewInt
(
1
),
local
)
}
once
.
Do
(
onceFunc
)
b
.
ReportAllocs
()
b
.
ReportAllocs
()
b
.
RunParallel
(
func
(
pb
*
testing
.
PB
)
{
b
.
RunParallel
(
func
(
pb
*
testing
.
PB
)
{
...
@@ -199,6 +326,7 @@ func BenchmarkStdTx(b *testing.B) {
...
@@ -199,6 +326,7 @@ func BenchmarkStdTx(b *testing.B) {
}
}
_
=
res
_
=
res
//fmt.Printf("%x \n", res.TxHash)
}
}
})
})
...
@@ -206,6 +334,18 @@ func BenchmarkStdTx(b *testing.B) {
...
@@ -206,6 +334,18 @@ func BenchmarkStdTx(b *testing.B) {
func
BenchmarkAnyTx
(
b
*
testing
.
B
)
{
func
BenchmarkAnyTx
(
b
*
testing
.
B
)
{
onceFunc
:=
func
()
{
local
,
_
:=
crypto
.
HexToECDSA
(
"FD5CC6F5E7E2805E920AC5DC83D5AF1106F9C92F0C04F9D5E1FD4261B4B4464A"
)
publicKey
:=
local
.
Public
()
publicKeyECDSA
,
_
:=
publicKey
.
(
*
ecdsa
.
PublicKey
)
fromAddr
=
crypto
.
PubkeyToAddress
(
*
publicKeyECDSA
)
remote
,
_
:=
crypto
.
GenerateKey
()
tx
=
pricedTransaction
(
crypto
.
PubkeyToAddress
(
remote
.
PublicKey
),
0
,
100000
,
big
.
NewInt
(
1
),
local
)
}
once
.
Do
(
onceFunc
)
b
.
ReportAllocs
()
b
.
ReportAllocs
()
// The loop body is executed b.N times total across all goroutines.
// The loop body is executed b.N times total across all goroutines.
...
@@ -245,7 +385,7 @@ func BenchmarkAnyTx(b *testing.B) {
...
@@ -245,7 +385,7 @@ func BenchmarkAnyTx(b *testing.B) {
ethTx
:=
base
.
EthTx
{
Inner
:
&
inner
}
ethTx
:=
base
.
EthTx
{
Inner
:
&
inner
}
ethTxAsAny
,
err
:=
pbany
(
ethTx
)
ethTxAsAny
,
err
:=
anypb
.
New
(
&
ethTx
)
res
,
err
:=
c
.
SendTxAsAny
(
ctx
,
&
base
.
Transaction
{
Tx
:
ethTxAsAny
})
res
,
err
:=
c
.
SendTxAsAny
(
ctx
,
&
base
.
Transaction
{
Tx
:
ethTxAsAny
})
...
@@ -259,16 +399,141 @@ func BenchmarkAnyTx(b *testing.B) {
...
@@ -259,16 +399,141 @@ func BenchmarkAnyTx(b *testing.B) {
})
})
}
}
func
pbany
(
v
interface
{})
(
*
anypb
.
Any
,
error
)
{
func
TestType
(
t
*
testing
.
T
)
{
pv
,
ok
:=
v
.
(
proto
.
Message
)
if
!
ok
{
ethTx
:=
base
.
EthTx
{}
return
&
anypb
.
Any
{},
fmt
.
Errorf
(
"%v is not proto.Message"
,
pv
)
fmt
.
Println
(
ethTx
.
ProtoReflect
()
.
Descriptor
()
.
FullName
())
}
func
TestAnyTx
(
t
*
testing
.
T
)
{
onceFunc
:=
func
()
{
local
,
_
:=
crypto
.
HexToECDSA
(
"FD5CC6F5E7E2805E920AC5DC83D5AF1106F9C92F0C04F9D5E1FD4261B4B4464A"
)
publicKey
:=
local
.
Public
()
publicKeyECDSA
,
_
:=
publicKey
.
(
*
ecdsa
.
PublicKey
)
fromAddr
=
crypto
.
PubkeyToAddress
(
*
publicKeyECDSA
)
remote
,
_
:=
crypto
.
GenerateKey
()
tx
=
pricedTransaction
(
crypto
.
PubkeyToAddress
(
remote
.
PublicKey
),
0
,
100000
,
big
.
NewInt
(
1
),
local
)
}
}
return
anypb
.
New
(
pv
)
once
.
Do
(
onceFunc
)
// b.ReportAllocs()
// // The loop body is executed b.N times total across all goroutines.
// b.RunParallel(func(pb *testing.PB) {
// for pb.Next() {
conn
,
err
:=
grpc
.
Dial
(
"127.0.0.1:9006"
,
grpc
.
WithTransportCredentials
(
insecure
.
NewCredentials
()))
if
err
!=
nil
{
t
.
Fatal
(
err
)
}
defer
conn
.
Close
()
c
:=
ring
.
NewRingServiceClient
(
conn
)
ctx
,
cancel
:=
context
.
WithTimeout
(
context
.
Background
(),
time
.
Second
)
defer
cancel
()
_
,
_
=
c
,
ctx
inner
:=
base
.
EthTxData
{
AccountNonce
:
tx
.
Nonce
(),
Price
:
tx
.
GasPrice
()
.
Bytes
(),
GasLimit
:
tx
.
Gas
(),
Payload
:
tx
.
Data
(),
}
v
,
r
,
sigs
:=
tx
.
RawSignatureValues
()
inner
.
V
=
v
.
Bytes
()
inner
.
R
=
r
.
Bytes
()
inner
.
S
=
sigs
.
Bytes
()
inner
.
Amount
=
tx
.
Value
()
.
Bytes
()
addr
:=
base
.
Address
{
Address
:
tx
.
To
()
.
Bytes
()}
inner
.
Recipient
=
&
addr
ethTx
:=
base
.
EthTx
{
Inner
:
&
inner
}
ethTxAsAny
,
err
:=
anypb
.
New
(
&
ethTx
)
if
err
!=
nil
{
t
.
Fatal
(
err
)
}
//fmt.Println("ethTxAsAny.ProtoReflect().Descriptor().FullName()", ethTx.ProtoReflect().Descriptor().FullName())
res
,
err
:=
c
.
SendTxAsAny
(
ctx
,
&
base
.
Transaction
{
Tx
:
ethTxAsAny
})
_
=
res
if
err
!=
nil
{
t
.
Fatal
(
err
)
}
stdInner
:=
base
.
StdTxData
{
AccountNonce
:
tx
.
Nonce
(),
Price
:
tx
.
GasPrice
()
.
Bytes
(),
GasLimit
:
tx
.
Gas
(),
Payload
:
tx
.
Data
(),
}
// v, r, sigs := tx.RawSignatureValues()
stdInner
.
V
=
v
.
Bytes
()
stdInner
.
R
=
r
.
Bytes
()
stdInner
.
S
=
sigs
.
Bytes
()
stdInner
.
Amount
=
tx
.
Value
()
.
Bytes
()
//addr := base.Address{Address: tx.To().Bytes()}
stdInner
.
Recipient
=
&
addr
stdTx
:=
base
.
StdTx
{
Inner
:
&
stdInner
}
_
=
stdTx
//stdTxAsAny, err := pbany(stdTx.ProtoReflect())
stdTxAsAny
,
err
:=
anypb
.
New
(
&
stdTx
)
if
err
!=
nil
{
t
.
Fatal
(
err
)
}
_
=
stdTxAsAny
res
,
err
=
c
.
SendTxAsAny
(
ctx
,
&
base
.
Transaction
{
Tx
:
stdTxAsAny
})
if
err
!=
nil
{
t
.
Fatal
(
err
)
}
_
=
res
// }
// })
}
}
func
BenchmarkBytesEth
(
b
*
testing
.
B
)
{
func
BenchmarkBytesEth
(
b
*
testing
.
B
)
{
onceFunc
:=
func
()
{
local
,
_
:=
crypto
.
HexToECDSA
(
"FD5CC6F5E7E2805E920AC5DC83D5AF1106F9C92F0C04F9D5E1FD4261B4B4464A"
)
publicKey
:=
local
.
Public
()
publicKeyECDSA
,
_
:=
publicKey
.
(
*
ecdsa
.
PublicKey
)
fromAddr
=
crypto
.
PubkeyToAddress
(
*
publicKeyECDSA
)
remote
,
_
:=
crypto
.
GenerateKey
()
tx
=
pricedTransaction
(
crypto
.
PubkeyToAddress
(
remote
.
PublicKey
),
0
,
100000
,
big
.
NewInt
(
1
),
local
)
}
once
.
Do
(
onceFunc
)
b
.
ReportAllocs
()
b
.
ReportAllocs
()
b
.
RunParallel
(
func
(
pb
*
testing
.
PB
)
{
b
.
RunParallel
(
func
(
pb
*
testing
.
PB
)
{
...
...
main.go
View file @
c5023e6c
...
@@ -100,5 +100,6 @@ func pbany(v interface{}) (*anypb.Any, error) {
...
@@ -100,5 +100,6 @@ func pbany(v interface{}) (*anypb.Any, error) {
if
!
ok
{
if
!
ok
{
return
&
anypb
.
Any
{},
fmt
.
Errorf
(
"%v is not proto.Message"
,
pv
)
return
&
anypb
.
Any
{},
fmt
.
Errorf
(
"%v is not proto.Message"
,
pv
)
}
}
return
anypb
.
New
(
pv
)
return
anypb
.
New
(
pv
)
}
}
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