2020-07-19 02:55:08 +00:00
|
|
|
#include <memory>
|
|
|
|
#include <Columns/ColumnString.h>
|
|
|
|
#include <DataStreams/materializeBlock.h>
|
|
|
|
#include <DataTypes/DataTypeString.h>
|
|
|
|
#include <Formats/FormatFactory.h>
|
|
|
|
#include <Functions/FunctionFactory.h>
|
|
|
|
#include <Functions/FunctionHelpers.h>
|
|
|
|
#include <Functions/IFunctionImpl.h>
|
|
|
|
#include <IO/WriteBufferFromVector.h>
|
|
|
|
#include <IO/WriteHelpers.h>
|
|
|
|
#include <Processors/Formats/IOutputFormat.h>
|
|
|
|
#include <ext/map.h>
|
|
|
|
|
|
|
|
|
|
|
|
namespace DB
|
|
|
|
{
|
|
|
|
namespace ErrorCodes
|
|
|
|
{
|
|
|
|
extern const int ILLEGAL_TYPE_OF_ARGUMENT;
|
|
|
|
extern const int NUMBER_OF_ARGUMENTS_DOESNT_MATCH;
|
|
|
|
extern const int UNKNOWN_FORMAT;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/** formatRow(<format>, x, y, ...) is a function that allows you to use RowOutputFormat over
|
|
|
|
* several columns to generate a string per row, such as CSV, TSV, JSONEachRow, etc.
|
|
|
|
* formatRowNoNewline(...) trims the newline character of each row.
|
|
|
|
*/
|
|
|
|
|
|
|
|
template <bool no_newline>
|
|
|
|
class FunctionFormatRow : public IFunction
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
static constexpr auto name = no_newline ? "formatRowNoNewline" : "formatRow";
|
|
|
|
|
|
|
|
FunctionFormatRow(const String & format_name_, const Context & context_) : format_name(format_name_), context(context_)
|
|
|
|
{
|
|
|
|
if (!FormatFactory::instance().getAllFormats().count(format_name))
|
|
|
|
throw Exception("Unknown format " + format_name, ErrorCodes::UNKNOWN_FORMAT);
|
|
|
|
}
|
|
|
|
|
|
|
|
String getName() const override { return name; }
|
|
|
|
size_t getNumberOfArguments() const override { return 0; }
|
|
|
|
bool useDefaultImplementationForNulls() const override { return false; }
|
|
|
|
bool useDefaultImplementationForConstants() const override { return true; }
|
|
|
|
ColumnNumbers getArgumentsThatAreAlwaysConstant() const override { return {0}; }
|
|
|
|
|
2020-07-22 02:28:20 +00:00
|
|
|
void executeImpl(Block & block, const ColumnNumbers & arguments, size_t result, size_t input_rows_count) const override
|
2020-07-19 02:55:08 +00:00
|
|
|
{
|
|
|
|
auto col_str = ColumnString::create();
|
|
|
|
ColumnString::Chars & vec = col_str->getChars();
|
|
|
|
WriteBufferFromVector buffer(vec);
|
|
|
|
ColumnString::Offsets & offsets = col_str->getOffsets();
|
|
|
|
offsets.resize(input_rows_count);
|
|
|
|
Block arg_block;
|
|
|
|
for (auto i = 1u; i < arguments.size(); ++i)
|
|
|
|
arg_block.insert(block.getByPosition(arguments[i]));
|
|
|
|
materializeBlockInplace(arg_block);
|
|
|
|
auto out = FormatFactory::instance().getOutputFormat(format_name, buffer, arg_block, context, [&](const Columns &, size_t row)
|
|
|
|
{
|
|
|
|
if constexpr (no_newline)
|
2020-07-22 02:28:20 +00:00
|
|
|
{
|
|
|
|
// replace '\n' with '\0'
|
2020-07-25 02:29:47 +00:00
|
|
|
if (buffer.position() != buffer.buffer().begin() && buffer.position()[-1] == '\n')
|
|
|
|
buffer.position()[-1] = '\0';
|
2020-07-22 02:28:20 +00:00
|
|
|
}
|
2020-07-19 02:55:08 +00:00
|
|
|
else
|
|
|
|
writeChar('\0', buffer);
|
|
|
|
offsets[row] = buffer.count();
|
|
|
|
});
|
|
|
|
out->write(arg_block);
|
|
|
|
block.getByPosition(result).column = std::move(col_str);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
String format_name;
|
|
|
|
const Context & context;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <bool no_newline>
|
|
|
|
class FormatRowOverloadResolver : public IFunctionOverloadResolverImpl
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
static constexpr auto name = no_newline ? "formatRowNoNewline" : "formatRow";
|
|
|
|
static FunctionOverloadResolverImplPtr create(const Context & context) { return std::make_unique<FormatRowOverloadResolver>(context); }
|
|
|
|
explicit FormatRowOverloadResolver(const Context & context_) : context(context_) { }
|
|
|
|
String getName() const override { return name; }
|
|
|
|
bool isVariadic() const override { return true; }
|
|
|
|
size_t getNumberOfArguments() const override { return 0; }
|
|
|
|
ColumnNumbers getArgumentsThatAreAlwaysConstant() const override { return {0}; }
|
|
|
|
bool useDefaultImplementationForNulls() const override { return false; }
|
|
|
|
|
|
|
|
FunctionBaseImplPtr build(const ColumnsWithTypeAndName & arguments, const DataTypePtr & return_type) const override
|
|
|
|
{
|
|
|
|
if (arguments.size() < 2)
|
|
|
|
throw Exception(
|
|
|
|
"Function " + getName() + " requires at least two arguments: the format name and its output expression(s)",
|
|
|
|
ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH);
|
|
|
|
|
|
|
|
if (const auto * name_col = checkAndGetColumnConst<ColumnString>(arguments.at(0).column.get()))
|
|
|
|
return std::make_unique<DefaultFunction>(
|
|
|
|
std::make_shared<FunctionFormatRow<no_newline>>(name_col->getValue<String>(), context),
|
|
|
|
ext::map<DataTypes>(arguments, [](const auto & elem) { return elem.type; }),
|
|
|
|
return_type);
|
|
|
|
else
|
|
|
|
throw Exception("First argument to " + getName() + " must be a format name", ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT);
|
|
|
|
}
|
|
|
|
|
|
|
|
DataTypePtr getReturnType(const DataTypes &) const override { return std::make_shared<DataTypeString>(); }
|
|
|
|
|
|
|
|
private:
|
|
|
|
const Context & context;
|
|
|
|
};
|
|
|
|
|
|
|
|
void registerFunctionFormatRow(FunctionFactory & factory)
|
|
|
|
{
|
|
|
|
factory.registerFunction<FormatRowOverloadResolver<true>>();
|
|
|
|
factory.registerFunction<FormatRowOverloadResolver<false>>();
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|