blob: 0685fc566a0fadd3f26a8d3986aad5e30046b3e2 [file] [log] [blame]
/*
* Copyright (C) 2012, 2014 Igalia S.L.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "UserAgentGtk.h"
#include "URL.h"
#include <wtf/NeverDestroyed.h>
#include <wtf/text/StringBuilder.h>
#if OS(UNIX)
#include <sys/utsname.h>
#endif
namespace WebCore {
class UserAgentQuirks {
public:
enum UserAgentQuirk {
NeedsMacintoshPlatform,
NumUserAgentQuirks
};
UserAgentQuirks()
: m_quirks(0)
{
COMPILE_ASSERT(sizeof(m_quirks) * 8 >= NumUserAgentQuirks, not_enough_room_for_quirks);
}
void add(UserAgentQuirk quirk)
{
ASSERT(quirk >= 0);
ASSERT_WITH_SECURITY_IMPLICATION(quirk < NumUserAgentQuirks);
m_quirks |= (1 << quirk);
}
bool contains(UserAgentQuirk quirk) const
{
return m_quirks & (1 << quirk);
}
bool isEmpty() const { return !m_quirks; }
private:
uint32_t m_quirks;
};
static const char* cpuDescriptionForUAString()
{
#if CPU(PPC) || CPU(PPC64)
return "PPC";
#elif CPU(X86) || CPU(X86_64)
return "Intel";
#elif CPU(ARM) || CPU(ARM64)
return "ARM";
#else
return "Unknown";
#endif
}
static const char* platformForUAString()
{
#if PLATFORM(X11)
return "X11";
#elif OS(WINDOWS)
return "";
#elif PLATFORM(MAC)
return "Macintosh";
#elif defined(GDK_WINDOWING_DIRECTFB)
return "DirectFB";
#else
return "Unknown";
#endif
}
static const String platformVersionForUAString()
{
#if OS(UNIX)
struct utsname name;
uname(&name);
static NeverDestroyed<const String> uaOSVersion(String::format("%s %s", name.sysname, name.machine));
return uaOSVersion;
#else
// We will always claim to be Safari in Mac OS X, since Safari in Linux triggers the iOS path on some websites.
static NeverDestroyed<const String> uaOSVersion(String::format("%s Mac OS X", cpuDescriptionForUAString()));
return uaOSVersion;
#endif
}
static const char* versionForUAString()
{
return USER_AGENT_GTK_MAJOR_VERSION "." USER_AGENT_GTK_MINOR_VERSION;
}
static String buildUserAgentString(const UserAgentQuirks& quirks)
{
StringBuilder uaString;
uaString.appendLiteral("Mozilla/5.0 ");
uaString.append('(');
if (quirks.contains(UserAgentQuirks::NeedsMacintoshPlatform))
uaString.appendLiteral("Macintosh");
else
uaString.append(platformForUAString());
uaString.appendLiteral("; ");
if (quirks.contains(UserAgentQuirks::NeedsMacintoshPlatform)) {
uaString.append(cpuDescriptionForUAString());
uaString.appendLiteral(" Mac OS X");
} else
uaString.append(platformVersionForUAString());
uaString.appendLiteral(") AppleWebKit/");
uaString.append(versionForUAString());
// Version/X is mandatory *before* Safari/X to be a valid Safari UA. See
// https://bugs.webkit.org/show_bug.cgi?id=133403 for details.
uaString.appendLiteral(" (KHTML, like Gecko) Version/8.0 Safari/");
uaString.append(versionForUAString());
return uaString.toString();
}
static const String standardUserAgentStatic()
{
static NeverDestroyed<const String> uaStatic(buildUserAgentString(UserAgentQuirks()));
return uaStatic;
}
String standardUserAgent(const String& applicationName, const String& applicationVersion)
{
// Create a default user agent string with a liberal interpretation of
// https://developer.mozilla.org/en-US/docs/User_Agent_Strings_Reference
//
// Forming a functional user agent is really difficult. We must mention Safari, because some
// sites check for that when detecting WebKit browsers. Additionally some sites assume that
// browsers that are "Safari" but not running on OS X are the Safari iOS browse. Getting this
// wrong can cause sites to load the wrong JavaScript, CSS, or custom fonts. In some cases
// sites won't load resources at all.
if (applicationName.isEmpty())
return standardUserAgentStatic();
String finalApplicationVersion = applicationVersion;
if (finalApplicationVersion.isEmpty())
finalApplicationVersion = versionForUAString();
return standardUserAgentStatic() + ' ' + applicationName + '/' + finalApplicationVersion;
}
String standardUserAgentForURL(const URL& url)
{
ASSERT(!url.isNull());
UserAgentQuirks quirks;
if (url.host().endsWith(".yahoo.com")) {
// www.yahoo.com redirects to the mobile version when Linux is present in the UA,
// use always Macintosh as platform. See https://bugs.webkit.org/show_bug.cgi?id=125444.
quirks.add(UserAgentQuirks::NeedsMacintoshPlatform);
}
// The null string means we don't need a specific UA for the given URL.
return quirks.isEmpty() ? String() : buildUserAgentString(quirks);
}
} // namespace WebCore