// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "base/bind.h" #include "base/file_util.h" #include "base/json/json_file_value_serializer.h" #include "base/message_loop.h" #include "base/path_service.h" #include "base/scoped_temp_dir.h" #include "base/synchronization/waitable_event.h" #include "base/utf_string_conversions.h" #include "chrome/browser/extensions/test_extension_service.h" #include "chrome/browser/intents/default_web_intent_service.h" #include "chrome/browser/intents/web_intents_registry.h" #include "chrome/browser/webdata/web_data_service.h" #include "chrome/common/chrome_paths.h" #include "chrome/common/extensions/extension_set.h" #include "content/public/test/test_browser_thread.h" #include "testing/gmock/include/gmock/gmock.h" #include "testing/gtest/include/gtest/gtest.h" using content::BrowserThread; using extensions::Extension; using webkit_glue::WebIntentServiceData; class MockExtensionService: public TestExtensionService { public: virtual ~MockExtensionService() {} MOCK_CONST_METHOD0(extensions, const ExtensionSet*()); MOCK_CONST_METHOD2(GetExtensionById, const Extension*(const std::string&, bool)); }; namespace { // TODO(groby): Unify loading functions with extension_manifest_unittest code. DictionaryValue* LoadManifestFile(const FilePath& path, std::string* error) { EXPECT_TRUE(file_util::PathExists(path)); JSONFileValueSerializer serializer(path); return static_cast(serializer.Deserialize(NULL, error)); } scoped_refptr LoadExtensionWithLocation( const std::string& name, Extension::Location location, std::string* error) { FilePath path; PathService::Get(chrome::DIR_TEST_DATA, &path); path = path.AppendASCII("extensions") .AppendASCII("manifest_tests") .AppendASCII(name.c_str()); scoped_ptr value(LoadManifestFile(path, error)); if (!value.get()) return NULL; return Extension::Create(path.DirName(), location, *value, Extension::NO_FLAGS, Extension::GenerateIdForPath(path), error); } scoped_refptr LoadExtension(const std::string& name, std::string* error) { return LoadExtensionWithLocation(name, Extension::INTERNAL, error); } scoped_refptr LoadAndExpectSuccess(const std::string& name) { std::string error; scoped_refptr extension = LoadExtension(name, &error); EXPECT_TRUE(extension) << name; EXPECT_EQ("", error) << name; return extension; } } // namespace class WebIntentsRegistryTest : public testing::Test { public: WebIntentsRegistryTest() : ui_thread_(BrowserThread::UI, &message_loop_), db_thread_(BrowserThread::DB) {} protected: virtual void SetUp() { CommandLine::ForCurrentProcess()->AppendSwitch("--enable-web-intents"); db_thread_.Start(); wds_ = new WebDataService(); ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); wds_->Init(temp_dir_.path()); registry_.Initialize(wds_, &extension_service_); EXPECT_CALL(extension_service_, extensions()). WillRepeatedly(testing::Return(&extensions_)); EXPECT_CALL(extension_service_, GetExtensionById(testing::_, testing::_)). WillRepeatedly( testing::Invoke(this, &WebIntentsRegistryTest::GetExtensionById)); } virtual void TearDown() { // Clear all references to wds to force it destruction. wds_->ShutdownOnUIThread(); wds_ = NULL; // Schedule another task on the DB thread to notify us that it's safe to // carry on with the test. base::WaitableEvent done(false, false); BrowserThread::PostTask(BrowserThread::DB, FROM_HERE, base::Bind(&base::WaitableEvent::Signal, base::Unretained(&done))); done.Wait(); db_thread_.Stop(); MessageLoop::current()->PostTask(FROM_HERE, MessageLoop::QuitClosure()); MessageLoop::current()->Run(); } const Extension* GetExtensionById(const std::string& extension_id, testing::Unused) { for (ExtensionSet::const_iterator iter = extensions_.begin(); iter != extensions_.end(); ++iter) { if ((*iter)->id() == extension_id) return &**iter; } return NULL; } MessageLoopForUI message_loop_; content::TestBrowserThread ui_thread_; content::TestBrowserThread db_thread_; scoped_refptr wds_; MockExtensionService extension_service_; ExtensionSet extensions_; WebIntentsRegistry registry_; ScopedTempDir temp_dir_; }; // Simple consumer for WebIntentsRegistry notifications. Stores result data and // terminates UI thread when callback is invoked. class TestConsumer { public: void OnIntentsQueryDone( const std::vector& services) { services_ = services; DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); MessageLoop::current()->Quit(); } bool ServicesContains(const webkit_glue::WebIntentServiceData& data) { for (size_t i = 0; i < services_.size(); ++i) { if (services_[i] == data) return true; } return false; } void OnIntentsDefaultsQueryDone( const DefaultWebIntentService& default_service) { default_ = default_service; DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); MessageLoop::current()->Quit(); } // Wait for the UI message loop to terminate - happens when OnIntesQueryDone // is invoked. void WaitForData() { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); MessageLoop::current()->Run(); } // Result data from callback. std::vector services_; // Result default data from callback. DefaultWebIntentService default_; }; TEST_F(WebIntentsRegistryTest, BasicTests) { webkit_glue::WebIntentServiceData service; service.service_url = GURL("http://google.com"); service.action = ASCIIToUTF16("share"); service.type = ASCIIToUTF16("image/*"); service.title = ASCIIToUTF16("Google's Sharing Service"); registry_.RegisterIntentService(service); service.type = ASCIIToUTF16("video/*"); service.title = ASCIIToUTF16("Second Service"); registry_.RegisterIntentService(service); service.action = ASCIIToUTF16("search"); registry_.RegisterIntentService(service); TestConsumer consumer; registry_.GetIntentServices(ASCIIToUTF16("share"), ASCIIToUTF16("*"), base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); EXPECT_EQ(2U, consumer.services_.size()); registry_.GetIntentServices(ASCIIToUTF16("search"), ASCIIToUTF16("*"), base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); EXPECT_EQ(1U, consumer.services_.size()); service.action = ASCIIToUTF16("share"); service.type = ASCIIToUTF16("image/*"); registry_.UnregisterIntentService(service); registry_.GetIntentServices(ASCIIToUTF16("share"), ASCIIToUTF16("*"), base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); EXPECT_EQ(1U, consumer.services_.size()); } TEST_F(WebIntentsRegistryTest, GetIntentServicesForExtensionFilter) { scoped_refptr share_extension( LoadAndExpectSuccess("intent_valid.json")); scoped_refptr edit_extension( LoadAndExpectSuccess("intent_valid_2.json")); extensions_.Insert(share_extension); extensions_.Insert(edit_extension); ASSERT_EQ(2U, extensions_.size()); TestConsumer consumer; registry_.GetIntentServicesForExtensionFilter( ASCIIToUTF16("http://webintents.org/edit"), ASCIIToUTF16("image/*"), edit_extension->id(), base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); ASSERT_EQ(1U, consumer.services_.size()); EXPECT_EQ(edit_extension->url().spec() + "services/edit", consumer.services_[0].service_url.spec()); } TEST_F(WebIntentsRegistryTest, GetAllIntents) { webkit_glue::WebIntentServiceData service; service.service_url = GURL("http://google.com"); service.action = ASCIIToUTF16("share"); service.type = ASCIIToUTF16("image/*"); service.title = ASCIIToUTF16("Google's Sharing Service"); registry_.RegisterIntentService(service); service.action = ASCIIToUTF16("search"); registry_.RegisterIntentService(service); TestConsumer consumer; registry_.GetAllIntentServices(base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); ASSERT_EQ(2U, consumer.services_.size()); if (consumer.services_[0].action != ASCIIToUTF16("share")) std::swap(consumer.services_[0],consumer.services_[1]); service.action = ASCIIToUTF16("share"); EXPECT_EQ(service, consumer.services_[0]); service.action = ASCIIToUTF16("search"); EXPECT_EQ(service, consumer.services_[1]); } TEST_F(WebIntentsRegistryTest, GetExtensionIntents) { extensions_.Insert(LoadAndExpectSuccess("intent_valid.json")); extensions_.Insert(LoadAndExpectSuccess("intent_valid_2.json")); ASSERT_EQ(2U, extensions_.size()); TestConsumer consumer; registry_.GetAllIntentServices(base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); ASSERT_EQ(2U, consumer.services_.size()); } TEST_F(WebIntentsRegistryTest, GetSomeExtensionIntents) { extensions_.Insert(LoadAndExpectSuccess("intent_valid.json")); extensions_.Insert(LoadAndExpectSuccess("intent_valid_2.json")); ASSERT_EQ(2U, extensions_.size()); TestConsumer consumer; registry_.GetIntentServices(ASCIIToUTF16("http://webintents.org/edit"), ASCIIToUTF16("*"), base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); ASSERT_EQ(1U, consumer.services_.size()); } TEST_F(WebIntentsRegistryTest, GetIntentsFromMixedSources) { extensions_.Insert(LoadAndExpectSuccess("intent_valid.json")); extensions_.Insert(LoadAndExpectSuccess("intent_valid_2.json")); ASSERT_EQ(2U, extensions_.size()); webkit_glue::WebIntentServiceData service; service.service_url = GURL("http://somewhere.com/intent/edit.html"); service.action = ASCIIToUTF16("http://webintents.org/edit"); service.type = ASCIIToUTF16("image/*"); service.title = ASCIIToUTF16("Image Editing Service"); registry_.RegisterIntentService(service); TestConsumer consumer; registry_.GetIntentServices( ASCIIToUTF16("http://webintents.org/edit"), ASCIIToUTF16("*"), base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); ASSERT_EQ(2U, consumer.services_.size()); registry_.GetIntentServices(ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("*"), base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); ASSERT_EQ(1U, consumer.services_.size()); } TEST_F(WebIntentsRegistryTest, GetIntentsWithMimeAndLiteralMatching) { WebIntentServiceData services[] = { WebIntentServiceData(GURL("http://elsewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("image/*"), ASCIIToUTF16("Image Sharing Service")), WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("image/jpeg"), ASCIIToUTF16("Specific Image Editing Service")), WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("text/uri-list"), ASCIIToUTF16("Text Link Sharing Service")), WebIntentServiceData(GURL("http://somewhere2.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("text/plain"), ASCIIToUTF16("Text Sharing Service")), WebIntentServiceData(GURL("http://elsewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("elsewhere"), ASCIIToUTF16("Text Sharing Service")), WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("somewhere"), ASCIIToUTF16("Text Sharing Service")), WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("nota/*"), ASCIIToUTF16("Text Sharing Service")), WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("*nomime"), ASCIIToUTF16("Text Sharing Service")), WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("*/nomime"), ASCIIToUTF16("Text Sharing Service")), WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("*/*nomime"), ASCIIToUTF16("Text Sharing Service")), WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("*/*/nomime"), ASCIIToUTF16("Text Sharing Service")), WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("nomime/*"), ASCIIToUTF16("Text Sharing Service")), WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("x-type/*"), ASCIIToUTF16("Text Sharing Service")), WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("x-/*"), // actually a string literal ASCIIToUTF16("Text Sharing Service")) }; registry_.RegisterIntentService(services[0]); registry_.RegisterIntentService(services[1]); registry_.RegisterIntentService(services[2]); registry_.RegisterIntentService(services[3]); registry_.RegisterIntentService(services[4]); registry_.RegisterIntentService(services[5]); registry_.RegisterIntentService(services[6]); registry_.RegisterIntentService(services[7]); registry_.RegisterIntentService(services[8]); registry_.RegisterIntentService(services[9]); registry_.RegisterIntentService(services[10]); registry_.RegisterIntentService(services[11]); registry_.RegisterIntentService(services[12]); registry_.RegisterIntentService(services[13]); TestConsumer consumer; // Test specific match on both sides. registry_.GetIntentServices(ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("text/uri-list"), base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); ASSERT_EQ(1U, consumer.services_.size()); EXPECT_EQ(services[2], consumer.services_[0]); // Test specific query, wildcard registration. registry_.GetIntentServices(ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("image/png"), base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); ASSERT_EQ(1U, consumer.services_.size()); EXPECT_EQ(services[0], consumer.services_[0]); // Test wildcard query, specific registration. registry_.GetIntentServices(ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("text/*"), base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); ASSERT_EQ(2U, consumer.services_.size()); EXPECT_EQ(services[2], consumer.services_[0]); EXPECT_EQ(services[3], consumer.services_[1]); // Test wildcard query, wildcard registration. registry_.GetIntentServices(ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("image/*"), base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); ASSERT_EQ(2U, consumer.services_.size()); EXPECT_EQ(services[0], consumer.services_[0]); EXPECT_EQ(services[1], consumer.services_[1]); // Test "catch-all" query. registry_.GetIntentServices(ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("*"), base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); ASSERT_EQ(5U, consumer.services_.size()); EXPECT_TRUE(consumer.ServicesContains(services[0])); EXPECT_TRUE(consumer.ServicesContains(services[1])); EXPECT_TRUE(consumer.ServicesContains(services[2])); EXPECT_TRUE(consumer.ServicesContains(services[3])); EXPECT_TRUE(consumer.ServicesContains(services[12])); // Test retrieve string literal match. registry_.GetIntentServices( ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("elsewhere"), base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); ASSERT_EQ(1U, consumer.services_.size()); EXPECT_EQ(services[4], consumer.services_[0]); // Test retrieve MIME-looking type but actually isn't // doesn't wildcard match. registry_.GetIntentServices( ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("nota/mimetype"), base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); ASSERT_EQ(0U, consumer.services_.size()); // Also a MIME-ish type that actually isn't. registry_.GetIntentServices( ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("x-/mimetype"), base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); ASSERT_EQ(0U, consumer.services_.size()); // Extension MIME type will match wildcard. registry_.GetIntentServices( ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("x-type/mimetype"), base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); ASSERT_EQ(1U, consumer.services_.size()); } TEST_F(WebIntentsRegistryTest, TestGetDefaults) { DefaultWebIntentService default_service; default_service.action = ASCIIToUTF16("share"); default_service.type = ASCIIToUTF16("text/*"); // Values here are just dummies to test for preservation. default_service.user_date = 1; default_service.suppression = 4; default_service.service_url = "service_url"; registry_.RegisterDefaultIntentService(default_service); TestConsumer consumer; // Test we can retrieve default entries by action. registry_.GetDefaultIntentService( ASCIIToUTF16("share"), ASCIIToUTF16("text/plain"), GURL("http://www.google.com/"), base::Bind(&TestConsumer::OnIntentsDefaultsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); EXPECT_EQ("service_url", consumer.default_.service_url); EXPECT_EQ(1, consumer.default_.user_date); EXPECT_EQ(4, consumer.default_.suppression); // Test that no action match means we don't retrieve any // default entries. consumer.default_ = DefaultWebIntentService(); ASSERT_EQ("", consumer.default_.service_url); registry_.GetDefaultIntentService( ASCIIToUTF16("no-share"), ASCIIToUTF16("text/plain"), GURL("http://www.google.com/"), base::Bind(&TestConsumer::OnIntentsDefaultsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); EXPECT_EQ("", consumer.default_.service_url); // Test that no type match means we don't retrieve any // default entries (they get filtered out). consumer.default_ = DefaultWebIntentService(); ASSERT_EQ("", consumer.default_.service_url); registry_.GetDefaultIntentService( ASCIIToUTF16("share"), ASCIIToUTF16("image/plain"), GURL("http://www.google.com/"), base::Bind(&TestConsumer::OnIntentsDefaultsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); EXPECT_EQ("", consumer.default_.service_url); // Check that a string-literal type won't match. consumer.default_ = DefaultWebIntentService(); ASSERT_EQ("", consumer.default_.service_url); registry_.GetDefaultIntentService( ASCIIToUTF16("share"), ASCIIToUTF16("literal"), GURL("http://www.google.com/"), base::Bind(&TestConsumer::OnIntentsDefaultsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); EXPECT_EQ("", consumer.default_.service_url); } // Verify that collapsing equivalent intents works properly. TEST_F(WebIntentsRegistryTest, CollapseIntents) { WebIntentsRegistry::IntentServiceList services; // Add two intents with identical |service_url|, |title|, and |action|. services.push_back( WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("image/png"), ASCIIToUTF16("Image Sharing Service"))); services.push_back( WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("image/jpg"), ASCIIToUTF16("Image Sharing Service"))); // Service that differs in disposition. services.push_back( WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("image/png"), ASCIIToUTF16("Image Sharing Service"))); ASSERT_EQ(WebIntentServiceData::DISPOSITION_WINDOW, services.back().disposition); services.back().disposition = WebIntentServiceData::DISPOSITION_INLINE; // Service that differs in title. services.push_back( WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("image/png"), ASCIIToUTF16("Sharing Service"))); // Service that differs in |action|. services.push_back( WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share-old"), ASCIIToUTF16("image/png"), ASCIIToUTF16("Image Sharing Service"))); // Service that differs in |service_url|. services.push_back( WebIntentServiceData(GURL("http://zoo.com/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("image/png"), ASCIIToUTF16("Image Sharing Service"))); // Only the first two services should be collapsed. registry_.CollapseIntents(&services); ASSERT_EQ(5UL, services.size()); // Joined services have their mime types combined EXPECT_EQ(ASCIIToUTF16("image/png,image/jpg"), services[0].type); // Verify the remaining services via distinguishing characteristics. EXPECT_EQ(WebIntentServiceData::DISPOSITION_INLINE, services[1].disposition); EXPECT_EQ(ASCIIToUTF16("Sharing Service"), services[2].title); EXPECT_EQ(ASCIIToUTF16("http://webintents.org/share-old"), services[3].action); EXPECT_EQ(GURL("http://zoo.com/share.html").spec(), services[4].service_url.spec()); } // Verify that GetIntentServices collapses equivalent intents. TEST_F(WebIntentsRegistryTest, GetIntentsCollapsesEquivalentIntents) { WebIntentServiceData services[] = { WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("image/png"), ASCIIToUTF16("Image Sharing Service")), WebIntentServiceData(GURL("http://somewhere.com/intent/share.html"), ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("image/jpg"), ASCIIToUTF16("Image Sharing Service")) }; registry_.RegisterIntentService(services[0]); registry_.RegisterIntentService(services[1]); TestConsumer consumer; registry_.GetIntentServices(ASCIIToUTF16("http://webintents.org/share"), ASCIIToUTF16("image/*"), base::Bind(&TestConsumer::OnIntentsQueryDone, base::Unretained(&consumer))); consumer.WaitForData(); ASSERT_EQ(1U, consumer.services_.size()); EXPECT_EQ(ASCIIToUTF16("image/png,image/jpg"), consumer.services_[0].type); }