1 // Copyright (c) 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
8 #include "base/files/file_util.h"
9 #include "base/strings/string_split.h"
10 #include "base/strings/string_util.h"
11 #include "tools/gn/err.h"
12 #include "tools/gn/filesystem_utils.h"
13 #include "tools/gn/functions.h"
14 #include "tools/gn/input_file.h"
15 #include "tools/gn/parse_tree.h"
16 #include "tools/gn/scheduler.h"
20 const char kWriteFile
[] = "write_file";
21 const char kWriteFile_HelpShort
[] =
22 "write_file: Write a file to disk.";
23 const char kWriteFile_Help
[] =
24 "write_file: Write a file to disk.\n"
26 " write_file(filename, data)\n"
28 " If data is a list, the list will be written one-item-per-line with no\n"
29 " quoting or brackets.\n"
31 " If the file exists and the contents are identical to that being\n"
32 " written, the file will not be updated. This will prevent unnecessary\n"
33 " rebuilds of targets that depend on this file.\n"
35 " TODO(brettw) we probably need an optional third argument to control\n"
41 " Filename to write. This must be within the output directory.\n"
44 " The list or string to write.\n";
46 Value
RunWriteFile(Scope
* scope
,
47 const FunctionCallNode
* function
,
48 const std::vector
<Value
>& args
,
50 if (args
.size() != 2) {
51 *err
= Err(function
->function(), "Wrong number of arguments to write_file",
52 "I expected two arguments.");
56 // Compute the file name and make sure it's in the output dir.
57 if (!args
[0].VerifyTypeIs(Value::STRING
, err
))
59 const SourceDir
& cur_dir
= scope
->GetSourceDir();
60 SourceFile source_file
= cur_dir
.ResolveRelativeFile(args
[0].string_value(),
61 scope
->settings()->build_settings()->root_path_utf8());
62 if (!EnsureStringIsInOutputDir(
63 scope
->settings()->build_settings()->build_dir(),
64 source_file
.value(), args
[0].origin(), err
))
68 std::ostringstream contents
;
69 if (args
[1].type() == Value::LIST
) {
70 const std::vector
<Value
>& list
= args
[1].list_value();
71 for (const auto& cur
: list
)
72 contents
<< cur
.ToString(false) << std::endl
;
74 contents
<< args
[1].ToString(false);
76 const std::string
& new_contents
= contents
.str();
77 base::FilePath file_path
=
78 scope
->settings()->build_settings()->GetFullPath(source_file
);
80 // Make sure we're not replacing the same contents.
81 std::string existing_contents
;
82 if (base::ReadFileToString(file_path
, &existing_contents
) &&
83 existing_contents
== new_contents
)
84 return Value(); // Nothing to do.
86 // Write file, creating the directory if necessary.
87 if (!base::CreateDirectory(file_path
.DirName())) {
88 *err
= Err(function
->function(), "Unable to create directory.",
89 "I was using \"" + FilePathToUTF8(file_path
.DirName()) + "\".");
93 int int_size
= static_cast<int>(new_contents
.size());
94 if (base::WriteFile(file_path
, new_contents
.c_str(), int_size
)
96 *err
= Err(function
->function(), "Unable to write file.",
97 "I was writing \"" + FilePathToUTF8(file_path
) + "\".");
103 } // namespace functions