mconnect - KDE Connect protocol implementation in Vala/C
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

85 lines
2.1 KiB

// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package protocol
import (
"context"
"crypto/tls"
"net"
"github.com/pkg/errors"
"github.com/bboozzoo/mconnect/logger"
"github.com/bboozzoo/mconnect/protocol/packet"
)
type Connection struct {
conn *tls.Conn
}
type Configuration struct {
Cert *tls.Certificate
Identity *packet.Identity
}
func Dial(ctx context.Context, where string, conf *Configuration) (*Connection, error) {
log := logger.FromContext(ctx)
dialer := net.Dialer{}
conn, err := dialer.DialContext(ctx, "tcp", where)
if err != nil {
return nil, errors.Wrapf(err, "failed to dial %s", where)
}
log.Debugf("connected to %v", conn.RemoteAddr())
e := packet.NewEncoder(conn)
if err := e.Encode(packet.NewIdentity(conf.Identity)); err != nil {
return nil, errors.Wrapf(err, "failed to send identity")
}
log.Debugf("identity sent")
// upgrade to secure connection
tlsConf := tls.Config{
InsecureSkipVerify: true,
Certificates: []tls.Certificate{*conf.Cert},
}
tlsConn := tls.Server(conn, &tlsConf)
if err := tlsConn.Handshake(); err != nil {
log.Errorf("TLS handshake failed: %v", err)
return nil, err
}
return &Connection{conn: tlsConn}, nil
}
func (c *Connection) Close() error {
if c.conn != nil {
c.conn.Close()
}
return nil
}
func (c *Connection) Receive() (*packet.Packet, error) {
d := packet.NewDecoder(c.conn)
var p packet.Packet
if err := d.Decode(&p); err != nil {
return nil, err
}
return &p, nil
}
func (c *Connection) Send(p packet.Packet) error {
e := packet.NewEncoder(c.conn)
return e.Encode(&p)
}