1
0
mirror of https://github.com/VCMP-SqMod/SqMod.git synced 2025-01-18 19:47:15 +01:00

WIP extended string formatting.

This commit is contained in:
Sandu Liviu Catalin 2021-04-01 19:31:33 +03:00
parent 4a238bc611
commit 0ec506f8e8
4 changed files with 980 additions and 0 deletions

View File

@ -66,6 +66,7 @@ add_library(SqModule MODULE SqBase.hpp Main.cpp
Library/Chrono/Timer.cpp Library/Chrono/Timer.hpp
Library/Chrono/Timestamp.cpp Library/Chrono/Timestamp.hpp
Library/CURL.cpp Library/CURL.hpp
Library/Format.cpp Library/Format.hpp
Library/IO.cpp Library/IO.hpp
Library/IO/Buffer.cpp Library/IO/Buffer.hpp
Library/IO/File.cpp Library/IO/File.hpp

903
module/Library/Format.cpp Normal file
View File

@ -0,0 +1,903 @@
// ------------------------------------------------------------------------------------------------
#include "Library/Format.hpp"
// ------------------------------------------------------------------------------------------------
namespace SqMod {
/* ------------------------------------------------------------------------------------------------
* Common implementation of formattable script types.
*/
struct SqObjectFmt
{
/* --------------------------------------------------------------------------------------------
* Object VM.
*/
HSQUIRRELVM mVM{};
/* --------------------------------------------------------------------------------------------
* Script table object.
*/
HSQOBJECT mObj{};
/* --------------------------------------------------------------------------------------------
* Base constructor.
*/
SqObjectFmt(HSQUIRRELVM vm, SQInteger idx, SQInteger & res)
: mVM(vm), mObj()
{
res = sq_getstackobj(vm, idx, &mObj);
// Not checking if this is the correct type because it shouldn't get here if it wasn't
}
/* --------------------------------------------------------------------------------------------
* Copy constructor.
*/
SqObjectFmt(const SqObjectFmt & o) = default;
/* --------------------------------------------------------------------------------------------
* Move constructor.
*/
SqObjectFmt(SqObjectFmt && o) noexcept = default;
/* --------------------------------------------------------------------------------------------
* Destructor.
*/
~SqObjectFmt() = default;
/* --------------------------------------------------------------------------------------------
* Copy assignment operator.
*/
SqObjectFmt & operator = (const SqObjectFmt & o) = default;
/* --------------------------------------------------------------------------------------------
* Move assignment operator.
*/
SqObjectFmt & operator = (SqObjectFmt && o) noexcept = default;
};
/* ------------------------------------------------------------------------------------------------
* Specialization of SqObjectFmt for array type.
*/
struct SqArrayFmt : public SqObjectFmt
{
using SqObjectFmt::SqObjectFmt;
using SqObjectFmt::operator =;
};
/* ------------------------------------------------------------------------------------------------
* Specialization of SqObjectFmt for table type.
*/
struct SqTableFmt : public SqObjectFmt
{
using SqObjectFmt::SqObjectFmt;
using SqObjectFmt::operator =;
};
/* ------------------------------------------------------------------------------------------------
* Specialization of SqObjectFmt for class type.
*/
struct SqClassFmt : public SqObjectFmt
{
using SqObjectFmt::SqObjectFmt;
using SqObjectFmt::operator =;
};
/* ------------------------------------------------------------------------------------------------
* Specialization of SqObjectFmt for instance type.
*/
struct SqInstanceFmt : public SqObjectFmt
{
using SqObjectFmt::SqObjectFmt;
using SqObjectFmt::operator =;
};
/* ------------------------------------------------------------------------------------------------
* Specialization of SqObjectFmt for closure type.
*/
struct SqClosureFmt : public SqObjectFmt
{
using SqObjectFmt::SqObjectFmt;
using SqObjectFmt::operator =;
};
/* ------------------------------------------------------------------------------------------------
* Specialization of SqObjectFmt for generator type.
*/
struct SqGeneratorFmt : public SqObjectFmt
{
using SqObjectFmt::SqObjectFmt;
using SqObjectFmt::operator =;
};
/* ------------------------------------------------------------------------------------------------
* Specialization of SqObjectFmt for weak reference type.
*/
struct SqWeakReferenceFmt : public SqObjectFmt
{
using SqObjectFmt::SqObjectFmt;
using SqObjectFmt::operator =;
};
// ------------------------------------------------------------------------------------------------
SQInteger FormatContext::Proc(HSQUIRRELVM vm, SQInteger text, SQInteger args, SQInteger end)
{
// Is there any argument limit?
if (end < 0) end = sq_gettop(vm);
// Common value storage
SQInteger i = 0;
SQBool b = 0;
SQFloat f = 0;
const SQChar * s = nullptr;
// Attempt to retrieve the string value from the stack
mRes = sq_getstringandsize(vm, text, &s, &i);
// By any chance we failed to retrieve string?
if (SQ_FAILED(mRes))
{
return mRes;
}
// Remember the format string and assume it lives on the stack
fmt::basic_string_view< SQChar > f_str(s, static_cast< size_t >(i));
//
for(SQInteger idx = args; SQ_SUCCEEDED(mRes) && idx <= end; ++idx)
{
switch(sq_gettype(vm, idx))
{
// Null?
case OT_NULL: {
mArgs.push_back(nullptr);
} break;
// Integer?
case OT_INTEGER: {
mRes = sq_getinteger(vm, idx, &i);
mArgs.push_back(i);
} break;
// Float?
case OT_FLOAT: {
mRes = sq_getfloat(vm, idx, &f);
mArgs.push_back(f);
} break;
// Boolean?
case OT_BOOL: {
mRes = sq_getbool(vm, idx, &b);
mArgs.push_back(static_cast< bool >(b));
} break;
// String?
case OT_STRING: {
mRes = sq_getstringandsize(vm, idx, &s, &i);
if (SQ_SUCCEEDED(mRes)) mArgs.push_back(fmt::basic_string_view< SQChar >(s, static_cast< size_t >(i)));
} break;
// Table?
case OT_TABLE: {
SqTableFmt o(vm, idx, mRes);
if (SQ_SUCCEEDED(mRes)) mArgs.push_back(o);
} break;
// Array?
case OT_ARRAY: {
SqArrayFmt o(vm, idx, mRes);
if (SQ_SUCCEEDED(mRes)) mArgs.push_back(o);
} break;
// User data?
case OT_USERDATA: {
SQUserPointer p = nullptr;
mRes = sq_getuserdata(vm, idx, &p, nullptr);
if (SQ_SUCCEEDED(mRes)) mArgs.push_back(static_cast< void * >(p));
} break;
// Closure?
case OT_CLOSURE:
case OT_NATIVECLOSURE: {
SqClosureFmt o(vm, idx, mRes);
if (SQ_SUCCEEDED(mRes)) mArgs.push_back(o);
} break;
// Generator?
case OT_GENERATOR: {
SqGeneratorFmt o(vm, idx, mRes);
if (SQ_SUCCEEDED(mRes)) mArgs.push_back(o);
} break;
// User pointer?
case OT_USERPOINTER: {
SQUserPointer p = nullptr;
mRes = sq_getuserpointer(vm, idx, &p);
if (SQ_SUCCEEDED(mRes)) mArgs.push_back(static_cast< void * >(p));
} break;
// Class?
case OT_CLASS: {
SqClassFmt o(vm, idx, mRes);
if (SQ_SUCCEEDED(mRes)) mArgs.push_back(o);
} break;
// Instance?
case OT_INSTANCE: {
SqInstanceFmt o(vm, idx, mRes);
if (SQ_SUCCEEDED(mRes)) mArgs.push_back(o);
} break;
// Weak reference?
case OT_WEAKREF: {
SqWeakReferenceFmt o(vm, idx, mRes);
if (SQ_SUCCEEDED(mRes)) mArgs.push_back(o);
} break;
// Unknown!
default: mRes = sq_throwerror(vm, "Unknown or unsupported object type");
}
}
// Attempt to generate the format string
if (SQ_SUCCEEDED(mRes))
{
try
{
mOut = fmt::vformat(f_str, mArgs);
}
catch (const std::exception & e)
{
mRes = sq_throwerror(vm, e.what());
}
}
// Return result
return mRes;
}
// ------------------------------------------------------------------------------------------------
template < class FormatContext >
inline auto FormatObjectInContext(HSQUIRRELVM vm, SQInteger idx, SQInteger src, FormatContext & ctx) -> decltype(ctx.out())
{
// Identify what type of value we should format
switch(sq_gettype(vm, idx))
{
// Null?
case OT_NULL: {
return fmt::format_to(ctx.out(), "0x0");
}
// Integer?
case OT_INTEGER: {
SQInteger i = 0;
sq_getinteger(vm, idx, &i);
return fmt::format_to(ctx.out(), "{}", i);
}
// Float?
case OT_FLOAT: {
SQFloat f = 0;
sq_getfloat(vm, idx, &f);
return fmt::format_to(ctx.out(), "{}", f);
}
// Boolean?
case OT_BOOL: {
SQBool b = 0;
sq_getbool(vm, idx, &b);
return fmt::format_to(ctx.out(), "{}", b ? "true" : "false");
}
// String?
case OT_STRING: {
SQInteger i = 0;
const SQChar * s = nullptr;
sq_getstringandsize(vm, idx, &s, &i);
return fmt::format_to(ctx.out(), "{}", fmt::basic_string_view< SQChar >(s, static_cast< size_t >(i)));
}
// Table? TODO: generate json-style string!
case OT_TABLE: throw fmt::format_error("no viable conversion from table to string");
// Array? TODO: generate json-style string!
case OT_ARRAY: throw fmt::format_error("no viable conversion from array to string");
case OT_USERDATA: {
SQUserPointer p = nullptr;
sq_getuserdata(vm, idx, &p, nullptr);
return fmt::format_to(ctx.out(), "{:p}", p);
}
// Closure?
case OT_CLOSURE:
case OT_NATIVECLOSURE: {
SQInteger n = 0, x;
// Retrieve the number of parameters that this closure expects
if (SQ_SUCCEEDED(sq_getclosureinfo(vm, idx, &n, &x)))
{
// Does this closure need any parameters?
if (n <= 1)
{
const SQObjectType st = sq_gettype(vm, src);
// is the source is a class or instance?
if (st == OT_CLASS || st == OT_INSTANCE)
{
sq_push(vm, src); // Use that as environment
}
else
{
sq_push(vm, 1); // Use the current environment
}
// Attempt to invoke the function
if (SQ_SUCCEEDED(sq_call(vm, 1, SQTrue, SQFalse)))
{
// At this point we have a new object on the stack that must be removed
SqPopTopGuard tpg(vm);
// Invoke this function again on the returned value
return FormatObjectInContext(vm, -1, src, ctx);
} else throw fmt::format_error(LastErrorString(vm));
}
else
{
std::string s("closure requires ");
fmt::format_int f(n);
s.append(f.data(), f.size());
s.append(" parameters and cannot be invoked");
throw fmt::format_error(s);
}
} else throw fmt::format_error("failed to retrieve closure information");
}
// Generator?
case OT_GENERATOR: {
// Guard the stack size
StackGuard sg(vm);
// Is the generator at the top of the stack?
if (idx != -1 && idx != sq_gettop(vm))
{
sq_push(vm, idx); // Then push it
}
// Keep resuming the generator until null is returned
while (true)
{
if (SQ_SUCCEEDED(sq_resume(vm, SQTrue, SQFalse)))
{
// At this point we have a new object on the stack that must be removed
::SqMod::SqPopTopGuard rpg(vm);
// Is the returned value not null?
if (sq_gettype(vm, -1) != OT_NULL)
{
// Invoke this function again on the returned value
return FormatObjectInContext(vm, -1, src, ctx);
} else break; // Stop
} else throw fmt::format_error(LastErrorString(vm));
}
}
// user pointer?
case OT_USERPOINTER: {
SQUserPointer p = nullptr;
sq_getuserpointer(vm, idx, &p);
return fmt::format_to(ctx.out(), "{:p}", p);
}
// Class?
case OT_CLASS: {
// Attempt to retrieve the type name of the specified value
if (SQ_SUCCEEDED(sq_typeof(vm, idx)))
{
// At this point we have a new object on the stack that must be removed
SqPopTopGuard tpg(vm);
// The value on the stack may not be a string
if (SQ_SUCCEEDED(sq_tostring(vm, -1)))
{
SQInteger i = 0;
const SQChar * s = nullptr;
// At this point we have a new object on the stack that must be removed
SqPopTopGuard spg(vm);
// Attempt to obtain the string pointer
if (SQ_SUCCEEDED(sq_getstringandsize(vm, -1, &s, &i)))
{
return fmt::format_to(ctx.out(), "{}", fmt::basic_string_view< SQChar >(s, static_cast< size_t >(i)));
} else throw fmt::format_error("failed to retrieve type name string from stack");
} else throw fmt::format_error("failed to convert class type name to string");
} else throw fmt::format_error("no viable conversion from class to string");
}
// Instance?
case OT_INSTANCE: {
// Attempt to convert the value from the stack to a string
if (SQ_SUCCEEDED(sq_tostring(vm, idx)))
{
SQInteger i = 0;
const SQChar * s = nullptr;
// At this point we have a new object on the stack that must be removed
SqPopTopGuard spg(vm);
// Attempt to obtain the string pointer
if (SQ_SUCCEEDED(sq_getstringandsize(vm, -1, &s, &i)))
{
return fmt::format_to(ctx.out(), "{}", fmt::basic_string_view< SQChar >(s, static_cast< size_t >(i)));
} else throw fmt::format_error("failed to retrieve instance string representation from stack");
} else throw fmt::format_error("no viable conversion from instance to string");
}
// Weak reference?
case OT_WEAKREF: {
// Push the referenced object on the stack
if (SQ_SUCCEEDED(sq_getweakrefval(vm, idx)))
{
// At this point we have a new object on the stack that must be removed
SqPopTopGuard vpg(vm);
// Invoke this function again on the referenced object
return FormatObjectInContext(vm, -1, src, ctx);
} else throw fmt::format_error("failed to retrieve weak reference value");
}
// Unknown
default: break;
}
// Unknown type
throw fmt::format_error("unknown or unsupported value type");
}
// ------------------------------------------------------------------------------------------------
SQInteger SqFormat(HSQUIRRELVM vm)
{
FormatContext ctx;
// Attempt to generate the formatted string
if (SQ_FAILED(ctx.Proc(vm, 2, 3, sq_gettop(vm))))
{
return ctx.mRes;
}
// Push it on the stack
sq_pushstring(vm, ctx.mOut.data(), static_cast< SQInteger >(ctx.mOut.size()));
// Specify that we returned a value
return 1;
}
// ================================================================================================
void Register_Format(HSQUIRRELVM vm)
{
RootTable(vm).SquirrelFunc(_SC("fmt"), SqFormat);
}
} // Namespace:: SqMod
// ------------------------------------------------------------------------------------------------
namespace fmt {
/* ------------------------------------------------------------------------------------------------
* Implements formatting for script arrays.
*/
template < > struct formatter< ::SqMod::SqArrayFmt >
{
size_t mIndex{};
/* --------------------------------------------------------------------------------------------
* Parse the specified string range.
*/
auto parse(format_parse_context & ctx) -> decltype(ctx.begin())
{
// Retrieve the range that to be parsed
auto it = ctx.begin(), begin = it, end = ctx.end();
// Find the closing brace
while (it != end && *it != '}') ++it;
// Check if closing brace was found
if (it != end && *it != '}')
{
throw format_error("missing curly brace }");
}
// Check if an index was specified
else if (it == begin)
{
mIndex = 0; // Default to first element
}
else
{
char * p_end = nullptr;
// Transform the value into an index
#ifdef _SQ64
mIndex = std::strtoull(&(*begin), &p_end, 10);
#else
mIndex = std::strtoul(&(*begin), &p_end, 10);
#endif
// Validate index range
if (errno == ERANGE)
{
throw format_error("index out of range");
}
// Validate index value
else if (*p_end != '}')
{
throw format_error("invalid integer index");
}
}
// Return an iterator past the end of the parsed range:
return it;
}
/* --------------------------------------------------------------------------------------------
* Perform the requested format.
*/
template < class FormatContext >
auto format(const ::SqMod::SqArrayFmt & o, FormatContext & ctx) -> decltype(ctx.out())
{
// Remember the stack size
SQInteger top = sq_gettop(o.mVM);
// Push the array object on the stack
sq_pushobject(o.mVM, o.mObj);
// At this point we have a new object on the stack that must be removed
::SqMod::SqPopTopGuard apg(o.mVM);
// Push the index on the stack
sq_pushinteger(o.mVM, static_cast< SQInteger >(mIndex));
// Attempt to retrieve the index at the specified index
if (SQ_FAILED(sq_get(o.mVM, -2)))
{
std::string s("no such index in array: ");
fmt::format_int f(mIndex);
s.append(f.data(), f.size());
throw format_error(s);
}
// At this point we have a new object on the stack that must be removed
::SqMod::SqPopTopGuard epg(o.mVM);
// Run the format on the object that was pushed on the stack
return ::SqMod::FormatObjectInContext(o.mVM, -1, top + 1, ctx);
}
};
/* ------------------------------------------------------------------------------------------------
* Implements formatting for script tables.
*/
template < > struct formatter< ::SqMod::SqTableFmt >
{
std::string mIndex{};
/* --------------------------------------------------------------------------------------------
* Parse the specified string range.
*/
auto parse(format_parse_context & ctx) -> decltype(ctx.begin())
{
// Retrieve the range that to be parsed
auto it = ctx.begin(), begin = it, end = ctx.end();
// Find the closing brace
while (it != end && *it != '}') ++it;
// Check if closing brace was found
if (it != end && *it != '}')
{
throw format_error("missing curly brace }");
}
// Check if a name was specified
else if (it == begin)
{
throw format_error("must specify a table index");
}
// Store the index name
mIndex.assign(begin, it);
// Return an iterator past the end of the parsed range:
return it;
}
/* --------------------------------------------------------------------------------------------
* Perform the requested format.
*/
template < class FormatContext >
auto format(const ::SqMod::SqTableFmt & o, FormatContext & ctx) -> decltype(ctx.out())
{
// Remember the stack size
SQInteger top = sq_gettop(o.mVM);
// Push the array object on the stack
sq_pushobject(o.mVM, o.mObj);
// At this point we have a new object on the stack that must be removed
::SqMod::SqPopTopGuard apg(o.mVM);
// Push the index on the stack (only works with string keys/elements)
sq_pushstring(o.mVM, mIndex.data(), static_cast< SQInteger >(mIndex.size()));
// Attempt to retrieve the index at the specified index
if (SQ_FAILED(sq_get(o.mVM, -2)))
{
std::string s("no such index in table: ");
s.append(mIndex);
throw format_error(s);
}
// At this point we have a new object on the stack that must be removed
::SqMod::SqPopTopGuard epg(o.mVM);
// Run the format on the object that was pushed on the stack
return ::SqMod::FormatObjectInContext(o.mVM, -1, top + 1, ctx);
}
};
/* ------------------------------------------------------------------------------------------------
* Implements formatting for script classes.
*/
template < > struct formatter< ::SqMod::SqClassFmt >
{
std::string mIndex{};
/* --------------------------------------------------------------------------------------------
* Parse the specified string range.
*/
auto parse(format_parse_context & ctx) -> decltype(ctx.begin())
{
// Retrieve the range that to be parsed
auto it = ctx.begin(), begin = it, end = ctx.end();
// Find the closing brace
while (it != end && *it != '}') ++it;
// Check if closing brace was found
if (it != end && *it != '}')
{
throw format_error("missing curly brace }");
}
// Check if a identifier was specified
else if (it == begin)
{
throw format_error("must specify a member identifier");
}
// Validate first character
else if (std::isalpha(*begin) == 0 && *begin != '_')
{
throw format_error("member names must start with an alphabetic or underscore character");
}
// Validate member identifier
else if (!std::all_of(begin, it, [](int c) { return std::isalnum(c) != 0 || c == '_'; }))
{
throw format_error("member names can only contain alpha-numeric or underscore characters");
}
// Store the member identifier
mIndex.assign(begin, it);
// Return an iterator past the end of the parsed range:
return it;
}
/* --------------------------------------------------------------------------------------------
* Perform the requested format.
*/
template < class FormatContext >
auto format(const ::SqMod::SqClassFmt & o, FormatContext & ctx) -> decltype(ctx.out())
{
// Remember the stack size
SQInteger top = sq_gettop(o.mVM);
// Push the class object on the stack
sq_pushobject(o.mVM, o.mObj);
// At this point we have a new object on the stack that must be removed
::SqMod::SqPopTopGuard apg(o.mVM);
// Push the index on the stack (only works with string keys/elements)
sq_pushstring(o.mVM, mIndex.data(), static_cast< SQInteger >(mIndex.size()));
// Attempt to retrieve the index at the specified index
if (SQ_FAILED(sq_get(o.mVM, -2)))
{
std::string s("no such identifier in class: ");
s.append(mIndex);
throw format_error(s);
}
// At this point we have a new object on the stack that must be removed
::SqMod::SqPopTopGuard epg(o.mVM);
// Run the format on the object that was pushed on the stack
return ::SqMod::FormatObjectInContext(o.mVM, -1, top + 1, ctx);
}
};
/* ------------------------------------------------------------------------------------------------
* Implements formatting for script instances.
*/
template < > struct formatter< ::SqMod::SqInstanceFmt >
{
std::string mIndex{};
/* --------------------------------------------------------------------------------------------
* Parse the specified string range.
*/
auto parse(format_parse_context & ctx) -> decltype(ctx.begin())
{
// Retrieve the range that to be parsed
auto it = ctx.begin(), begin = it, end = ctx.end();
// Find the closing brace
while (it != end && *it != '}') ++it;
// Check if closing brace was found
if (it != end && *it != '}')
{
throw format_error("missing curly brace }");
}
// Check if a identifier was specified
else if (it == begin)
{
throw format_error("must specify a member identifier");
}
// Validate first character
else if (std::isalpha(*begin) == 0 && *begin != '_')
{
throw format_error("member names must start with an alphabetic or underscore character");
}
// Validate member identifier
else if (!std::all_of(begin, it, [](int c) { return std::isalnum(c) != 0 || c == '_'; }))
{
throw format_error("member names can only contain alpha-numeric or underscore characters");
}
// Store the member identifier
mIndex.assign(begin, it);
// Return an iterator past the end of the parsed range:
return it;
}
/* --------------------------------------------------------------------------------------------
* Perform the requested format.
*/
template < class FormatContext >
auto format(const ::SqMod::SqInstanceFmt & o, FormatContext & ctx) -> decltype(ctx.out())
{
// Remember the stack size
SQInteger top = sq_gettop(o.mVM);
// Push the class object on the stack
sq_pushobject(o.mVM, o.mObj);
// At this point we have a new object on the stack that must be removed
::SqMod::SqPopTopGuard apg(o.mVM);
// Push the index on the stack (only works with string keys/elements)
sq_pushstring(o.mVM, mIndex.data(), static_cast< SQInteger >(mIndex.size()));
// Attempt to retrieve the index at the specified index
if (SQ_FAILED(sq_get(o.mVM, -2)))
{
std::string s("no such identifier in instance: ");
s.append(mIndex);
throw format_error(s);
}
// At this point we have a new object on the stack that must be removed
::SqMod::SqPopTopGuard epg(o.mVM);
// Run the format on the object that was pushed on the stack
return ::SqMod::FormatObjectInContext(o.mVM, -1, top + 1, ctx);
}
};
/* ------------------------------------------------------------------------------------------------
* Implements formatting for script closures.
*/
template < > struct formatter< ::SqMod::SqClosureFmt >
{
std::string mParameter{};
/* --------------------------------------------------------------------------------------------
* Parse the specified string range.
*/
auto parse(format_parse_context & ctx) -> decltype(ctx.begin())
{
// Retrieve the range that to be parsed
auto it = ctx.begin(), begin = it, end = ctx.end();
// Find the closing brace
while (it != end && *it != '}') ++it;
// Check if closing brace was found
if (it != end && *it != '}')
{
throw format_error("missing curly brace }");
}
// Check if a parameter was specified
else if (it != begin)
{
mParameter.assign(begin, it); // Store the parameter
}
// Return an iterator past the end of the parsed range:
return it;
}
/* --------------------------------------------------------------------------------------------
* Perform the requested format.
*/
template < class FormatContext >
auto format(const ::SqMod::SqClosureFmt & o, FormatContext & ctx) -> decltype(ctx.out())
{
SQInteger n = 0, x;
// Push the closure object on the stack
sq_pushobject(o.mVM, o.mObj);
// At this point we have a new object on the stack that must be removed
::SqMod::SqPopTopGuard opg(o.mVM);
// Retrieve the number of parameters that this closure expects
if (SQ_SUCCEEDED(sq_getclosureinfo(o.mVM, -1, &n, &x)))
{
// Does this closure need any parameters?
if (n <= 2)
{
// Use the current environment
sq_push(o.mVM, 1);
// If it requires a parameter then we push the string anyway
if (n == 2 || !mParameter.empty())
{
sq_pushstring(o.mVM, mParameter.data(), static_cast< SQInteger >(mParameter.size()));
x = 2; // Include string in parameter count
} else x = 1; // Exclude string from parameter count
// Attempt to invoke the function
if (SQ_SUCCEEDED(sq_call(o.mVM, x, SQTrue, SQFalse)))
{
// At this point we have a new object on the stack that must be removed
::SqMod::SqPopTopGuard tpg(o.mVM);
// Invoke this function again on the returned value
::SqMod::FormatObjectInContext(o.mVM, -1, 1, ctx);
} else throw fmt::format_error(::SqMod::LastErrorString(o.mVM));
}
else
{
std::string s("closure requires ");
fmt::format_int f(n);
s.append(f.data(), f.size());
s.append(" parameters and cannot be invoked");
throw fmt::format_error(s);
}
} else throw fmt::format_error("failed to retrieve closure information");
// Return the output iterator
return ctx.out();
}
};
/* ------------------------------------------------------------------------------------------------
* Implements formatting for script generators.
*/
template < > struct formatter< ::SqMod::SqGeneratorFmt >
{
std::string mDelimiter{};
/* --------------------------------------------------------------------------------------------
* Parse the specified string range.
*/
auto parse(format_parse_context & ctx) -> decltype(ctx.begin())
{
// Retrieve the range that to be parsed
auto it = ctx.begin(), begin = it, end = ctx.end();
// Find the closing brace
while (it != end && *it != '}') ++it;
// Check if closing brace was found
if (it != end && *it != '}')
{
throw format_error("missing curly brace }");
}
// Check if a delimiter was specified
else if (it != begin)
{
mDelimiter.assign(begin, it); // Store the delimiter
}
// Return an iterator past the end of the parsed range:
return it;
}
/* --------------------------------------------------------------------------------------------
* Perform the requested format.
*/
template < class FormatContext >
auto format(const ::SqMod::SqGeneratorFmt & o, FormatContext & ctx) -> decltype(ctx.out())
{
// Push the generator object on the stack
sq_pushobject(o.mVM, o.mObj);
// At this point we have a new object on the stack that must be removed
::SqMod::SqPopTopGuard opg(o.mVM);
// Keep resuming the generator until null is returned
for (SQInteger n = 0; /*...*/ ; ++n)
{
if (SQ_SUCCEEDED(sq_resume(o.mVM, SQTrue, SQFalse)))
{
// At this point we have a new object on the stack that must be removed
::SqMod::SqPopTopGuard rpg(o.mVM);
// Is the returned value not null?
if (sq_gettype(o.mVM, -1) != OT_NULL)
{
// Do not include the delimiter before the first or after last value
if (n != 0) format_to(ctx.out(), "{:s}", mDelimiter);
// Run the format on the object that was pushed on the stack
::SqMod::FormatObjectInContext(o.mVM, -1, 1, ctx);
} else break; // Stop
} else throw format_error(::SqMod::LastErrorString(o.mVM));
}
// Return the output iterator
return ctx.out();
}
};
/* ------------------------------------------------------------------------------------------------
* Implements formatting for script generators.
*/
template < > struct formatter< ::SqMod::SqWeakReferenceFmt >
{
/* --------------------------------------------------------------------------------------------
* Parse the specified string range.
*/
auto parse(format_parse_context & ctx) -> decltype(ctx.begin()) // NOLINT(readability-convert-member-functions-to-static)
{
// Retrieve the range that to be parsed
auto it = ctx.begin(), begin = it, end = ctx.end();
// Find the closing brace
while (it != end && *it != '}') ++it;
// Check if closing brace was found
if (it != end && *it != '}')
{
throw format_error("missing curly brace }");
}
// Check if a delimiter was specified
else if (it != begin)
{
throw format_error("weak references don't accept format specifiers");
}
// Return an iterator past the end of the parsed range:
return it;
}
/* --------------------------------------------------------------------------------------------
* Perform the requested format.
*/
template < class FormatContext >
auto format(const ::SqMod::SqWeakReferenceFmt & o, FormatContext & ctx) -> decltype(ctx.out())
{
// Push the reference object on the stack
sq_pushobject(o.mVM, o.mObj);
// At this point we have a new object on the stack that must be removed
::SqMod::SqPopTopGuard opg(o.mVM);
// Push the referenced object on the stack
if (SQ_FAILED(sq_getweakrefval(o.mVM, -1)))
{
throw fmt::format_error("failed to retrieve weak reference value");
}
// At this point we have a new object on the stack that must be removed
::SqMod::SqPopTopGuard vpg(o.mVM);
// Run the format on the object that was pushed on the stack
return ::SqMod::FormatObjectInContext(o.mVM, -1, 1, ctx);
}
};
} // Namespace:: fmt

74
module/Library/Format.hpp Normal file
View File

@ -0,0 +1,74 @@
#pragma once
// ------------------------------------------------------------------------------------------------
#include "Core/Utility.hpp"
// ------------------------------------------------------------------------------------------------
#include <fmt/args.h>
#include <fmt/format.h>
// ------------------------------------------------------------------------------------------------
namespace SqMod {
/* ------------------------------------------------------------------------------------------------
* Utility used to generate a formatted string with parameters from the VM stack.
*/
struct FormatContext
{
/* --------------------------------------------------------------------------------------------
* The type of container that will store the argument values.
*/
using Args = fmt::dynamic_format_arg_store< fmt::format_context >;
/* --------------------------------------------------------------------------------------------
* Format arguments container.
*/
Args mArgs{};
/* --------------------------------------------------------------------------------------------
* Script result status.
*/
SQInteger mRes{SQ_OK};
/* --------------------------------------------------------------------------------------------
* Output string buffer.
*/
String mOut{};
/* --------------------------------------------------------------------------------------------
* Default constructor.
*/
FormatContext() = default;
/* --------------------------------------------------------------------------------------------
* Copy constructor. (disabled)
*/
FormatContext(const FormatContext & o) = delete;
/* --------------------------------------------------------------------------------------------
* Move constructor. (disabled)
*/
FormatContext(FormatContext && o) = delete;
/* --------------------------------------------------------------------------------------------
* Destructor.
*/
~FormatContext() = default;
/* --------------------------------------------------------------------------------------------
* Copy assignment operator. (disabled)
*/
FormatContext & operator = (const FormatContext & o) = delete;
/* --------------------------------------------------------------------------------------------
* Move assignment operator. (disabled)
*/
FormatContext & operator = (FormatContext && o) = delete;
/* --------------------------------------------------------------------------------------------
* Process the formatted string.
*/
SQMOD_NODISCARD SQInteger Proc(HSQUIRRELVM vm, SQInteger text, SQInteger args, SQInteger end = -1);
};
} // Namespace:: SqMod

View File

@ -33,6 +33,7 @@ extern void Register_CVehicle(HSQUIRRELVM vm);
// ------------------------------------------------------------------------------------------------
extern void Register_Chrono(HSQUIRRELVM vm);
extern void Register_CURL(HSQUIRRELVM vm);
extern void Register_Format(HSQUIRRELVM vm);
extern void Register_IO(HSQUIRRELVM vm);
extern void Register_MMDB(HSQUIRRELVM vm);
extern void Register_Numeric(HSQUIRRELVM vm);
@ -87,6 +88,7 @@ bool RegisterAPI(HSQUIRRELVM vm)
Register_Chrono(vm);
Register_CURL(vm);
Register_Format(vm);
Register_IO(vm);
Register_MMDB(vm);
Register_Numeric(vm);