package client import ( "bytes" "context" "encoding/json" "fmt" "io" "net/http" "time" ) type QQMusic struct { http *http.Client } func (c *QQMusic) rpcDoRequest(ctx context.Context, reqBody any) ([]byte, error) { reqBodyBuf, err := json.Marshal(reqBody) if err != nil { return nil, fmt.Errorf("qqMusicClient[rpcDoRequest] marshal request: %w", err) } const endpointURL = "https://u.y.qq.com/cgi-bin/musicu.fcg" req, err := http.NewRequestWithContext(ctx, http.MethodPost, endpointURL+fmt.Sprintf("?pcachetime=%d", time.Now().Unix()), bytes.NewReader(reqBodyBuf), ) if err != nil { return nil, fmt.Errorf("qqMusicClient[rpcDoRequest] create request: %w", err) } req.Header.Set("Accept", "*/*") req.Header.Set("Accept-Language", "zh-CN") req.Header.Set("User-Agent", "Mozilla/5.0 (compatible; MSIE 9.0; Windows NT 6.1; WOW64; Trident/5.0)") req.Header.Set("Content-Type", "application/x-www-form-urlencoded") // req.Header.Set("Accept-Encoding", "gzip, deflate") reqp, err := c.http.Do(req) if err != nil { return nil, fmt.Errorf("qqMusicClient[rpcDoRequest] send request: %w", err) } defer reqp.Body.Close() respBodyBuf, err := io.ReadAll(reqp.Body) if err != nil { return nil, fmt.Errorf("qqMusicClient[rpcDoRequest] read response: %w", err) } return respBodyBuf, nil } type rpcRequest struct { Method string `json:"method"` Module string `json:"module"` Param any `json:"param"` } type rpcResponse struct { Code int `json:"code"` Ts int64 `json:"ts"` StartTs int64 `json:"start_ts"` TraceID string `json:"traceid"` } type rpcSubResponse struct { Code int `json:"code"` Data json.RawMessage `json:"data"` } func (c *QQMusic) rpcCall(ctx context.Context, protocol string, method string, module string, param any, ) (json.RawMessage, error) { reqBody := map[string]any{protocol: rpcRequest{ Method: method, Module: module, Param: param, }} respBodyBuf, err := c.rpcDoRequest(ctx, reqBody) if err != nil { return nil, fmt.Errorf("qqMusicClient[rpcCall] do request: %w", err) } // check rpc response status respStatus := rpcResponse{} if err := json.Unmarshal(respBodyBuf, &respStatus); err != nil { return nil, fmt.Errorf("qqMusicClient[rpcCall] unmarshal response: %w", err) } if respStatus.Code != 0 { return nil, fmt.Errorf("qqMusicClient[rpcCall] rpc error: %d", respStatus.Code) } // parse response data var respBody map[string]json.RawMessage if err := json.Unmarshal(respBodyBuf, &respBody); err != nil { return nil, fmt.Errorf("qqMusicClient[rpcCall] unmarshal response: %w", err) } subRespBuf, ok := respBody[protocol] if !ok { return nil, fmt.Errorf("qqMusicClient[rpcCall] sub-response not found") } subResp := rpcSubResponse{} if err := json.Unmarshal(subRespBuf, &subResp); err != nil { return nil, fmt.Errorf("qqMusicClient[rpcCall] unmarshal sub-response: %w", err) } if subResp.Code != 0 { return nil, fmt.Errorf("qqMusicClient[rpcCall] sub-response error: %d", subResp.Code) } return subResp.Data, nil } func (c *QQMusic) downloadFile(ctx context.Context, url string) ([]byte, error) { req, err := http.NewRequestWithContext(ctx, http.MethodGet, url, nil) if err != nil { return nil, fmt.Errorf("qmc[downloadFile] init request: %w", err) } //req.Header.Set("Accept", "image/webp,image/*,*/*;q=0.8") // jpeg is preferred to embed in audio req.Header.Set("Accept-Language", "zh-CN,zh;q=0.8,en-US;q=0.6,en;q=0.5;q=0.4") req.Header.Set("User-Agent", "Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/53.0.47.134 Safari/537.36 QBCore/3.53.47.400 QQBrowser/9.0.2524.400") resp, err := c.http.Do(req) if err != nil { return nil, fmt.Errorf("qmc[downloadFile] send request: %w", err) } defer resp.Body.Close() if resp.StatusCode != http.StatusOK { return nil, fmt.Errorf("qmc[downloadFile] unexpected http status %s", resp.Status) } return io.ReadAll(resp.Body) } func NewQQMusicClient() *QQMusic { return &QQMusic{ http: &http.Client{ Timeout: 10 * time.Second, }, } }