FTQuant  0.1
SyntaxParser.cpp
Go to the documentation of this file.
1 
8 #include <SyntaxParser.hpp>
9 
10 bool is_double(const std::string& str) {
11  std::istringstream iss(str);
12  double d;
13  iss >> std::noskipws >> d;
14  return iss.eof() && !iss.fail();
15 }
16 
17 std::string Command::to_json() const {
18  std::string json = "{";
19  json += "code: " + std::to_string(_code) + ", " + "keyNumbers: ";
20  for (const auto& [key, value] : _key_numbers)
21  json += '[' + key + "] = " + value + "; ";
22  json += '}';
23 
24  return json;
25 }
26 
27 Command::Command(std::string com) {
28  std::stringstream ss(com);
29  std::vector<std::string> words;
30 
31  std::string temp;
32 
33  int flag = 0;
34  std::string word;
35  ss >> word;
36  if (Commands.count(word)) {
37  _code = Commands[word];
38  } else {
39  _code = Commands["INVALID_COMMAND"];
40  return;
41  }
42 
43  while (ss >> word) {
44  words.push_back(word);
45  }
46 
47  for (const auto& w : words) {
48  if (Fields.count(w)) {
49  if (!flag) {
50  _key_numbers[w] = "0";
51  flag = 1;
52  temp = w;
53  }
54  continue;
55  }
56 
57  if (flag == 1) {
58  if (is_double(w) || temp == "FILE")
59  _key_numbers[temp] = w;
60  else {
61  // std::cout << "cant convert " << w << " in double \n";
62  _code = INVALID_COMMAND;
63  return;
64  }
65  flag = 0;
66  }
67  }
68 
69  for (const auto& w : RequiredFields[_code]) {
70  if (!_key_numbers.count(w)) {
71  _code = INVALID_COMMAND;
72  }
73  }
74 }
75 
76 int Command::code() const {
77  return _code;
78 }
79 
80 std::vector<double> readStocksFromFile(std::string fname) {
81  std::string line;
82  std::ifstream in(fname);
83  std::vector<double> ans;
84 
85  if (in.is_open()) {
86  while (std::getline(in, line)) {
87  if (is_double(line))
88  ans.push_back(std::stod(line));
89  else {
90  ans.push_back(-1);
91  in.close();
92  return ans;
93  }
94  }
95  } else {
96  ans.push_back(-1);
97  in.close();
98  return ans;
99  }
100  in.close();
101  return ans;
102 }
103 
104 std::vector<std::vector<double>> read2DFromFile(std::string fname) {
105  std::string line;
106  std::ifstream in(fname);
107  std::vector<std::vector<double>> ans;
108  if (in.is_open()) {
109  while (std::getline(in, line)) {
110  std::stringstream ss(line);
111  std::vector<double> numbers;
112  std::string word;
113 
114  while (ss >> word) {
115  if (is_double(word)) {
116  numbers.push_back(std::stod(word));
117  } else {
118  numbers.push_back(-1);
119  }
120  }
121  ans.push_back(numbers);
122  }
123  } else {
124  std::vector<double> num{-1};
125  ans.push_back(num);
126  in.close();
127  return ans;
128  }
129  in.close();
130  return ans;
131 }
132 
134  double r = 0.;
135  double sigma = 1.;
136 
137  std::vector<std::vector<double>> w;
138  std::vector<double> T;
139  std::vector<double> y;
140 
141  std::vector<std::vector<double>> trajs;
142 
143  BlackScholes BSmodel(r, sigma);
144  switch (C._code) {
145  case INVALID_COMMAND: {
146  std::cout << "Invalid"
147  << " \n";
148  return -1;
149  break;
150  }
151 
152  case BLACK_SCHOLES: {
153  // std::cout << "BlackScholes model(" << C._key_numbers.at("INTEREST_RATE")
154  // << " \n";
155  BSmodel = BlackScholes(std::stod(C._key_numbers.at("INTEREST_RATE")),
156  std::stod(C._key_numbers.at("SIGMA")));
157  isBS = 1;
158  // std::cout << "BlackScholes model(" << _key_numbers.at("INTEREST_RATE"] << " \n";
159  // "_key_numbers.at(\"SIGMA\"]); \n";
160  break;
161  }
162 
163  case BLACK_SCHOLES_F: {
164  // std::cout << "BlackScholes model(" << C._key_numbers.at("FILE") << " \n";
165  auto stocks = readStocksFromFile(C._key_numbers.at("FILE"));
166  for (auto& s : stocks) {
167  if (s < 0)
168  return -1;
169  }
170  BSmodel.calibrate(stocks);
171 
172  isBS = 1;
173  // std::cout << "BlackScholes model(_key_numbers.at(\"INTEREST_RATE\"], "
174  // "_key_numbers.at(\"SIGMA\"]); \n";
175  break;
176  }
177 
178  case LOCVOL: {
179  // std::cout << "LOCVOL model(" << C._key_numbers.at("FILE_T") << " \n";
180  T = readStocksFromFile(C._key_numbers.at("FILE_T"));
181  for (auto& s : T) {
182  if (s < 0)
183  return -1;
184  }
185 
186  y = readStocksFromFile(C._key_numbers.at("FILE_y"));
187  for (auto& s : y) {
188  if (s < 0)
189  return -1;
190  }
191 
192  w = read2DFromFile(C._key_numbers.at("FILE_w"));
193  for (auto& s1 : w) {
194  for (auto& s2 : s1) {
195  if (s2 < 0)
196  return -1;
197  }
198  }
199 
200  LVmodel.calibrate_dupire(w, T, y,
201  std::stod(C._key_numbers.at("SPOT_PRICE")));
202  isBS = 0;
203  break;
204  }
205 
206  case GENERATE_TRAJECTORIES: {
207  if (isBS) {
208  // std::cout << "BS Tr num, steps: " << std::stod(C._key_numbers.at("TRAJECTORIES_NUMBER")) << std::stod(C._key_numbers.at("STEPS_NUMBER")) << std::endl;
209  traj = BSmodel.generate_paths(
210  std::stod(C._key_numbers.at("TRAJECTORIES_NUMBER")),
211  std::stod(C._key_numbers.at("SPOT_PRICE")),
212  std::stod(C._key_numbers.at("STEPS_NUMBER")),
213  std::stod(C._key_numbers.at("EXP_T")));
214  traj_generated = 1;
215  std::cout << "tr gen \n" << traj_generated << std::endl;
216  } else {
217  // std::cout << "LV Tr num, steps: " << std::stod(C._key_numbers.at("TRAJECTORIES_NUMBER")) << std::stod(C._key_numbers.at("STEPS_NUMBER")) << std::endl;
218  traj = LVmodel.generate_paths(
219  std::stod(C._key_numbers.at("TRAJECTORIES_NUMBER")),
220  std::stod(C._key_numbers.at("STEPS_NUMBER")),
221  std::stod(C._key_numbers.at("EXP_T")),
222  std::stod(C._key_numbers.at("SPOT_PRICE")));
223  traj_generated = 1;
224  }
225  break;
226  }
227  case EURO_CALL: {
228  auto bs_pricer = MonteCarloPricer<BlackScholes>(BSmodel);
229  // double res = 0;
230 
231  double strike = std::stod(C._key_numbers.at("STRIKE_PRICE"));
232 
233  // traj = BSmodel.generate_paths(
234  // std::stoi(C._key_numbers.at("TRAJECTORIES_NUMBER")),
235  // std::stoi(C._key_numbers.at("STEPS_NUMBER")),
236  // std::stod(C._key_numbers.at("EXP_T")),
237  // std::stod(C._key_numbers.at("SPOT_PRICE")));
238 
239  // for (auto t : traj) {
240  // res += ((t.back() - strike) > 0 ? (t.back() - strike) : 0);
241  // }
242 
243  // std::cout << "steps_number: "<<std::stoi(C._key_numbers.at("STEPS_NUMBER")) << std::endl;
244 
245  // std::cout << "tra size: " << traj[0].size() << std::endl;
246  // std::cout << std::endl;
247 
248  std::function<double(double)> payoff = [strike](double v) {
249  return ((v - strike) > 0 ? (v - strike) : 0);
250  };
251 
252  MonteCarloResult res = bs_pricer.estimate_price(
253  payoff, std::stod(C._key_numbers.at("ERROR")),
254  std::stod(C._key_numbers.at("STEPS_NUMBER")),
255  std::stod(C._key_numbers.at("EXP_T")),
256  std::stod(C._key_numbers.at("SPOT_PRICE")), true, 0.95, 1000);
257 
258  std::cout << "ans: " << res.to_json() << std::endl;
259 
260  break;
261  }
262  case EURO_PUT: {
263  auto bs_pricer = MonteCarloPricer<BlackScholes>(BSmodel);
264 
265  double strike = std::stod(C._key_numbers.at("STRIKE_PRICE"));
266  // double error = std::stod(C._key_numbers.at("ERROR"));
267 
268  std::function<double(double)> payoff = [strike](double v) {
269  return ((strike - v) > 0 ? (strike - v) : 0);
270  };
271 
272  MonteCarloResult res = bs_pricer.estimate_price(
273  payoff, std::stod(C._key_numbers.at("ERROR")),
274  std::stod(C._key_numbers.at("STEPS_NUMBER")),
275  std::stod(C._key_numbers.at("EXP_T")),
276  std::stod(C._key_numbers.at("SPOT_PRICE")),
277  std::stod(C._key_numbers.at("TRAJECTORIES_NUMBER")));
278 
279  std::cout << res.to_json() << std::endl;
280 
281  break;
282  }
283  // case PUT_KNOCK_OUT: {
284  // auto bs_pricer = MonteCarloPricer<BlackScholes>(BSmodel);
285 
286  // double strike = std::stod(C._key_numbers.at("STRIKE_PRICE"));
287  // double l_barrier = std::stod(C._key_numbers.at("LOWER_BARRIER"));
288 
289  // const std::function<double(std::vector<double>)> payoff =
290  // [strike, l_barrier](std::vector<double> v) {
291  // double m = std::min_element(std::begin(v), std::end(v));
292  // if (m <= l_barrier)
293  // return 0.;
294  // return ((strike - v.back()) > 0 ? (strike - v.back()) : 0);
295  // };
296 
297  // MonteCarloResult res = bs_pricer.estimate_price(
298  // payoff, std::stod(C._key_numbers.at("ERROR")));
299 
300  // std::cout << res.to_json() << std::endl;
301 
302  // break;
303  // }
304  // case CALL_KNOCK_OUT: {
305  // auto bs_pricer = MonteCarloPricer<BlackScholes>(BSmodel);
306 
307  // double strike = std::stod(C._key_numbers.at("STRIKE_PRICE"));
308  // double l_barrier = std::stod(C._key_numbers.at("LOWER_BARRIER"));
309 
310  // const std::function<double(std::vector<double>)> payoff =
311  // [strike, l_barrier](std::vector<double> v) {
312  // double m = std::min_element(std::begin(v), std::end(v));
313  // if (m <= l_barrier)
314  // return 0.;
315  // return ((v.back() - strike) > 0 ? (v.back() - strike) : 0);
316  // };
317 
318  // MonteCarloResult res = bs_pricer.estimate_price(
319  // payoff, std::stod(C._key_numbers.at("ERROR")));
320 
321  // std::cout << res.to_json() << std::endl;
322 
323  // break;
324  // }
325  // case PUT_KNOCK_IN: {
326  // auto bs_pricer = MonteCarloPricer<BlackScholes>(BSmodel);
327 
328  // double strike = std::stod(C._key_numbers.at("STRIKE_PRICE"));
329  // double u_barrier = std::stod(C._key_numbers.at("UPPER_BARRIER"));
330 
331  // const std::function<double(std::vector<double>)> payoff =
332  // [strike, u_barrier](std::vector<double> v) {
333  // double m = std::min_element(std::begin(v), std::end(v));
334  // if (m >= u_barrier)
335  // ((strike - v.back()) > 0 ? (strike - v.back()) : 0);
336  // return 0.;
337  // };
338 
339  // MonteCarloResult res = bs_pricer.estimate_price(
340  // payoff, std::stod(C._key_numbers.at("ERROR")));
341 
342  // std::cout << res.to_json() << std::endl;
343 
344  // break;
345  // }
346  // case CALL_KNOCK_IN: {
347  // auto bs_pricer = MonteCarloPricer<BlackScholes>(BSmodel);
348 
349  // double strike = std::stod(C._key_numbers.at("STRIKE_PRICE"));
350  // double u_barrier = std::stod(C._key_numbers.at("UPPER_BARRIER"));
351 
352  // const std::function<double(std::vector<double>)> payoff =
353  // [strike, u_barrier](std::vector<double> v) {
354  // double m = std::min_element(std::begin(v), std::end(v));
355  // if (m >= u_barrier)
356  // return ((v.back() - strike) > 0 ? (v.back() - strike) : 0);
357  // return 0.;
358  // };
359 
360  // MonteCarloResult res = bs_pricer.estimate_price(
361  // payoff, std::stod(C._key_numbers.at("ERROR")));
362 
363  // std::cout << res.to_json() << std::endl;
364 
365  // break;
366  // }
367  default: {
368  return -1;
369  break;
370  }
371  }
372  return 0;
373 }
374 
375 std::ostream& operator<<(std::ostream& os, const Command& C) {
376  std::cout << "code: " << C._code << std::endl;
377  std::cout << "keyNumbers: ";
378  for (const auto& [key, value] : C._key_numbers)
379  std::cout << '[' << key << "] = " << value << "; ";
380 
381  std::cout << std::endl;
382 
383  return os;
384 }
bool is_double(const std::string &str)
@function is_double
std::vector< std::vector< double > > read2DFromFile(std::string fname)
std::vector< double > readStocksFromFile(std::string fname)
std::ostream & operator<<(std::ostream &os, const Command &C)
Implements the Black-Scholes model.
Definition: BlackSholes.hpp:32
std::vector< std::vector< double > > generate_paths(int n_paths, int steps, double T, double spot, bool antithetic=true)
Definition: BlackSholes.cpp:7
void calibrate(std::vector< double > &stock_prices)
Definition: BlackSholes.cpp:59
Container for the commands Class contains the parsed command.
int code() const
std::string to_json() const
int execute(Command C)
int calibrate_dupire(std::vector< std::vector< double >> &w, std::vector< double > &T, std::vector< double > &y, double spot)
std::vector< std::vector< double > > generate_paths(int n_paths, double spot, int steps, double T)
A controller for derivatives pricing via a Monte-Carlo simulation.
Definition: ftqlib.hpp:67
A container for storing the result of a Monte-Carlo simulation.
Definition: ftqlib.hpp:43
std::string to_json() const
A function that converts a MonteCarloResult to a JSON string.
Definition: MonteCarlo.cpp:14