美文网首页C语言C++
跨平台 C++ 纯socket 访问webapi json

跨平台 C++ 纯socket 访问webapi json

作者: 毛毛v5 | 来源:发表于2019-05-27 20:06 被阅读0次

webapi.h

#pragma once
#include <string>
#include <list>
#include <map>
class webapi
{
    class urlitem {
    public:
        urlitem() {
        }
        //std::string url;
        std::string fullpath; // http://www.pic98.com:8080/file/beautileg/samansha/201901011506.jpg#abc
        std::string query; // session=a231-dd33&token=aabbcc
        std::map<std::string, std::string> param; // {session =>a231-dd33,token=>aabbcc}
        std::string verb; // POST or GET
        std::string prot; // http
        std::string host; // www.pic98.com
        uint16_t port; // 8080
        std::string path; // /file/beautileg/samansha/201901011506.jpg
        std::string pathdir; // /file/beautileg/samasha
        std::string file; // 201901011506.jpg 
        std::list<std::string> ip;  // dns result for www.pic98.com
        std::string officalhost;
        std::list<std::string> aliases; // dns result for www.pic98.com
    };
private:
    webapi();
    ~webapi();
public:
    static webapi* me();
    std::string post(char* url,char* sid, char* par);
    bool urlparse(char* urlin,urlitem & out);
    std::string prepareheader(const urlitem* ui,char*,std::string post);
};


webapi.cpp

#include "webapi.h"
#include <vector>
#include <map>
#include <list>
#include <sstream>

#ifdef _WIN32
#include <winsock.h>
#include <share.h>
#include <io.h>
#include <direct.h> 
#else
#include <unistd.h>
#include <sys/stat.h>
#include <sys/file.h>
#include <sys/socket.h>
#include <fcntl.h>
#include <errno.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <netdb.h>
#include <sys/stat.h>


#include <unistd.h>
#include <sys/uio.h>


#include <sys/types.h>
#include <errno.h>


#endif


webapi::webapi(){
}


webapi::~webapi(){
}

webapi* webapi::me() {
    static webapi* wa = nullptr;
    if (wa == nullptr) {
        wa = new webapi();
    }
    return wa;
}

std::string UTF8_To_string(const std::string & str){
    int nwLen = MultiByteToWideChar(CP_UTF8, 0, str.c_str(), -1, NULL, 0);
    wchar_t * pwBuf = new wchar_t[nwLen + 1];
    memset(pwBuf, 0, nwLen * 2 + 2);
    MultiByteToWideChar(CP_UTF8, 0, str.c_str(), str.length(), pwBuf, nwLen);
    int nLen = WideCharToMultiByte(CP_ACP, 0, pwBuf, -1, NULL, NULL, NULL, NULL);
    char * pBuf = new char[nLen + 1];
    memset(pBuf, 0, nLen + 1);
    WideCharToMultiByte(CP_ACP, 0, pwBuf, nwLen, pBuf, nLen, NULL, NULL);
    std::string retStr = pBuf;
    delete[]pBuf;
    delete[]pwBuf;
    pBuf = NULL;
    pwBuf = NULL;
    return retStr;
}

std::string string_To_UTF8(const std::string & str){
    int nwLen = ::MultiByteToWideChar(CP_ACP, 0, str.c_str(), -1, NULL, 0);
    wchar_t * pwBuf = new wchar_t[nwLen + 1];
    ZeroMemory(pwBuf, nwLen * 2 + 2);
    ::MultiByteToWideChar(CP_ACP, 0, str.c_str(), str.length(), pwBuf, nwLen);
    int nLen = ::WideCharToMultiByte(CP_UTF8, 0, pwBuf, -1, NULL, NULL, NULL, NULL);
    char * pBuf = new char[nLen + 1];
    ZeroMemory(pBuf, nLen + 1);
    ::WideCharToMultiByte(CP_UTF8, 0, pwBuf, nwLen, pBuf, nLen, NULL, NULL);
    std::string retStr(pBuf);
    delete[]pwBuf;
    delete[]pBuf;
    pwBuf = NULL;
    pBuf = NULL;
    return retStr;
}


