| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284 | //Package whatsapp provides a developer API to interact with the WhatsAppWeb-Servers.package whatsappimport (	"crypto/hmac"	"crypto/sha256"	"encoding/json"	"fmt"	"github.com/Rhymen/go-whatsapp/binary"	"github.com/Rhymen/go-whatsapp/crypto/cbc"	"github.com/gorilla/websocket"	"math/rand"	"net/http"	"os"	"strconv"	"strings"	"sync"	"time")type metric byteconst (	debugLog metric = iota + 1	queryResume	queryReceipt	queryMedia	queryChat	queryContacts	queryMessages	presence	presenceSubscribe	group	read	chat	received	pic	status	message	queryActions	block	queryGroup	queryPreview	queryEmoji	queryMessageInfo	spam	querySearch	queryIdentity	queryUrl	profile	contact	queryVcard	queryStatus	queryStatusUpdate	privacyStatus	queryLiveLocations	liveLocation	queryVname	queryLabels	call	queryCall	queryQuickReplies)type flag byteconst (	ignore flag = 1 << (7 - iota)	ackRequest	available	notAvailable	expires	skipOffline)/*Conn is created by NewConn. Interacting with the initialized Conn is the main way of interacting with our package.It holds all necessary information to make the package work internally.*/type Conn struct {	wsConn         *websocket.Conn	session        *Session	listener       map[string]chan string	listenerMutex  sync.RWMutex	writeChan      chan wsMsg	handler        []Handler	msgCount       int	msgTimeout     time.Duration	Info           *Info	Store          *Store	ServerLastSeen time.Time}type wsMsg struct {	messageType int	data        []byte}/*Creates a new connection with a given timeout. The websocket connection to the WhatsAppWeb servers get´s established.The goroutine for handling incoming messages is started*/func NewConn(timeout time.Duration) (*Conn, error) {	dialer := &websocket.Dialer{		ReadBufferSize:   25 * 1024 * 1024,		WriteBufferSize:  10 * 1024 * 1024,		HandshakeTimeout: timeout,	}	headers := http.Header{"Origin": []string{"https://web.whatsapp.com"}}	wsConn, _, err := dialer.Dial("wss://w3.web.whatsapp.com/ws", headers)	if err != nil {		return nil, fmt.Errorf("couldn't dial whatsapp web websocket: %v", err)	}	wac := &Conn{		wsConn:        wsConn,		listener:      make(map[string]chan string),		listenerMutex: sync.RWMutex{},		writeChan:     make(chan wsMsg),		handler:       make([]Handler, 0),		msgCount:      0,		msgTimeout:    timeout,		Store:         newStore(),	}	go wac.readPump()	go wac.writePump()	go wac.keepAlive(20000, 90000)	return wac, nil}func (wac *Conn) write(data []interface{}) (<-chan string, error) {	d, err := json.Marshal(data)	if err != nil {		return nil, err	}	ts := time.Now().Unix()	messageTag := fmt.Sprintf("%d.--%d", ts, wac.msgCount)	msg := fmt.Sprintf("%s,%s", messageTag, d)	ch := make(chan string, 1)	wac.listenerMutex.Lock()	wac.listener[messageTag] = ch	wac.listenerMutex.Unlock()	wac.writeChan <- wsMsg{websocket.TextMessage, []byte(msg)}	wac.msgCount++	return ch, nil}func (wac *Conn) writeBinary(node binary.Node, metric metric, flag flag, tag string) (<-chan string, error) {	if len(tag) < 2 {		return nil, fmt.Errorf("no tag specified or to short")	}	b, err := binary.Marshal(node)	if err != nil {		return nil, err	}	cipher, err := cbc.Encrypt(wac.session.EncKey, nil, b)	if err != nil {		return nil, err	}	h := hmac.New(sha256.New, wac.session.MacKey)	h.Write(cipher)	hash := h.Sum(nil)	data := []byte(tag + ",")	data = append(data, byte(metric), byte(flag))	data = append(data, hash[:32]...)	data = append(data, cipher...)	ch := make(chan string, 1)	wac.listenerMutex.Lock()	wac.listener[tag] = ch	wac.listenerMutex.Unlock()	msg := wsMsg{websocket.BinaryMessage, data}	wac.writeChan <- msg	wac.msgCount++	return ch, nil}func (wac *Conn) readPump() {	defer wac.wsConn.Close()	for {		msgType, msg, err := wac.wsConn.ReadMessage()		if err != nil {			if websocket.IsUnexpectedCloseError(err, websocket.CloseGoingAway) {				wac.handle(fmt.Errorf("unexpected websocket close: %v", err))			}			break		}		data := strings.SplitN(string(msg), ",", 2)		//Kepp-Alive Timestmap		if data[0][0] == '!' {			msecs, err := strconv.ParseInt(data[0][1:], 10, 64)			if err != nil {				fmt.Fprintf(os.Stderr, "Error converting time string to uint: %v\n", err)				continue			}			wac.ServerLastSeen = time.Unix(msecs/1000, (msecs%1000)*int64(time.Millisecond))			continue		}		wac.listenerMutex.RLock()		listener, hasListener := wac.listener[data[0]]		wac.listenerMutex.RUnlock()		if hasListener && len(data[1]) > 0 {			listener <- data[1]			wac.listenerMutex.Lock()			delete(wac.listener, data[0])			wac.listenerMutex.Unlock()		} else if msgType == 2 && wac.session != nil && wac.session.EncKey != nil {			message, err := wac.decryptBinaryMessage([]byte(data[1]))			if err != nil {				wac.handle(fmt.Errorf("error decoding binary: %v", err))				continue			}			wac.dispatch(message)		} else {			if len(data[1]) > 0 {				wac.handle(string(data[1]))			}		}	}}func (wac *Conn) writePump() {	for msg := range wac.writeChan {		if err := wac.wsConn.WriteMessage(msg.messageType, msg.data); err != nil {			fmt.Fprintf(os.Stderr, "error writing to socket: %v", err)		}	}}func (wac *Conn) keepAlive(minIntervalMs int, maxIntervalMs int) {	for {		wac.writeChan <- wsMsg{			messageType: websocket.TextMessage,			data:        []byte("?,,"),		}		interval := rand.Intn(maxIntervalMs-minIntervalMs) + minIntervalMs		<-time.After(time.Duration(interval) * time.Millisecond)	}}func (wac *Conn) decryptBinaryMessage(msg []byte) (*binary.Node, error) {	//message validation	h2 := hmac.New(sha256.New, wac.session.MacKey)	h2.Write([]byte(msg[32:]))	if !hmac.Equal(h2.Sum(nil), msg[:32]) {		return nil, fmt.Errorf("message received with invalid hmac")	}	// message decrypt	d, err := cbc.Decrypt(wac.session.EncKey, nil, msg[32:])	if err != nil {		return nil, fmt.Errorf("error decrypting message with AES: %v", err)	}	// message unmarshal	message, err := binary.Unmarshal(d)	if err != nil {		return nil, fmt.Errorf("error decoding binary: %v", err)	}	return message, nil}
 |