File: shortcut_creation_test_support_linux.cc

package info (click to toggle)
chromium 138.0.7204.157-1
  • links: PTS, VCS
  • area: main
  • in suites: trixie
  • size: 6,071,864 kB
  • sloc: cpp: 34,936,859; ansic: 7,176,967; javascript: 4,110,704; python: 1,419,953; asm: 946,768; xml: 739,967; pascal: 187,324; sh: 89,623; perl: 88,663; objc: 79,944; sql: 50,304; cs: 41,786; fortran: 24,137; makefile: 21,806; php: 13,980; tcl: 13,166; yacc: 8,925; ruby: 7,485; awk: 3,720; lisp: 3,096; lex: 1,327; ada: 727; jsp: 228; sed: 36
file content (178 lines) | stat: -rw-r--r-- 6,548 bytes parent folder | download | duplicates (4)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
// Copyright 2024 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "chrome/browser/shortcuts/shortcut_creation_test_support.h"

#include <stdlib.h>

#include "base/command_line.h"
#include "base/files/file_path.h"
#include "base/files/file_util.h"
#include "base/logging.h"
#include "base/strings/string_util.h"
#include "base/strings/stringprintf.h"
#include "base/strings/utf_string_conversions.h"
#include "base/threading/thread_restrictions.h"
#include "chrome/browser/chrome_browser_main.h"
#include "chrome/browser/shell_integration_linux.h"
#include "chrome/browser/shortcuts/fake_linux_xdg_wrapper.h"
#include "chrome/browser/shortcuts/linux_xdg_wrapper.h"
#include "chrome/browser/shortcuts/shortcut_creation_test_support_linux.h"
#include "chrome/browser/shortcuts/shortcut_creator_linux.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "url/gurl.h"

namespace shortcuts {

namespace {
std::string GetDesktopEntry(const std::string& key,
                            const std::string& desktop_file_contents) {
  return shell_integration_linux::internal::
      GetDesktopEntryStringValueFromFromDesktopFileForTest(
          key, desktop_file_contents);
}
}  // namespace

namespace internal {

std::vector<std::string> ParseDesktopExecForCommandLine(const std::string& s) {
  std::vector<std::string> result;
  std::string current_arg;
  bool in_quoted_arg = false;
  for (size_t i = 0; i < s.length(); ++i) {
    char c = s[i];
    if (in_quoted_arg) {
      // We're inside a "quoted argument" (i.e. have encountered a opening `"`
      // but not yet a closing `"`).
      if (c == '\\') {
        // `\` is an escape character for the next character, so skip the '\'
        // and append the next character to the current argument.
        CHECK_LT(i + 1, s.length());
        current_arg += s[++i];
      } else if (c == '"') {
        // `"` (which was not escaped) indicates the end of the quoted argument.
        // Push the argument to `result` and reset state ready for the next
        // argument.
        result.push_back(std::move(current_arg));
        in_quoted_arg = false;
        current_arg = "";
      } else {
        current_arg += c;
      }
    } else if (c == '"') {
      // `"` signals the start of a quoted argument. This should only occur when
      // we're not already in the middle of an argument.
      CHECK(current_arg.empty());
      in_quoted_arg = true;
    } else if (c == ' ' || c == '\t' || c == '\n') {
      // Whitespace deliminates arguments, so if the current argument is
      // non-empty, push it to `result`. The non-empty check ensures any
      // sequence of whitespace is treated as a single argument deliminator.
      if (!current_arg.empty()) {
        result.push_back(current_arg);
        current_arg = "";
      }
    } else {
      current_arg += c;
    }
  }
  CHECK(!in_quoted_arg);
  if (!current_arg.empty()) {
    result.push_back(current_arg);
  }
  return result;
}

}  // namespace internal

ShortcutCreationTestSupport::ShortcutCreationTestSupport() {
  std::unique_ptr<LinuxXdgWrapper> xdg_wrapper =
      std::make_unique<FakeLinuxXdgWrapper>();
  SetDefaultXdgWrapperForTesting(xdg_wrapper.get());
  cleanup_test_overrides_ = base::ScopedClosureRunner(base::BindOnce(
      [](std::unique_ptr<LinuxXdgWrapper>) {
        SetDefaultXdgWrapperForTesting(nullptr);
      },
      std::move(xdg_wrapper)));
}

ShortcutCreationTestSupport::~ShortcutCreationTestSupport() = default;

// static
void ShortcutCreationTestSupport::LaunchShortcut(const base::FilePath& path) {
  base::ScopedAllowBlockingForTesting allow_io;
  std::string file;
  CHECK(base::ReadFileToString(path, &file));
  base::CommandLine command_line(
      internal::ParseDesktopExecForCommandLine(GetDesktopEntry("Exec", file)));
  ChromeBrowserMainParts::ProcessSingletonNotificationCallback(
      command_line, /*current_directory=*/{});
}

// static
bool ShortcutCreationTestSupport::ShortcutIsForUrl(
    const base::FilePath& path,
    const GURL& url,
    testing::MatchResultListener* result_listener) {
  base::ScopedAllowBlockingForTesting allow_io;
  std::string file;
  if (!base::ReadFileToString(path, &file)) {
    *result_listener << "Could not load .desktop file from " << path;
    return false;
  }

  const std::string exec_value = GetDesktopEntry("Exec", file);
  const std::string url_value = GetDesktopEntry("URL", file);
  *result_listener << "Exec: " << exec_value << ", URL: " << url_value;
  return url_value == url.spec() &&
         exec_value.find(url.spec()) != std::string::npos;
}

// static
bool ShortcutCreationTestSupport::ShortcutIsForProfile(
    const base::FilePath& path,
    const base::FilePath& profile_path,
    testing::MatchResultListener* result_listener) {
  base::ScopedAllowBlockingForTesting allow_io;
  std::string file;
  if (!base::ReadFileToString(path, &file)) {
    *result_listener << "Could not load .desktop file from " << path;
    return false;
  }

  const std::string exec_value = GetDesktopEntry("Exec", file);
  *result_listener << "Exec: " << exec_value;
  return exec_value.find(base::StringPrintf(
             "--profile-directory=%s",
             profile_path.BaseName().value().c_str())) != std::string::npos;
}

// static
bool ShortcutCreationTestSupport::ShortcutHasTitle(
    const base::FilePath& path,
    const std::u16string& title,
    ::testing::MatchResultListener* result_listener) {
  base::ScopedAllowBlockingForTesting allow_io;
  std::string file;
  if (!base::ReadFileToString(path, &file)) {
    *result_listener << "Could not load .desktop file from " << path;
    return false;
  }

  const std::string name_value = GetDesktopEntry("Name", file);

  std::string safe_title = base::UTF16ToUTF8(title);
  base::ReplaceChars(safe_title, " \n\r", "_", &safe_title);
  // The actual linux shortcut code has more logic around illegal characters in
  // file names that might need to be stripped. For now tests using this matcher
  // don't use such file names, so the simpler approach here is good enough. If
  // we do need to use this matcher in tests for such edge cases, the logic here
  // might need to be updated.
  return ExplainMatchResult(::testing::HasSubstr(safe_title),
                            path.BaseName().value(), result_listener) &&
         ExplainMatchResult(::testing::Eq(base::UTF16ToUTF8(title)), name_value,
                            result_listener);
}

}  // namespace shortcuts