std::string webapi::post(char* url,char* sid, char* par) {
    std::string ret="";
    struct sockaddr_in stSockAddr;
    std::vector<char> vbuf;
    char buf[2048] = { 0 };
    urlitem ui;
    urlparse(url, ui);
    int sockfd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
    do {
        if (-1 == sockfd) {
            break;
        }
        int recvTimeout = 30 * 1000;  //30s
        int sendTimeout = 60 * 1000;  //30s
        setsockopt(sockfd, SOL_SOCKET, SO_RCVTIMEO, (char *)&recvTimeout, sizeof(int));
        setsockopt(sockfd, SOL_SOCKET, SO_SNDTIMEO, (char *)&sendTimeout, sizeof(int));
        memset(&stSockAddr, 0, sizeof(struct sockaddr_in));
        stSockAddr.sin_family = AF_INET;
        stSockAddr.sin_port = htons(ui.port);
        stSockAddr.sin_addr.s_addr = inet_addr(ui.ip.front().c_str());
        do {
            if (-1 == connect(sockfd, (const struct sockaddr *)&stSockAddr, sizeof(struct sockaddr_in))) {
                //WCLOG(LS_ERROR) << "socket connect error:" << errno;
                break;
            }
            
            std::string post1 = prepareheader(&ui,sid,par);
            std::string post = post1;// string_To_UTF8(post1);
            OutputDebugStringA(post.c_str());
            int left = post.length();
            int length = post.length();
            int len, i, j;
            while (left > 0) {
                len = send(sockfd, post.c_str() + length - left, left, 0);
                
                if (len <= 0) {
                    //WCLOG(LS_ERROR) << "socket send error:" << errno;
                    break;
                }
                left -= len;
            }
            if (len <= 0)
                break;
            std::string header;
            i = 0;
            while (true) {
                len = recv(sockfd, buf, sizeof(buf), 0);
                if (len <= 0) {
                    break;
                }
                vbuf.insert(vbuf.end(), buf, buf + len);
                if (vbuf.size() >= 4) {
                    for (; i < vbuf.size() - 3; i++) {
                        if (vbuf[i] == '\r' && vbuf[i + 1] == '\n' && vbuf[i + 2] == '\r' && vbuf[i + 3] == '\n') {
                            header.append(&vbuf[0], i);
                            //WCLOG(LS_ERROR) << "http header:" << header;
                            vbuf.erase(vbuf.begin(), vbuf.begin() + i + 4);
                            break;
                        }
                    }
                }
                if (!header.empty()) {
                    break;
                }
            }

            if (len <= 0)
                break;
            header.append("\r\n");
            size_t pos = 0, lastpos = 0;
            std::string temp;
            std::map<std::string, std::string> headers; // Content-Disposition
            std::string httpver;
            for (i = 0; i < header.length() - 1; i++) {
                if (header[i] == '\r' && header[i + 1] == '\n') {
                    temp = header.substr(lastpos, i - lastpos);
                    lastpos = i + 2;
                    for (j = 0; j < temp.length(); j++) {
                        if (temp[j] == ':') {
                            headers[temp.substr(0, j)] = temp.substr(j + 1);
                            break;
                        }
                    }
                    if (j == temp.length())
                        httpver = temp;
                }
            }
            std::vector<std::string> httpvers;
            lastpos = 0;
            httpver.append(" ");
            for (i = 0; i < httpver.length(); i++) {
                if (httpver[i] == ' ') {
                    std::string s0 = httpver.substr(lastpos, i - lastpos);
                    s0.erase(std::find(s0.begin(), s0.end(), ' '));
                    if (s0.length() > 0)
                        httpvers.push_back(s0);
                    lastpos = i;
                }
            }
            if (httpvers.size() < 2 || !(httpvers[1] == "200" )) {
                break;
            }
            std::map<std::string, std::string>::iterator iter = headers.find("Content-Length");
            if (iter != headers.end()) {
                int contentlength = atoi((*iter).second.c_str());
                while (contentlength > vbuf.size()) {
                    len = recv(sockfd, buf, sizeof(buf), 0);
                    if (len <= 0) {
                        break;
                    }
                    vbuf.insert(vbuf.end(), buf, buf + len);
                }
                ret = UTF8_To_string(std::string(&vbuf[0],vbuf.size()));
            }


        } while (false);
    } while (false);
    return ret;
}

std::string webapi::prepareheader(const urlitem* ui,char* sid,std::string post) {
    std::ostringstream os;
    os << "POST " << ui->path << " HTTP/1.1\r\n"
        << "Host:" << ui->host << "\r\n"
        << "Content-Type:" << "application/json; charset=utf-8\r\n"
        << "mster-token:" << sid << "\r\n"
        //<< "Connection:keep-alive\r\n"
        //<< "Upgrade-Insecure-Requests:1\r\n"
        << "User-Agent:Mozilla/5.0(Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/70.0.3538.25 Safari/537.36 Core/1.70.3676.400 QQBrowser/10.4.3457.400\r\n"
        << "Accept:*/*\r\n"
        << "Content-Length:" << post.size() << "\r\n"
        << "Accept-Language: zh-CN,zh;q=0.9\r\n\r\n"
        << post;
    return os.str();
}

