• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

mendersoftware / gui / 1113439055

19 Dec 2023 09:01PM UTC coverage: 82.752% (-17.2%) from 99.964%
1113439055

Pull #4258

gitlab-ci

mender-test-bot
chore: Types update

Signed-off-by: Mender Test Bot <mender@northern.tech>
Pull Request #4258: chore: Types update

4326 of 6319 branches covered (0.0%)

8348 of 10088 relevant lines covered (82.75%)

189.39 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

27.5
/src/js/utils/sockethook.js
1
// Copyright 2022 Northern.tech AS
2
//
3
//    Licensed under the Apache License, Version 2.0 (the "License");
4
//    you may not use this file except in compliance with the License.
5
//    You may obtain a copy of the License at
6
//
7
//        http://www.apache.org/licenses/LICENSE-2.0
8
//
9
//    Unless required by applicable law or agreed to in writing, software
10
//    distributed under the License is distributed on an "AS IS" BASIS,
11
//    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
//    See the License for the specific language governing permissions and
13
//    limitations under the License.
14
import { useCallback, useRef, useState } from 'react';
15

16
import msgpack5 from 'msgpack5';
17
import Cookies from 'universal-cookie';
18

19
import { apiUrl } from '../api/general-api';
20
import { TIMEOUTS } from '../constants/appConstants';
21
import { DEVICE_MESSAGE_PROTOCOLS as MessageProtocols, DEVICE_MESSAGE_TYPES as MessageTypes } from '../constants/deviceConstants';
22

23
const cookies = new Cookies();
15✔
24

25
const MessagePack = msgpack5();
15✔
26

27
export const byteArrayToString = body => String.fromCharCode(...body);
15✔
28

29
export const blobToString = blob =>
15✔
30
  new Promise(resolve => {
×
31
    let fr = new FileReader();
×
32
    fr.onload = () => {
×
33
      resolve(fr.result);
×
34
    };
35
    fr.readAsArrayBuffer(blob);
×
36
  });
37

38
export const useSession = ({ onClose, onHealthCheckFailed, onMessageReceived, onNotify, onOpen, token }) => {
15✔
39
  const [sessionId, setSessionId] = useState();
6✔
40
  const healthcheckTimeout = useRef();
6✔
41
  const socketRef = useRef();
6✔
42

43
  const sendMessage = useCallback(({ typ, body, props }) => {
6✔
44
    if (!socketRef.current) {
12!
45
      return;
12✔
46
    }
47
    const proto_header = { proto: MessageProtocols.Shell, typ, sid: socketRef.current.sessionId, props };
×
48
    const encodedData = MessagePack.encode({ hdr: proto_header, body });
×
49
    socketRef.current.send(encodedData);
×
50
  }, []);
51

52
  const close = useCallback(() => {
6✔
53
    if (socketRef.current?.readyState !== WebSocket.OPEN) {
8!
54
      return;
×
55
    }
56
    sendMessage({ typ: MessageTypes.Stop, body: {}, props: {} });
8✔
57
    socketRef.current.close();
8✔
58
    setSessionId();
×
59
  }, [sendMessage]);
60

61
  const healthcheckFailed = useCallback(() => {
6✔
62
    onHealthCheckFailed();
×
63
    close();
×
64
  }, [close, onHealthCheckFailed]);
65

66
  const onSocketMessage = useCallback(
6✔
67
    event =>
68
      blobToString(event.data).then(data => {
×
69
        const {
70
          hdr: { props = {}, proto, sid, typ },
×
71
          body
72
        } = MessagePack.decode(data);
×
73
        if (proto !== MessageProtocols.Shell) {
×
74
          return;
×
75
        }
76
        switch (typ) {
×
77
          case MessageTypes.New: {
78
            if (props.status == WebSocket.CLOSING) {
×
79
              onNotify(`Error: ${byteArrayToString(body)}`);
×
80
              setSessionId();
×
81
              return close();
×
82
            } else {
83
              clearTimeout(healthcheckTimeout.current);
×
84
              healthcheckTimeout.current = setTimeout(healthcheckFailed, 65 * TIMEOUTS.oneSecond);
×
85
              socketRef.current.sessionId = sid;
×
86
              return setSessionId(sid);
×
87
            }
88
          }
89
          case MessageTypes.Shell:
90
            return onMessageReceived(body);
×
91
          case MessageTypes.Stop:
92
            return close();
×
93
          case MessageTypes.Ping: {
94
            if (healthcheckTimeout.current) {
×
95
              clearTimeout(healthcheckTimeout.current);
×
96
            }
97
            sendMessage({ typ: MessageTypes.Pong });
×
98
            //
99
            const timeout = parseInt(props.timeout);
×
100
            if (timeout > 0) {
×
101
              healthcheckTimeout.current = setTimeout(healthcheckFailed, timeout * TIMEOUTS.oneSecond);
×
102
            }
103
            return;
×
104
          }
105
          default:
106
            break;
×
107
        }
108
      }),
109
    [close, healthcheckFailed, onMessageReceived, onNotify, sendMessage]
110
  );
111

112
  const onSocketError = useCallback(
6✔
113
    error => {
114
      onNotify(`WebSocket error: ${error.message}`);
×
115
      close();
×
116
      clearTimeout(healthcheckTimeout.current);
×
117
    },
118
    [close, onNotify]
119
  );
120

121
  const onSocketOpen = useCallback(() => {
6✔
122
    sendMessage({ typ: MessageTypes.New, props: {} });
×
123
    onOpen(true);
×
124
  }, [onOpen, sendMessage]);
125

126
  const onSocketClose = useCallback(
6✔
127
    e => {
128
      console.log('closing');
×
129
      onClose(e);
×
130
      close();
×
131
    },
132
    [close, onClose]
133
  );
134

135
  const connect = useCallback(
6✔
136
    deviceId => {
137
      const uri = `wss://${window.location.host}${apiUrl.v1}/deviceconnect/devices/${deviceId}/connect`;
×
138
      setSessionId();
×
139
      cookies.set('JWT', token, { path: '/', secure: true, sameSite: 'strict', maxAge: 5 });
×
140
      try {
×
141
        socketRef.current = new WebSocket(uri);
×
142
        socketRef.current.addEventListener('close', onSocketClose);
×
143
        socketRef.current.addEventListener('error', onSocketError);
×
144
        socketRef.current.addEventListener('message', onSocketMessage);
×
145
        socketRef.current.addEventListener('open', onSocketOpen);
×
146
      } catch (error) {
147
        console.log(error);
×
148
      }
149
      return () => {
×
150
        socketRef.current.removeEventListener('close', onSocketClose);
×
151
        socketRef.current.removeEventListener('error', onSocketError);
×
152
        socketRef.current.removeEventListener('message', onSocketMessage);
×
153
        socketRef.current.removeEventListener('open', onSocketOpen);
×
154
      };
155
    },
156
    [onSocketClose, onSocketOpen, onSocketError, onSocketMessage, token]
157
  );
158

159
  return [connect, sendMessage, close, socketRef.current?.readyState ?? WebSocket.CLOSED, sessionId];
6✔
160
};
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2025 Coveralls, Inc