]> git.tdb.fi Git - builder.git/blob - source/config.cpp
Evaluate conditions at load time to allow more flexibility
[builder.git] / source / config.cpp
1 #include <msp/core/maputils.h>
2 #include <msp/fs/stat.h>
3 #include <msp/fs/utils.h>
4 #include <msp/io/file.h>
5 #include <msp/io/print.h>
6 #include <msp/time/utils.h>
7 #include "builder.h"
8 #include "config.h"
9 #include "sourcepackage.h"
10
11 using namespace std;
12 using namespace Msp;
13
14 Config::Config(SourcePackage &p):
15         package(p),
16         changed(false)
17 { }
18
19 void Config::add_option(const string &n, const string &v, const string &d)
20 {
21         Option opt(n, v, d);
22         if(pending_options.count(n))
23                 opt.value = pending_options[n];
24         options.insert(OptionMap::value_type(n, opt));
25 }
26
27 const Config::Option &Config::get_option(const string &name) const
28 {
29         return get_item(options, name);
30 }
31
32 bool Config::is_option(const string &name) const
33 {
34         return options.count(name);
35 }
36
37 bool Config::update(const StringMap &opts)
38 {
39         bool changed_now = false;
40         for(StringMap::const_iterator i=opts.begin(); i!=opts.end(); ++i)
41         {
42                 if(set_option(i->first, i->second) && i->first!="profile")
43                         changed_now = true;
44         }
45
46         if(changed_now)
47         {
48                 mtime = Time::now();
49                 changed = true;
50         }
51
52         return changed_now;
53 }
54
55 void Config::finish()
56 {
57         for(OptionMap::iterator i=options.begin(); i!=options.end(); ++i)
58                 i->second.value = package.expand_string(i->second.value);
59 }
60
61 void Config::save() const
62 {
63         if(!changed)
64                 return;
65
66         FS::Path fn = package.get_source()/".config";
67
68         package.get_builder().get_logger().log("files", format("Writing %s", fn));
69         IO::BufferedFile out(fn.str(), IO::M_WRITE);
70
71         for(OptionMap::const_iterator i=options.begin(); i!=options.end(); ++i)
72                 IO::print(out, "option \"%s\" \"%s\";\n", i->second.name, i->second.value);
73 }
74
75 bool Config::set_option(const string &opt, const string &val)
76 {
77         bool result = false;
78
79         OptionMap::iterator i = options.find(opt);
80         if(i!=options.end())
81         {
82                 if(i->second.value!=val)
83                 {
84                         result = true;
85                         changed = true;
86                 }
87                 i->second.value = val;
88         }
89
90         return result;
91 }
92
93 void Config::load()
94 {
95         FS::Path fn = package.get_source()/".config";
96
97         FS::Stat stat = FS::stat(fn);
98         if(stat)
99         {
100                 package.get_builder().get_logger().log("files", format("Reading %s", fn));
101                 IO::BufferedFile in(fn.str());
102
103                 mtime = stat.get_modify_time();
104
105                 DataFile::Parser parser(in, fn.str());
106                 Loader loader(*this);
107                 loader.load(parser);
108         }
109 }
110
111
112 Config::Option::Option(const string &n, const string &v, const string &d):
113         name(n),
114         defv(v),
115         descr(d),
116         value(v)
117 { }
118
119
120 Config::Loader::Loader(Config &c):
121         DataFile::ObjectLoader<Config>(c)
122 {
123         add("option", &Loader::option);
124 }
125
126 void Config::Loader::option(const string &n, const string &v)
127 {
128         if(obj.options.count(n))
129                 obj.set_option(n, v);
130         else
131                 obj.pending_options[n] = v;
132 }