// Copyright 2016 The PDFium Authors // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com #include "core/fpdfdoc/cpdf_filespec.h" #include #include #include "build/build_config.h" #include "constants/stream_dict_common.h" #include "core/fpdfapi/parser/cpdf_dictionary.h" #include "core/fpdfapi/parser/cpdf_name.h" #include "core/fpdfapi/parser/cpdf_object.h" #include "core/fpdfapi/parser/cpdf_stream.h" #include "core/fpdfapi/parser/cpdf_string.h" #include "core/fpdfapi/parser/fpdf_parser_decode.h" #include "core/fxcrt/fx_system.h" #include "third_party/base/check.h" #include "third_party/base/notreached.h" namespace { #if BUILDFLAG(IS_APPLE) || BUILDFLAG(IS_WIN) WideString ChangeSlashToPlatform(const wchar_t* str) { WideString result; while (*str) { if (*str == '/') { #if BUILDFLAG(IS_APPLE) result += L':'; #else result += L'\\'; #endif } else { result += *str; } str++; } return result; } WideString ChangeSlashToPDF(const wchar_t* str) { WideString result; while (*str) { if (*str == '\\' || *str == ':') result += L'/'; else result += *str; str++; } return result; } #endif // BUILDFLAG(IS_APPLE) || BUILDFLAG(IS_WIN) } // namespace CPDF_FileSpec::CPDF_FileSpec(RetainPtr pObj) : m_pObj(std::move(pObj)) { DCHECK(m_pObj); } CPDF_FileSpec::~CPDF_FileSpec() = default; WideString CPDF_FileSpec::DecodeFileName(const WideString& filepath) { if (filepath.GetLength() <= 1) return WideString(); #if BUILDFLAG(IS_APPLE) if (filepath.First(sizeof("/Mac") - 1) == WideStringView(L"/Mac")) return ChangeSlashToPlatform(filepath.c_str() + 1); return ChangeSlashToPlatform(filepath.c_str()); #elif BUILDFLAG(IS_WIN) if (filepath[0] != L'/') return ChangeSlashToPlatform(filepath.c_str()); if (filepath[1] == L'/') return ChangeSlashToPlatform(filepath.c_str() + 1); if (filepath[2] == L'/') { WideString result; result += filepath[1]; result += L':'; result += ChangeSlashToPlatform(filepath.c_str() + 2); return result; } WideString result; result += L'\\'; result += ChangeSlashToPlatform(filepath.c_str()); return result; #else return WideString(filepath); #endif } WideString CPDF_FileSpec::GetFileName() const { WideString csFileName; if (const CPDF_Dictionary* pDict = m_pObj->AsDictionary()) { RetainPtr pUF = ToString(pDict->GetDirectObjectFor("UF")); if (pUF) csFileName = pUF->GetUnicodeText(); if (csFileName.IsEmpty()) { RetainPtr pK = ToString(pDict->GetDirectObjectFor(pdfium::stream::kF)); if (pK) csFileName = WideString::FromDefANSI(pK->GetString().AsStringView()); } if (pDict->GetByteStringFor("FS") == "URL") return csFileName; if (csFileName.IsEmpty()) { for (const auto* key : {"DOS", "Mac", "Unix"}) { RetainPtr pValue = ToString(pDict->GetDirectObjectFor(key)); if (pValue) { csFileName = WideString::FromDefANSI(pValue->GetString().AsStringView()); break; } } } } else if (const CPDF_String* pString = m_pObj->AsString()) { csFileName = WideString::FromDefANSI(pString->GetString().AsStringView()); } return DecodeFileName(csFileName); } RetainPtr CPDF_FileSpec::GetFileStream() const { const CPDF_Dictionary* pDict = m_pObj->AsDictionary(); if (!pDict) return nullptr; // Get the embedded files dictionary. RetainPtr pFiles = pDict->GetDictFor("EF"); if (!pFiles) return nullptr; // List of keys to check for the file specification string. // Follows the same precedence order as GetFileName(). static constexpr const char* kKeys[] = {"UF", "F", "DOS", "Mac", "Unix"}; size_t end = pDict->GetByteStringFor("FS") == "URL" ? 2 : std::size(kKeys); for (size_t i = 0; i < end; ++i) { ByteString key = kKeys[i]; if (!pDict->GetUnicodeTextFor(key).IsEmpty()) { RetainPtr pStream = pFiles->GetStreamFor(key); if (pStream) return pStream; } } return nullptr; } RetainPtr CPDF_FileSpec::GetParamsDict() const { RetainPtr pStream = GetFileStream(); if (!pStream) return nullptr; RetainPtr pDict = pStream->GetDict(); return pDict ? pDict->GetDictFor("Params") : nullptr; } RetainPtr CPDF_FileSpec::GetMutableParamsDict() { return pdfium::WrapRetain( const_cast(GetParamsDict().Get())); } WideString CPDF_FileSpec::EncodeFileName(const WideString& filepath) { if (filepath.GetLength() <= 1) return WideString(); #if BUILDFLAG(IS_WIN) if (filepath[1] == L':') { WideString result(L'/'); result += filepath[0]; if (filepath[2] != L'\\') result += L'/'; result += ChangeSlashToPDF(filepath.c_str() + 2); return result; } if (filepath[0] == L'\\' && filepath[1] == L'\\') return ChangeSlashToPDF(filepath.c_str() + 1); if (filepath[0] == L'\\') return L'/' + ChangeSlashToPDF(filepath.c_str()); return ChangeSlashToPDF(filepath.c_str()); #elif BUILDFLAG(IS_APPLE) if (filepath.First(sizeof("Mac") - 1).EqualsASCII("Mac")) return L'/' + ChangeSlashToPDF(filepath.c_str()); return ChangeSlashToPDF(filepath.c_str()); #else return WideString(filepath); #endif }