mirror of
https://github.com/GothenburgBitFactory/timewarrior.git
synced 2025-06-26 10:54:28 +02:00

- Remove unused includes - Use '<..>' style consistently - Strip any paths from include - Replace deprecated C++ headers - Sort includes according to LLVM rules https://llvm.org/docs/CodingStandards.html#include-style Signed-off-by: Thomas Lauf <thomas.lauf@tngtech.com>
158 lines
4.7 KiB
C++
158 lines
4.7 KiB
C++
////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Copyright 2018 - 2021, Thomas Lauf, Paul Beckingham, Federico Hernandez.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
// in the Software without restriction, including without limitation the rights
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
// furnished to do so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
// THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
// SOFTWARE.
|
|
//
|
|
// https://www.opensource.org/licenses/mit-license.php
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
#include <IntervalFactory.h>
|
|
#include <JSON.h>
|
|
#include <Lexer.h>
|
|
#include <format.h>
|
|
|
|
static std::vector <std::string> tokenizeSerialization (const std::string& line)
|
|
{
|
|
std::vector <std::string> tokens;
|
|
|
|
Lexer lexer (line);
|
|
std::string token;
|
|
Lexer::Type type;
|
|
|
|
// When parsing the serialization, we only need the lexer to look for strings
|
|
// and words since we're not using the provided type information
|
|
lexer.noDate ();
|
|
lexer.noDuration ();
|
|
lexer.noUUID ();
|
|
lexer.noHexNumber ();
|
|
lexer.noURL ();
|
|
lexer.noPath ();
|
|
lexer.noPattern ();
|
|
lexer.noOperator ();
|
|
|
|
while (lexer.token (token, type))
|
|
{
|
|
tokens.push_back (Lexer::dequote (token));
|
|
}
|
|
|
|
return tokens;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// Syntax:
|
|
// 'inc' [ <iso> [ '-' <iso> ]] [ '#' <tag> [ <tag> ... ]]
|
|
Interval IntervalFactory::fromSerialization (const std::string& line)
|
|
{
|
|
std::vector <std::string> tokens = tokenizeSerialization (line);
|
|
|
|
// Minimal requirement 'inc'.
|
|
if (!tokens.empty () && tokens[0] == "inc")
|
|
{
|
|
Interval interval = Interval ();
|
|
|
|
unsigned int offset = 0;
|
|
|
|
// Optional <iso>
|
|
if (tokens.size () > 1 &&
|
|
tokens[1].length () == 16)
|
|
{
|
|
interval.start = Datetime (tokens[1]);
|
|
offset = 1;
|
|
|
|
// Optional '-' <iso>
|
|
if (tokens.size () > 3 &&
|
|
tokens[2] == "-" &&
|
|
tokens[3].length () == 16)
|
|
{
|
|
interval.end = Datetime (tokens[3]);
|
|
offset = 3;
|
|
}
|
|
}
|
|
|
|
// Optional '#' <tag>
|
|
if (tokens.size () > 2 + offset && tokens[1 + offset] == "#")
|
|
{
|
|
// Optional <tag> ...
|
|
auto index = 2 + offset;
|
|
|
|
while (index < tokens.size () && tokens[index] != "#")
|
|
{
|
|
interval.tag (tokens[index]);
|
|
index++;
|
|
}
|
|
|
|
// Optional '#' <annotation>
|
|
if (index < tokens.size () && tokens[index] == "#")
|
|
{
|
|
std::string annotation;
|
|
|
|
// Optional <annotation> ...
|
|
for (unsigned int i = index + 1; i < tokens.size (); ++i)
|
|
{
|
|
annotation += (i > index +1 ? " " : "") + tokens[i];
|
|
}
|
|
|
|
interval.setAnnotation (annotation);
|
|
}
|
|
}
|
|
|
|
return interval;
|
|
}
|
|
|
|
throw format ("Unrecognizable line '{1}'.", line);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
Interval IntervalFactory::fromJson (const std::string& jsonString)
|
|
{
|
|
Interval interval = Interval ();
|
|
|
|
if (!jsonString.empty ())
|
|
{
|
|
std::unique_ptr <json::object> json (dynamic_cast <json::object *> (json::parse (jsonString)));
|
|
|
|
json::array* tags = (json::array*) json->_data["tags"];
|
|
|
|
if (tags != nullptr)
|
|
{
|
|
for (auto& tag : tags->_data)
|
|
{
|
|
auto* value = (json::string*) tag;
|
|
interval.tag (json::decode (value->_data));
|
|
}
|
|
}
|
|
|
|
json::string* annotation = (json::string*) json->_data["annotation"];
|
|
interval.annotation = (annotation != nullptr) ? json::decode (annotation->_data) : "";
|
|
|
|
json::string* start = (json::string*) json->_data["start"];
|
|
interval.start = (start != nullptr) ? Datetime(start->_data) : 0;
|
|
json::string* end = (json::string*) json->_data["end"];
|
|
interval.end = (end != nullptr) ? Datetime(end->_data) : 0;
|
|
|
|
json::number* id = (json::number*) json->_data["id"];
|
|
interval.id = (id != nullptr) ? id->_dvalue : 0;
|
|
}
|
|
|
|
return interval;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|