| // 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 <iterator> |
| #include <utility> |
| |
| #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<const CPDF_Object> 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<const CPDF_String> pUF = |
| ToString(pDict->GetDirectObjectFor("UF")); |
| if (pUF) |
| csFileName = pUF->GetUnicodeText(); |
| if (csFileName.IsEmpty()) { |
| RetainPtr<const CPDF_String> 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<const CPDF_String> 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<const CPDF_Stream> CPDF_FileSpec::GetFileStream() const { |
| const CPDF_Dictionary* pDict = m_pObj->AsDictionary(); |
| if (!pDict) |
| return nullptr; |
| |
| // Get the embedded files dictionary. |
| RetainPtr<const CPDF_Dictionary> 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<const CPDF_Stream> pStream = pFiles->GetStreamFor(key); |
| if (pStream) |
| return pStream; |
| } |
| } |
| return nullptr; |
| } |
| |
| RetainPtr<const CPDF_Dictionary> CPDF_FileSpec::GetParamsDict() const { |
| RetainPtr<const CPDF_Stream> pStream = GetFileStream(); |
| if (!pStream) |
| return nullptr; |
| |
| RetainPtr<const CPDF_Dictionary> pDict = pStream->GetDict(); |
| return pDict ? pDict->GetDictFor("Params") : nullptr; |
| } |
| |
| RetainPtr<CPDF_Dictionary> CPDF_FileSpec::GetMutableParamsDict() { |
| return pdfium::WrapRetain( |
| const_cast<CPDF_Dictionary*>(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 |
| } |