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

  1. // Licensed under the Apache License, Version 2.0 (the "License");
  2. // you may not use this file except in compliance with the License.
  3. // You may obtain a copy of the License at
  4. //
  5. // http://www.apache.org/licenses/LICENSE-2.0
  6. //
  7. // Unless required by applicable law or agreed to in writing, software
  8. // distributed under the License is distributed on an "AS IS" BASIS,
  9. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. // See the License for the specific language governing permissions and
  11. // limitations under the License.
  12. package protocol
  13. import (
  14. "context"
  15. "crypto/tls"
  16. "net"
  17. "github.com/pkg/errors"
  18. "github.com/bboozzoo/mconnect/logger"
  19. "github.com/bboozzoo/mconnect/protocol/packet"
  20. )
  21. type Connection struct {
  22. conn *tls.Conn
  23. }
  24. type Configuration struct {
  25. Cert *tls.Certificate
  26. Identity *packet.Identity
  27. }
  28. func Dial(ctx context.Context, where string, conf *Configuration) (*Connection, error) {
  29. log := logger.FromContext(ctx)
  30. dialer := net.Dialer{}
  31. conn, err := dialer.DialContext(ctx, "tcp", where)
  32. if err != nil {
  33. return nil, errors.Wrapf(err, "failed to dial %s", where)
  34. }
  35. log.Debugf("connected to %v", conn.RemoteAddr())
  36. e := packet.NewEncoder(conn)
  37. if err := e.Encode(packet.NewIdentity(conf.Identity)); err != nil {
  38. return nil, errors.Wrapf(err, "failed to send identity")
  39. }
  40. log.Debugf("identity sent")
  41. // upgrade to secure connection
  42. tlsConf := tls.Config{
  43. InsecureSkipVerify: true,
  44. Certificates: []tls.Certificate{*conf.Cert},
  45. }
  46. tlsConn := tls.Server(conn, &tlsConf)
  47. if err := tlsConn.Handshake(); err != nil {
  48. log.Errorf("TLS handshake failed: %v", err)
  49. return nil, err
  50. }
  51. return &Connection{conn: tlsConn}, nil
  52. }
  53. func (c *Connection) Close() error {
  54. if c.conn != nil {
  55. c.conn.Close()
  56. }
  57. return nil
  58. }
  59. func (c *Connection) Receive() (*packet.Packet, error) {
  60. d := packet.NewDecoder(c.conn)
  61. var p packet.Packet
  62. if err := d.Decode(&p); err != nil {
  63. return nil, err
  64. }
  65. return &p, nil
  66. }
  67. func (c *Connection) Send(p packet.Packet) error {
  68. e := packet.NewEncoder(c.conn)
  69. return e.Encode(&p)
  70. }