• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2011 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "content/test/plugin/plugin_geturl_test.h"
6 
7 #include <stdio.h>
8 
9 #include "base/basictypes.h"
10 #include "base/files/file_util.h"
11 #include "base/strings/string_number_conversions.h"
12 #include "base/strings/string_util.h"
13 #include "base/strings/utf_string_conversions.h"
14 
15 // url for "self".  The %22%22 is to make a statement for javascript to
16 // evaluate and return.
17 #define SELF_URL "javascript:window.location+\"\""
18 
19 // The identifier for the self url stream.
20 #define SELF_URL_STREAM_ID 1
21 
22 // The identifier for the fetched url stream.
23 #define FETCHED_URL_STREAM_ID 2
24 
25 // url for testing GetURL with a bogus URL.
26 #define BOGUS_URL "bogoproto:///x:/asdf.xysdhffieasdf.asdhj/"
27 
28 // url for testing redirect notifications sent to plugins.
29 #define REDIRECT_SRC_URL \
30     "http://mock.http/npapi/plugin_read_page_redirect_src.html"
31 
32 // The notification id for the redirect notification url that we will cancel.
33 #define REDIRECT_SRC_URL_NOTIFICATION_CANCEL_ID 4
34 
35 // The notification id for the redirect notification url that we will accept.
36 #define REDIRECT_SRC_URL_NOTIFICATION_ALLOW_ID 5
37 
38 // The identifier for the bogus url stream.
39 #define BOGUS_URL_STREAM_ID 3
40 
41 // The maximum chunk size of stream data.
42 #define STREAM_CHUNK 197
43 
44 namespace NPAPIClient {
45 
PluginGetURLTest(NPP id,NPNetscapeFuncs * host_functions)46 PluginGetURLTest::PluginGetURLTest(NPP id, NPNetscapeFuncs *host_functions)
47   : PluginTest(id, host_functions),
48     tests_started_(false),
49     tests_in_progress_(0),
50     test_file_(NULL),
51     expect_404_response_(false),
52     npn_evaluate_context_(false),
53     handle_url_redirects_(false),
54     received_url_redirect_cancel_notification_(false),
55     received_url_redirect_allow_notification_(false),
56     check_cookies_(false) {
57 }
58 
~PluginGetURLTest()59 PluginGetURLTest::~PluginGetURLTest() {}
60 
New(uint16 mode,int16 argc,const char * argn[],const char * argv[],NPSavedData * saved)61 NPError PluginGetURLTest::New(uint16 mode, int16 argc, const char* argn[],
62                               const char* argv[], NPSavedData* saved) {
63   const char* page_not_found_url = GetArgValue("page_not_found_url", argc,
64                                                argn, argv);
65   if (page_not_found_url) {
66     page_not_found_url_ = page_not_found_url;
67     expect_404_response_ = true;
68   }
69 
70   const char* fail_write_url = GetArgValue("fail_write_url", argc,
71                                            argn, argv);
72   if (fail_write_url) {
73     fail_write_url_ = fail_write_url;
74   }
75 
76   const char* referrer_target_url = GetArgValue("ref_target", argc,
77                                                 argn, argv);
78   if (referrer_target_url) {
79     referrer_target_url_ = referrer_target_url;
80   }
81 
82   if (!base::strcasecmp(GetArgValue("name", argc, argn, argv),
83                         "geturlredirectnotify")) {
84     handle_url_redirects_ = true;
85   }
86 
87   NPError error = PluginTest::New(mode, argc, argn, argv, saved);
88 
89   // The above sets test_name().
90   if (test_name() == "cookies")
91     check_cookies_ = true;
92 
93   return error;
94 }
95 
SetWindow(NPWindow * pNPWindow)96 NPError PluginGetURLTest::SetWindow(NPWindow* pNPWindow) {
97 #if !defined(OS_MACOSX)
98   if (pNPWindow->window == NULL)
99     return NPERR_NO_ERROR;
100 #endif
101 
102   if (!tests_started_) {
103     tests_started_ = true;
104 
105     tests_in_progress_++;
106 
107     if (expect_404_response_) {
108       HostFunctions()->geturl(id(), page_not_found_url_.c_str(), NULL);
109       return NPERR_NO_ERROR;
110     } else if (!fail_write_url_.empty()) {
111       HostFunctions()->geturl(id(), fail_write_url_.c_str(), NULL);
112       return NPERR_NO_ERROR;
113     } else if (!referrer_target_url_.empty()) {
114       HostFunctions()->pushpopupsenabledstate(id(), true);
115       HostFunctions()->geturl(id(), referrer_target_url_.c_str(), "_blank");
116       HostFunctions()->poppopupsenabledstate(id());
117       return NPERR_NO_ERROR;
118     } else if (handle_url_redirects_) {
119       HostFunctions()->geturlnotify(
120           id(), REDIRECT_SRC_URL, NULL,
121           reinterpret_cast<void*>(REDIRECT_SRC_URL_NOTIFICATION_CANCEL_ID));
122       return NPERR_NO_ERROR;
123     } else if (check_cookies_) {
124       HostFunctions()->geturlnotify(
125           id(),
126           "plugin_ref_target_page.html",
127           NULL,
128           reinterpret_cast<void*>(SELF_URL_STREAM_ID));
129       return NPERR_NO_ERROR;
130     }
131 
132     std::string url = SELF_URL;
133     HostFunctions()->geturlnotify(id(), url.c_str(), NULL,
134                                   reinterpret_cast<void*>(SELF_URL_STREAM_ID));
135 
136     tests_in_progress_++;
137     std::string bogus_url = BOGUS_URL;
138     HostFunctions()->geturlnotify(id(), bogus_url.c_str(), NULL,
139                                   reinterpret_cast<void*>(BOGUS_URL_STREAM_ID));
140   }
141   return NPERR_NO_ERROR;
142 }
143 
NewStream(NPMIMEType type,NPStream * stream,NPBool seekable,uint16 * stype)144 NPError PluginGetURLTest::NewStream(NPMIMEType type, NPStream* stream,
145                               NPBool seekable, uint16* stype) {
146   if (stream == NULL) {
147     SetError("NewStream got null stream");
148     return NPERR_INVALID_PARAM;
149   }
150 
151   if (test_completed()) {
152     return PluginTest::NewStream(type, stream, seekable, stype);
153   }
154 
155   if (!referrer_target_url_.empty()) {
156     return NPERR_NO_ERROR;
157   }
158 
159   COMPILE_ASSERT(sizeof(unsigned long) <= sizeof(stream->notifyData),
160                  cast_validity_check);
161 
162   if (expect_404_response_) {
163     NPObject *window_obj = NULL;
164     HostFunctions()->getvalue(id(), NPNVWindowNPObject, &window_obj);
165     if (!window_obj) {
166       SetError("Failed to get NPObject for plugin instance2");
167       SignalTestCompleted();
168       return NPERR_NO_ERROR;
169     }
170 
171     std::string script = "javascript:document.title=\"OK\"";
172     NPString script_string;
173     script_string.UTF8Characters = script.c_str();
174     script_string.UTF8Length = static_cast<unsigned int>(script.length());
175     NPVariant result_var;
176 
177     npn_evaluate_context_ = true;
178     HostFunctions()->evaluate(id(), window_obj, &script_string, &result_var);
179     npn_evaluate_context_ = false;
180     return NPERR_NO_ERROR;
181   }
182 
183   if (!fail_write_url_.empty() || check_cookies_) {
184     return NPERR_NO_ERROR;
185   }
186 
187 
188   unsigned long stream_id = reinterpret_cast<unsigned long>(
189       stream->notifyData);
190 
191   switch (stream_id) {
192     case SELF_URL_STREAM_ID:
193       break;
194     case FETCHED_URL_STREAM_ID:
195       {
196         std::string filename = self_url_;
197         if (filename.find("file:///", 0) != 0) {
198           SetError("Test expects a file-url.");
199           break;
200         }
201 
202         // TODO(evanm): use the net:: functions to convert file:// URLs to
203         // on-disk file paths.  But it probably doesn't actually matter in
204         // this test.
205 
206 #if defined(OS_WIN)
207         filename = filename.substr(8);  // remove "file:///"
208         // Assume an ASCII path on Windows.
209         base::FilePath path = base::FilePath(base::ASCIIToWide(filename));
210 #else
211         filename = filename.substr(7);  // remove "file://"
212         base::FilePath path = base::FilePath(filename);
213 #endif
214 
215         test_file_ = base::OpenFile(path, "r");
216         if (!test_file_) {
217           SetError("Could not open source file");
218         }
219       }
220       break;
221     case BOGUS_URL_STREAM_ID:
222       SetError("Unexpected NewStream for BOGUS_URL");
223       break;
224     case REDIRECT_SRC_URL_NOTIFICATION_CANCEL_ID:
225       SetError("Should not redirect to URL when plugin denied it.");
226       break;
227     case REDIRECT_SRC_URL_NOTIFICATION_ALLOW_ID:
228       break;
229     default:
230       SetError("Unexpected NewStream callback");
231       break;
232   }
233   return NPERR_NO_ERROR;
234 }
235 
WriteReady(NPStream * stream)236 int32 PluginGetURLTest::WriteReady(NPStream *stream) {
237   if (test_completed()) {
238     return PluginTest::WriteReady(stream);
239   }
240 
241   if (!referrer_target_url_.empty() || check_cookies_) {
242     return STREAM_CHUNK;
243   }
244 
245   COMPILE_ASSERT(sizeof(unsigned long) <= sizeof(stream->notifyData),
246                  cast_validity_check);
247   unsigned long stream_id = reinterpret_cast<unsigned long>(
248       stream->notifyData);
249   if (stream_id == BOGUS_URL_STREAM_ID)
250     SetError("Received WriteReady for BOGUS_URL");
251 
252   return STREAM_CHUNK;
253 }
254 
Write(NPStream * stream,int32 offset,int32 len,void * buffer)255 int32 PluginGetURLTest::Write(NPStream *stream, int32 offset, int32 len,
256                               void *buffer) {
257   if (test_completed()) {
258     return PluginTest::Write(stream, offset, len, buffer);
259   }
260 
261   if (!fail_write_url_.empty()) {
262     SignalTestCompleted();
263     return -1;
264   }
265 
266   if (!referrer_target_url_.empty() || check_cookies_) {
267     return len;
268   }
269 
270   if (stream == NULL) {
271     SetError("Write got null stream");
272     return -1;
273   }
274   if (len < 0 || len > STREAM_CHUNK) {
275     SetError("Write got bogus stream chunk size");
276     return -1;
277   }
278 
279   COMPILE_ASSERT(sizeof(unsigned long) <= sizeof(stream->notifyData),
280                  cast_validity_check);
281   unsigned long stream_id = reinterpret_cast<unsigned long>(
282       stream->notifyData);
283   switch (stream_id) {
284     case SELF_URL_STREAM_ID:
285       self_url_.append(static_cast<char*>(buffer), len);
286       break;
287     case FETCHED_URL_STREAM_ID:
288       {
289         char read_buffer[STREAM_CHUNK];
290         int32 bytes =
291             static_cast<int32>(fread(read_buffer, 1, len, test_file_));
292         // Technically, fread could return fewer than len
293         // bytes.  But this is not likely.
294         if (bytes != len)
295           SetError("Did not read correct bytelength from source file");
296         if (memcmp(read_buffer, buffer, len))
297           SetError("Content mismatch between data and source!");
298       }
299       break;
300     case BOGUS_URL_STREAM_ID:
301       SetError("Unexpected write callback for BOGUS_URL");
302       break;
303     case REDIRECT_SRC_URL_NOTIFICATION_ALLOW_ID:
304       break;
305     default:
306       SetError("Unexpected write callback");
307       break;
308   }
309   // Pretend that we took all the data.
310   return len;
311 }
312 
313 
DestroyStream(NPStream * stream,NPError reason)314 NPError PluginGetURLTest::DestroyStream(NPStream *stream, NPError reason) {
315   if (test_completed()) {
316     return PluginTest::DestroyStream(stream, reason);
317   }
318 
319   if (stream == NULL) {
320     SetError("NewStream got null stream");
321     return NPERR_INVALID_PARAM;
322   }
323 
324   COMPILE_ASSERT(sizeof(unsigned long) <= sizeof(stream->notifyData),
325                  cast_validity_check);
326 
327   if (expect_404_response_) {
328     if (npn_evaluate_context_) {
329       SetError("Received destroyStream in the context of NPN_Evaluate.");
330     }
331 
332     SignalTestCompleted();
333     return NPERR_NO_ERROR;
334   }
335 
336   if (!referrer_target_url_.empty()) {
337     return NPERR_NO_ERROR;
338   }
339 
340   if (check_cookies_) {
341     SignalTestCompleted();
342     return NPERR_NO_ERROR;
343   }
344 
345   unsigned long stream_id =
346       reinterpret_cast<unsigned long>(stream->notifyData);
347   switch (stream_id) {
348     case SELF_URL_STREAM_ID:
349       // don't care
350       break;
351     case REDIRECT_SRC_URL_NOTIFICATION_ALLOW_ID:
352       break;
353     case FETCHED_URL_STREAM_ID:
354       {
355         char read_buffer[STREAM_CHUNK];
356         size_t bytes = fread(read_buffer, 1, sizeof(read_buffer), test_file_);
357         if (bytes != 0)
358           SetError("Data and source mismatch on length");
359         base::CloseFile(test_file_);
360       }
361       break;
362     default:
363       SetError("Unexpected NewStream callback");
364       break;
365   }
366   return NPERR_NO_ERROR;
367 }
368 
StreamAsFile(NPStream * stream,const char * fname)369 void PluginGetURLTest::StreamAsFile(NPStream* stream, const char* fname) {
370   if (stream == NULL) {
371     SetError("NewStream got null stream");
372     return;
373   }
374 
375   COMPILE_ASSERT(sizeof(unsigned long) <= sizeof(stream->notifyData),
376                  cast_validity_check);
377   unsigned long stream_id =
378       reinterpret_cast<unsigned long>(stream->notifyData);
379   switch (stream_id) {
380     case SELF_URL_STREAM_ID:
381       // don't care
382       break;
383     default:
384       SetError("Unexpected NewStream callback");
385       break;
386   }
387 }
388 
URLNotify(const char * url,NPReason reason,void * data)389 void PluginGetURLTest::URLNotify(const char* url, NPReason reason, void* data) {
390   if (!tests_in_progress_) {
391     SetError("URLNotify received after tests completed");
392     return;
393   }
394 
395   if (!url) {
396     SetError("URLNotify received NULL url");
397     return;
398   }
399 
400   if (check_cookies_)
401     return;
402 
403   COMPILE_ASSERT(sizeof(unsigned long) <= sizeof(data), cast_validity_check);
404   unsigned long stream_id = reinterpret_cast<unsigned long>(data);
405   switch (stream_id) {
406     case SELF_URL_STREAM_ID:
407       if (strcmp(url, SELF_URL) != 0)
408         SetError("URLNotify reported incorrect url for SELF_URL");
409 
410       // We have our stream url.  Go fetch it.
411       HostFunctions()->geturlnotify(id(), self_url_.c_str(), NULL,
412                             reinterpret_cast<void*>(FETCHED_URL_STREAM_ID));
413       break;
414     case FETCHED_URL_STREAM_ID:
415       if (!url || strcmp(url, self_url_.c_str()) != 0)
416         SetError("URLNotify reported incorrect url for FETCHED_URL");
417       tests_in_progress_--;
418       break;
419     case BOGUS_URL_STREAM_ID:
420       if (reason != NPRES_NETWORK_ERR) {
421         std::string err = "BOGUS_URL received unexpected URLNotify status: ";
422         err.append(base::IntToString(reason));
423         SetError(err);
424       }
425       tests_in_progress_--;
426       break;
427     case REDIRECT_SRC_URL_NOTIFICATION_CANCEL_ID: {
428       if (!received_url_redirect_cancel_notification_) {
429         SetError("Failed to receive URLRedirect notification for cancel");
430       }
431       if (reason != NPRES_NETWORK_ERR)  {
432         SetError("Redirected URL didn't get canceled");
433       }
434       break;
435     }
436     case REDIRECT_SRC_URL_NOTIFICATION_ALLOW_ID: {
437       if (!received_url_redirect_allow_notification_) {
438         SetError("Failed to receive URLRedirect notification for allow");
439       }
440       if (reason != NPRES_DONE) {
441         SetError("Redirected URL didn't complete successfully");
442       }
443       tests_in_progress_--;
444       break;
445     }
446     default:
447       SetError("Unexpected NewStream callback");
448       break;
449   }
450 
451   if (tests_in_progress_ == 0)
452       SignalTestCompleted();
453 }
454 
URLRedirectNotify(const char * url,int32_t status,void * notify_data)455 void PluginGetURLTest::URLRedirectNotify(const char* url,
456                                          int32_t status,
457                                          void* notify_data) {
458   unsigned long stream_id = reinterpret_cast<unsigned long>(notify_data);
459   if (stream_id == REDIRECT_SRC_URL_NOTIFICATION_CANCEL_ID) {
460     if (!base::strcasecmp(url,
461                           "http://mock.http/npapi/plugin_read_page.html")) {
462       received_url_redirect_cancel_notification_ = true;
463       // Disallow redirect notification.
464       HostFunctions()->urlredirectresponse(id(), notify_data, false);
465 
466       // Now start a request that we will allow to redirect.
467       HostFunctions()->geturlnotify(
468         id(), REDIRECT_SRC_URL, NULL,
469         reinterpret_cast<void*>(REDIRECT_SRC_URL_NOTIFICATION_ALLOW_ID));
470     }
471   } else if (stream_id == REDIRECT_SRC_URL_NOTIFICATION_ALLOW_ID) {
472     received_url_redirect_allow_notification_ = true;
473     HostFunctions()->urlredirectresponse(id(), notify_data, true);
474   }
475 }
476 
477 } // namespace NPAPIClient
478