bool webapi::urlparse(char* urlin1, urlitem & out) {
    bool ret = false;
    int i = 0;
    std::string urlin(urlin1);
    if (strcmp(urlin.substr(0, 4).c_str(), "http") != 0)
        return false;
    int lastpos = 0;
    int pos = 0;
    std::string childs[10];
    std::string temp;
    int idx = 0;
    out.fullpath = urlin;
    pos = urlin.find('?', 0);
    if (pos >= 0) {
        out.fullpath = urlin.substr(0, pos);
        out.query = urlin.substr(pos + 1);
    }
    //
    lastpos = 0;
    for (i = 0; i < out.query.length(); i++) {
        if (out.query[i] == '&') {
            temp = out.query.substr(lastpos, i - lastpos);
            pos = temp.find('=', 0);
            if (pos >= 0) {
                out.param[temp.substr(0, pos)] = temp.substr(pos + 1);
            }
            lastpos = i + 1;
        }
    }
    temp = out.query.substr(lastpos);
    pos = temp.find('=', 0);
    if (pos >= 0) {
        out.param[temp.substr(0, pos)] = temp.substr(pos + 1);
    }
    lastpos = 0;
    idx = 0;
    for (i = 0; i < out.fullpath.length(); i++) {
        if (out.fullpath[i] == ':') {
            childs[idx] = out.fullpath.substr(lastpos, i - lastpos);
            lastpos = i + 1;
            idx++;
            break;
        }
    }
    out.prot = childs[0];
    out.port = 80;
    std::string fullpath = out.fullpath.substr(out.prot.length() + 3);
    pos = fullpath.find('/');
    if (pos >= 0) {
        out.host = fullpath.substr(0, pos);
        out.path = fullpath.substr(pos);
    }
    else if (pos = fullpath.find('\\') >= 0) {
        out.host = fullpath.substr(0, pos);
        out.path = fullpath.substr(pos);
    }
    pos = out.host.find(':');
    if (pos >= 0) {
        out.port = atoi(out.host.substr(pos + 1).c_str());
        out.host = out.host.substr(0, pos);
    }
    pos = out.path.rfind('/');
    if (pos >= 0) {
        out.file = out.path.substr(pos + 1);
    }
    else if (pos = out.path.rfind('\\') >= 0) {
        out.file = out.path.substr(pos + 1);
    }

    out.pathdir = out.path.substr(1, out.path.length() - out.file.length() - 1);

    struct hostent *hptr;
    char **pptr;
    if ((hptr = gethostbyname(out.host.c_str())) != nullptr) {
        if (hptr->h_name != nullptr)
            out.officalhost = hptr->h_name;
        for (pptr = hptr->h_aliases; *pptr != NULL; pptr++)
            out.aliases.push_back(*pptr);
        switch (hptr->h_addrtype) {
        case AF_INET:
            //case AF_INET6:
            pptr = hptr->h_addr_list;
            for (; *pptr != NULL; pptr++) {
                out.ip.push_back(inet_ntoa(*((in_addr *)*pptr)));
            }
            break;
        default:
            break;
        }
    }
    return true;
}

就这样调用吧

                std::thread t1([hWndThis,str1,strid,sessionid,userid, token]() {
                    std::string posturl = "http://www.pic98.com:8092/api/1.00/private";
                    if (token == "pulldlghis117") {
                        posturl = "http://www.pic98.com:8092/api/1.00/public";
                    }
                    std::string par;
                    std::stringstream ss;
                    ss << "{\"action\":\"caniplay\",\"id\":\"" << strid << "\"}";
                    std::string ret = webapi::me()->post((char*)posturl.c_str(),(char*)sessionid.ToString().c_str() ,(char*)ss.str().c_str());
                    char* pbuf1 = new char[str1.length() + 1];
                    strcpy(pbuf1, str1.c_str());
                    char* pbuf2 = new char[ret.length() + 1];
                    strcpy(pbuf2, ret.c_str());
                    ::PostMessage(hWndThis, WM_DLGFLVPLAYER_PLAY,(WPARAM)pbuf1,(LPARAM)pbuf2);
                });

                t1.detach();

相关文章

  • 跨平台 C++ 纯socket 访问webapi json

    webapi.h webapi.cpp 就这样调用吧

  • json与jsonp

    什么是JSON JSON的优点: 1、基于纯文本,跨平台传递极其简单; 2、Javascript原生支持,后台语言...

  • ios底层Socket编程理解

    前言 Socket网络编程在任何一门编程语言中都很重要,而且socket底层是纯C语言,跨平台,了解并熟悉底层交互...

  • json和jsonp

    JSON的优点: 1、基于纯文本,跨平台传递极其简单; 2、Javascript原生支持,后台语言几乎全部支持; ...

  • linux, windows, mac, ios等平台GCC预编

    写跨平台c/c++程序的时候,需要搞清各平台下面的预编译宏,区分各平台代码。而跨平台c/c++编程,GCC基本在各...

  • DolphinDB WebApi

    DolphinDB WebApi 简介 DolphinDB WebApi是DolphinDB提供的访问Server...

  • 网络协议(一)

    以下内容来自于小码哥"网络协议从入门到底层原理" C\C++跨平台的原理: C\C++跨平台: 使用平台相关的编译...

  • 编程语言介绍

    Java:跨平台,自动内存管理; python: ; c:; c++:Essential C++,C++Prime...

  • Asp.Net Core WebAPI 跨域处理

    一、Core WebAPI中的跨域处理 在使用WebAPI项目的时候基本上都会用到跨域处理 Core WebAP...

  • WebApi 跨域

    要解决WebApi的跨域问题,可以通过nuget引入microsoft.aspnet.webapi.cors包,然...

网友评论

    本文标题:跨平台 C++ 纯socket 访问webapi json

    本文链接:https://www.haomeiwen.com/subject/ezyktctx.html