Avionics
Dropship Simulator
YouTube.cpp
Go to the documentation of this file.
1 #include "YouTube.h"
2 
3 #include "D:/Rise Base/RakNet/Source/TCPInterface.h"
4 #include "D:/Rise Base/RakNet/Source/GetTime.h"
5 #include "D:/Rise Base/RakNet/Source/RakSleep.h"
6 #include "D:/Rise Base/RakNet/Source/Base64Encoder.h"
7 #include "D:/Rise Base/RakNet/Source/HTTPConnection.h"
8 
9 #include <io.h>
10 #include <fcntl.h>
11 #include <share.h>
12 #include <sys/stat.h>
13 
14 namespace Devices
15 {
16  void YouTube::Initialize(Logger* prmLogger, Bus* prmBus, YouTubeConfig* prmConfig)
17  {
18  logger = prmLogger;
19  bus = prmBus;
20  config = prmConfig;
21  }
22 
23  void YouTube::FrameMove() const
24  {
25  for (UINT i = 0; i < bus->commandStream.size(); i++)
26  {
27  Command command = bus->commandStream.at(i);
28  if (command.delay != 0.0f) continue;
29 
30  if (command.name == "YouTubePublish")
31  {
32  }
33  }
34  }
35 
36  std::string YouTube::Publish(PublishParameters params) const
37  {
38  std::string url = "";
39 
40  std::string authToken = Login();
41  if (authToken.empty())
42  {
43  logger->Log("YouTube::Publish No AuthToken available, bailing out!", Logger::Error);
44  return url;
45  }
46 
47  RakNet::TCPInterface tcp;
48  RakNet::HTTPConnection http;
49  http.Init(&tcp, "uploads.gdata.youtube.com", 80);
50 
51  std::string apiUrl = "/feeds/api/users/default/uploads";
52  std::string boundary = "f93dcbA3";
53 
54  std::string header = "";
55  header += "Authorization: GoogleLogin auth=" + authToken + "\n";
56  header += "Content-Type: multipart/related; boundary=\"" + boundary + "\"\n";
57  header += "GData-Version: 2.1\n";
58  header += "X-GData-Key: key=" + config->developerKey + "\n";
59  header += "Slug: test.mp4\n";
60  header += "Connection: close\n";
61 
62  std::string body = "";
63  body += "--" + boundary + "\n";
64  body += "Content-Type: application/atom+xml; charset=UTF-8\n";
65  body += "\n";
66  body += "<?xml version=\"1.0\"?>\n";
67  body += "<entry xmlns=\"http://www.w3.org/2005/Atom\"\n";
68  body += " xmlns:media=\"http://search.yahoo.com/mrss/\"\n";
69  body += " xmlns:yt=\"http://gdata.youtube.com/schemas/2007\">\n";
70  body += " <media:group>\n";
71  body += " <media:title type=\"plain\">" + params.title + "</media:title>\n";
72  body += " <media:description type=\"plain\">\n";
73  body += " " + params.description + "\n";
74  body += " </media:description>\n";
75  body += " <media:category\n";
76  body += " scheme=\"http://gdata.youtube.com/schemas/2007/categories.cat\">" + params.category + "\n";
77  body += " </media:category>\n";
78  body += " <media:keywords>" + params.keywords + "</media:keywords>\n";
79  body += " </media:group>\n";
80  if (params.privacy == PrivacyEnum::Unlisted)
81  {
82  body += " <yt:accessControl action='list' permission='denied' />\n"; // unlisted... can also be the channel default!
83  }
84  body += "</entry>\n";
85  body += "--" + boundary + "\n";
86  body += "Content-Type: video/mp4\n"; // can also be video/mpeg or application/octet-stream
87  body += "Content-Transfer-Encoding: base64\n";
88  body += "\n";
89 
90  int handle = -1;
91  _sopen_s(&handle, params.pathfilename.c_str(), _O_RDONLY | _O_BINARY, SH_DENYNO, S_IWRITE);
92  if (handle == -1)
93  {
94  std::string msg = "YouTube::Publish Could not open file: " + params.pathfilename;
95  logger->Log(msg.c_str(), Logger::Error, errno);
96  return url;
97  }
98 
99  int bytes = _filelength(handle);
100  void* data = malloc(bytes);
101  _read(handle, data, bytes);
102  _close(handle);
103 
104  char* outputData = new char[bytes * 3];
105  Base64Encoding(static_cast<unsigned char *>(data), bytes, outputData);
106  body += outputData;
107  SAFE_DELETE(outputData);
108  SAFE_DELETE(data);
109 
110  body += "--" + boundary + "--";
111 
112  http.Post(apiUrl.c_str(), body.c_str(), header.c_str());
113 
114  if (!tcp.Start(0, 0))
115  {
116  logger->Log("YouTube::Publish Unknown error starting TCP", Logger::Error);
117  return url;
118  }
119 
120  logger->Log("YouTube::Publish Processing HTTP POST...");
121  RakNet::TimeMS timeoutTime = RakNet::GetTimeMS() + 60000;
122  while (RakNet::GetTimeMS() < timeoutTime)
123  {
124  RakNet::Packet* packet = tcp.Receive();
125  if (packet)
126  {
127  http.ProcessTCPPacket(packet);
128  tcp.DeallocatePacket(packet);
129  }
130 
131  http.Update();
132  if (!http.IsBusy())
133  {
134  int code = 0;
135  RakNet::RakString fileContents;
136  if (http.HasBadResponse(&code, &fileContents))
137  {
138  logger->Log(fileContents.C_String(), Logger::Error, code);
139  }
140  else
141  {
142  fileContents = http.Read();
143  url = fileContents.C_String();
144  UINT ordinal = url.find("href='");
145  if (ordinal != url.npos)
146  {
147  UINT endOrdinal = url.find("&amp;feature=youtube_gdata'");
148  if (endOrdinal != url.npos)
149  {
150  ordinal += 6; // get rid of leading junk
151  url = url.substr(ordinal, endOrdinal - ordinal);
152  std::string msg = "YouTube::Publish Success! Returning: " + url;
153  logger->Log(msg.c_str());
154  }
155  else
156  {
157  logger->Log("YouTube::Publish Unexpected return data to follow...", Logger::Warn);
158  logger->Log(url.c_str(), Logger::Error);
159  url = "";
160  }
161  }
162  else
163  {
164  logger->Log("YouTube::Publish Unexpected return data to follow...", Logger::Warn);
165  logger->Log(url.c_str(), Logger::Error);
166  url = "";
167  }
168  }
169 
170  break;
171  }
172 
173  // Prevent 100% cpu usage
174  RakSleep(30);
175  }
176 
177  if (RakNet::GetTimeMS() >= timeoutTime)
178  {
179  logger->Log("YouTube::Publish Timeout!", Logger::Error);
180  }
181 
182  tcp.Stop();
183 
184  return url;
185  }
186 
189  std::string YouTube::Login() const
190  {
191  std::string authToken = "";
192 
193  RakNet::TCPInterface tcp;
194  RakNet::HTTPConnection http;
195  http.Init(&tcp, "www.google.com", 443);
196 
197  std::string url = "/accounts/ClientLogin?Email=" + config->username + "&Passwd=" + config->password + "&service=youtube&source=" + config->source;
198  http.Get(url.c_str());
199 
200  if (!tcp.Start(0, 0))
201  {
202  logger->Log("YouTube::Login Unknown error starting TCP", Logger::Error);
203  return authToken;
204  }
205 
206  logger->Log("YouTube::Login Starting SSL Client...");
207  tcp.StartSSLClient(http.GetServerAddress());
208 
209  logger->Log("YouTube::Login Processing HTTP GET...");
210  RakNet::TimeMS timeoutTime = RakNet::GetTimeMS() + 5000;
211  while (RakNet::GetTimeMS() < timeoutTime)
212  {
213  RakNet::Packet* packet = tcp.Receive();
214  if (packet)
215  {
216  http.ProcessTCPPacket(packet);
217  tcp.DeallocatePacket(packet);
218  }
219 
220  http.Update(true);
221  if (!http.IsBusy())
222  {
223  RakNet::RakString fileContents = http.Read();
224  authToken = fileContents.C_String();
225  break;
226  }
227 
228  // Prevent 100% cpu usage
229  RakSleep(30);
230  }
231 
232  if (RakNet::GetTimeMS() >= timeoutTime)
233  {
234  logger->Log("YouTube::Login Timeout!", Logger::Error);
235  }
236  else if (authToken.empty())
237  {
238  logger->Log("YouTube::Login Response was empty!", Logger::Error);
239  }
240  else
241  {
242  UINT ordinal = authToken.find("Auth=");
243  //UINT ordinal = authToken.find("SID=");
244  if (ordinal != std::string::npos)
245  {
246  authToken = authToken.substr(ordinal + 5);
247  //authToken = authToken.substr(ordinal);
248  authToken = authToken.substr(0, authToken.length() - 1); // trim trailing return
249  char msg[999];
250  sprintf_s(msg, 999, "YouTube::Login Returning AuthToken: %s (...)", authToken.substr(0, 32).c_str());
251  logger->Log(msg);
252  }
253  else
254  {
255  logger->Log("YouTube::Login Could not find Auth= in reponse! Response text to follow...", Logger::Warn);
256  logger->Log(authToken.c_str(), Logger::Error);
257  authToken = "";
258  }
259  }
260 
261  tcp.Stop();
262 
263  return authToken;
264  }
265 }
Logger * logger
Definition: YouTube.h:44
std::string Publish(PublishParameters publishParameters) const
Definition: YouTube.cpp:36
std::vector< Command > commandStream
Definition: Bus.h:20
Definition: Logger.h:5
void FrameMove() const
Definition: YouTube.cpp:23
YouTubeConfig * config
Definition: YouTube.h:46
std::string Login() const
need to move RakNet post out of this module and just get a pointer to RakNet device! ...
Definition: YouTube.cpp:189
okay, the portable keyboard numbers don&#39;t work like the outside keypad because the outside keypad is ...
Definition: Analog.cpp:3
Definition: Bus.h:12
std::string name
command name
Definition: Command.h:11
Definition: Command.h:5
std::string password
Definition: YouTube.h:12
std::string source
Definition: YouTube.h:13
void Log(const char *msg, Level level=Info, int errorCode=0)
These have to be in this order.
Definition: Logger.cpp:16
std::string developerKey
Definition: YouTube.h:14
void Initialize(Logger *, Bus *, YouTubeConfig *)
Definition: YouTube.cpp:16
std::string username
Definition: YouTube.h:11
float delay
wait number of seconds before executing command
Definition: Command.h:8