#include #include #include class AMFType { public: std::string Indice(){return myIndice;}; unsigned char GetType(){return myType;}; double NumValue(){return numval;}; std::string StrValue(){return strval;}; const char * Str(){return strval.c_str();}; int hasContent(){ if (!contents){return 0;} return contents->size(); }; void addContent(AMFType c){if (contents != 0){contents->push_back(c);}}; AMFType* getContentP(int i){if (contents != 0){return &contents->at(i);}else{return 0;}}; AMFType getContent(int i){if (contents != 0){return contents->at(i);}else{return AMFType("error");}}; AMFType* getContentP(std::string s){ if (contents != 0){ for (std::vector::iterator it = contents->begin(); it != contents->end(); it++){ if (it->Indice() == s){ return &(*it); } } } return this; }; AMFType getContent(std::string s){ if (contents != 0){ for (std::vector::iterator it = contents->begin(); it != contents->end(); it++){ if (it->Indice() == s){ return *it; } } } return AMFType("error"); }; AMFType(std::string indice, double val, unsigned char setType = 0x00){//num type initializer myIndice = indice; myType = setType; strval = ""; numval = val; contents = 0; }; AMFType(std::string indice, std::string val, unsigned char setType = 0x02){//str type initializer myIndice = indice; myType = setType; strval = val; numval = 0; contents = 0; }; AMFType(std::string indice, unsigned char setType = 0x03){//object type initializer myIndice = indice; myType = setType; strval = ""; numval = 0; contents = new std::vector; }; ~AMFType(){if (contents != 0){delete contents;contents=0;}}; AMFType& operator=(const AMFType &a) { myIndice = a.myIndice; myType = a.myType; strval = a.strval; numval = a.numval; if (contents){ if (a.contents != contents){ delete contents; if (a.contents){ contents = new std::vector; for (std::vector::iterator it = a.contents->begin(); it < a.contents->end(); it++){ contents->push_back(*it); } }else{ contents = 0; } } }else{ if (a.contents){ contents = new std::vector; for (std::vector::iterator it = a.contents->begin(); it < a.contents->end(); it++){ contents->push_back(*it); } } } return *this; }; AMFType(const AMFType &a){ myIndice = a.myIndice; myType = a.myType; strval = a.strval; numval = a.numval; if (a.contents){ contents = new std::vector; for (std::vector::iterator it = a.contents->begin(); it < a.contents->end(); it++){ contents->push_back(*it); } }else{contents = 0;} }; void Print(std::string indent = ""){ std::cerr << indent; switch (myType){ case 0x00: std::cerr << "Number"; break; case 0x01: std::cerr << "Bool"; break; case 0x02://short string case 0x0C: std::cerr << "String"; break; case 0x03: std::cerr << "Object"; break; case 0x05: std::cerr << "Null"; break; case 0x06: std::cerr << "Undefined"; break; case 0x0D: std::cerr << "Unsupported"; break; case 0xFF: std::cerr << "Container"; break; } std::cerr << " " << myIndice << " "; switch (myType){ case 0x00: case 0x01: std::cerr << numval; break; case 0x02: case 0x0C: std::cerr << strval; break; } std::cerr << std::endl; if (contents){ for (std::vector::iterator it = contents->begin(); it != contents->end(); it++){it->Print(indent+" ");} } }; protected: std::string myIndice; unsigned char myType; std::string strval; double numval; std::vector * contents; };//AMFType AMFType parseOneAMF(unsigned char *& data, unsigned int &len, unsigned int &i, std::string name){ std::string tmpstr; unsigned int tmpi = 0; unsigned char tmpdbl[8]; switch (data[i]){ case 0x00://number tmpdbl[7] = data[i+1]; tmpdbl[6] = data[i+2]; tmpdbl[5] = data[i+3]; tmpdbl[4] = data[i+4]; tmpdbl[3] = data[i+5]; tmpdbl[2] = data[i+6]; tmpdbl[1] = data[i+7]; tmpdbl[0] = data[i+8]; i+=9; fprintf(stderr, "AMF: Number %f\n", *(double*)tmpdbl); return AMFType(name, *(double*)tmpdbl, 0x00); break; case 0x01://bool i+=2; if (data[i-1] == 0){ fprintf(stderr, "AMF: Bool false\n"); return AMFType(name, (double)0, 0x01); }else{ fprintf(stderr, "AMF: Bool true\n"); return AMFType(name, (double)1, 0x01); } break; case 0x0C://long string tmpi = data[i+1]*256*256*256+data[i+2]*256*256+data[i+3]*256+data[i+4]; tmpstr = (char*)(data+i+5); i += tmpi + 5; fprintf(stderr, "AMF: String %s\n", tmpstr.c_str()); return AMFType(name, tmpstr, 0x0C); break; case 0x02://string tmpi = data[i+1]*256+data[i+2]; tmpstr = (char*)(data+i+3); i += tmpi + 3; fprintf(stderr, "AMF: String %s\n", tmpstr.c_str()); return AMFType(name, tmpstr, 0x02); break; case 0x05://null case 0x06://undefined case 0x0D://unsupported fprintf(stderr, "AMF: Null\n"); ++i; return AMFType(name, (double)0, data[i-1]); break; case 0x03:{//object ++i; AMFType ret = AMFType(name, data[i-1]); while (data[i] + data[i+1] != 0){ tmpi = data[i]*256+data[i+1]; tmpstr = (char*)(data+i+2); i += tmpi + 2; fprintf(stderr, "AMF: Indice %s\n", tmpstr.c_str()); ret.addContent(parseOneAMF(data, len, i, tmpstr)); } i += 3; return ret; } break; case 0x07://reference case 0x08://array case 0x0A://strict array case 0x0B://date case 0x0F://XML case 0x10://typed object case 0x11://AMF+ default: fprintf(stderr, "Error: Unimplemented AMF type %hhx - returning.\n", data[i]); return AMFType("error", (unsigned char)0xFF); break; } }//parseOneAMF AMFType parseAMF(unsigned char * data, unsigned int len){ AMFType ret("returned", (unsigned char)0xFF);//container type unsigned int i = 0; while (i < len){ret.addContent(parseOneAMF(data, len, i, ""));} return ret; }//parseAMF