summaryrefslogtreecommitdiffstats
path: root/src/arrow/c_glib/arrow-glib/table.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/arrow/c_glib/arrow-glib/table.cpp')
-rw-r--r--src/arrow/c_glib/arrow-glib/table.cpp965
1 files changed, 965 insertions, 0 deletions
diff --git a/src/arrow/c_glib/arrow-glib/table.cpp b/src/arrow/c_glib/arrow-glib/table.cpp
new file mode 100644
index 000000000..f303c0999
--- /dev/null
+++ b/src/arrow/c_glib/arrow-glib/table.cpp
@@ -0,0 +1,965 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements. See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership. The ASF licenses this file
+ * to you under the Apache License, Version 2.0 (the
+ * "License"); you may not use this file except in compliance
+ * with the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing,
+ * software distributed under the License is distributed on an
+ * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ * KIND, either express or implied. See the License for the
+ * specific language governing permissions and limitations
+ * under the License.
+ */
+
+#include <arrow-glib/array.hpp>
+#include <arrow-glib/chunked-array.hpp>
+#include <arrow-glib/enums.h>
+#include <arrow-glib/error.hpp>
+#include <arrow-glib/field.hpp>
+#include <arrow-glib/internal-index.hpp>
+#include <arrow-glib/output-stream.hpp>
+#include <arrow-glib/record-batch.hpp>
+#include <arrow-glib/schema.hpp>
+#include <arrow-glib/table.hpp>
+
+#include <sstream>
+
+G_BEGIN_DECLS
+
+/**
+ * SECTION: table
+ * @section_id: GArrowTable
+ * @title: GArrowTable
+ * @short_description: Table class
+ * @include: arrow-glib/arrow-glib.h
+ *
+ * #GArrowTableConcatenateOptions is a class for customizing
+ * garrow_table_concatenate() behavior.
+ *
+ * #GArrowTable is a class for table. Table has zero or more
+ * #GArrowChunkedArrays and zero or more records.
+ *
+ * #GArrowFeatherWriteProperties is a class to customize how to write
+ * Feather data.
+ */
+
+typedef struct GArrowTableConcatenateOptionsPrivate_ {
+ arrow::ConcatenateTablesOptions options;
+} GArrowTableConcatenateOptionsPrivate;
+
+enum {
+ PROP_UNIFY_SCHEMAS = 1,
+ PROP_PROMOTE_NULLABILITY,
+};
+
+G_DEFINE_TYPE_WITH_PRIVATE(GArrowTableConcatenateOptions,
+ garrow_table_concatenate_options,
+ G_TYPE_OBJECT)
+
+#define GARROW_TABLE_CONCATENATE_OPTIONS_GET_PRIVATE(obj) \
+ static_cast<GArrowTableConcatenateOptionsPrivate *>( \
+ garrow_table_concatenate_options_get_instance_private( \
+ GARROW_TABLE_CONCATENATE_OPTIONS(obj)))
+
+static void
+garrow_table_concatenate_options_finalize(GObject *object)
+{
+ auto priv = GARROW_TABLE_CONCATENATE_OPTIONS_GET_PRIVATE(object);
+ priv->options.~ConcatenateTablesOptions();
+ G_OBJECT_CLASS(garrow_table_concatenate_options_parent_class)->finalize(object);
+}
+
+static void
+garrow_table_concatenate_options_set_property(GObject *object,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ auto priv = GARROW_TABLE_CONCATENATE_OPTIONS_GET_PRIVATE(object);
+
+ switch (prop_id) {
+ case PROP_UNIFY_SCHEMAS:
+ priv->options.unify_schemas = g_value_get_boolean(value);
+ break;
+ case PROP_PROMOTE_NULLABILITY:
+ priv->options.field_merge_options.promote_nullability =
+ g_value_get_boolean(value);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+garrow_table_concatenate_options_get_property(GObject *object,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ auto priv = GARROW_TABLE_CONCATENATE_OPTIONS_GET_PRIVATE(object);
+
+ switch (prop_id) {
+ case PROP_UNIFY_SCHEMAS:
+ g_value_set_boolean(value, priv->options.unify_schemas);
+ break;
+ case PROP_PROMOTE_NULLABILITY:
+ g_value_set_boolean(value,
+ priv->options.field_merge_options.promote_nullability);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+garrow_table_concatenate_options_init(GArrowTableConcatenateOptions *object)
+{
+ auto priv = GARROW_TABLE_CONCATENATE_OPTIONS_GET_PRIVATE(object);
+ new(&(priv->options)) arrow::ConcatenateTablesOptions;
+}
+
+static void
+garrow_table_concatenate_options_class_init(
+ GArrowTableConcatenateOptionsClass *klass)
+{
+ auto gobject_class = G_OBJECT_CLASS(klass);
+ gobject_class->finalize = garrow_table_concatenate_options_finalize;
+ gobject_class->set_property = garrow_table_concatenate_options_set_property;
+ gobject_class->get_property = garrow_table_concatenate_options_get_property;
+
+ GParamSpec *spec;
+
+ auto default_options = arrow::ConcatenateTablesOptions::Defaults();
+
+ /**
+ * GArrowTableConcatenateOptions:unify-schemas:
+ *
+ * If true, the schemas of the tables will be first unified with
+ * fields of the same name being merged, according to
+ * #GArrowTableConcatenateOptions:promote-nullability, then each
+ * table will be promoted to the unified schema before being
+ * concatenated.
+ *
+ * Otherwise, all tables should have the same schema. Each column in
+ * the output table is the result of concatenating the corresponding
+ * columns in all input tables.
+ *
+ * Since: 6.0.0
+ */
+ spec = g_param_spec_boolean("unify-schemas",
+ "Unify schemas",
+ "Whether unifying schemas or not",
+ default_options.unify_schemas,
+ static_cast<GParamFlags>(G_PARAM_READWRITE));
+ g_object_class_install_property(gobject_class, PROP_UNIFY_SCHEMAS, spec);
+
+ /**
+ * GArrowTableConcatenateOptions:promote-nullability:
+ *
+ * If true, a #GArrowField of #GArrowNullDataType can be unified
+ * with a #GArrowField of another type. The unified field will be of
+ * the other type and become nullable. Nullability will be promoted
+ * to the looser option (nullable if one is not nullable).
+ *
+ * Since: 6.0.0
+ */
+ spec = g_param_spec_boolean("promote-nullability",
+ "Promote nullability",
+ "Whether promoting nullability or not",
+ default_options.field_merge_options.promote_nullability,
+ static_cast<GParamFlags>(G_PARAM_READWRITE));
+ g_object_class_install_property(gobject_class, PROP_PROMOTE_NULLABILITY, spec);
+}
+
+/**
+ * garrow_table_concatenate_options_new:
+ *
+ * Returns: A newly created #GArrowTableConcatenateOptions.
+ *
+ * Since: 6.0.0
+ */
+GArrowTableConcatenateOptions *
+garrow_table_concatenate_options_new(void)
+{
+ return GARROW_TABLE_CONCATENATE_OPTIONS(
+ g_object_new(GARROW_TYPE_TABLE_CONCATENATE_OPTIONS,
+ NULL));
+}
+
+
+typedef struct GArrowTablePrivate_ {
+ std::shared_ptr<arrow::Table> table;
+} GArrowTablePrivate;
+
+enum {
+ PROP_TABLE = 1,
+};
+
+G_DEFINE_TYPE_WITH_PRIVATE(GArrowTable,
+ garrow_table,
+ G_TYPE_OBJECT)
+
+#define GARROW_TABLE_GET_PRIVATE(obj) \
+ static_cast<GArrowTablePrivate *>( \
+ garrow_table_get_instance_private( \
+ GARROW_TABLE(obj)))
+
+static void
+garrow_table_finalize(GObject *object)
+{
+ auto priv = GARROW_TABLE_GET_PRIVATE(object);
+
+ priv->table.~shared_ptr();
+
+ G_OBJECT_CLASS(garrow_table_parent_class)->finalize(object);
+}
+
+static void
+garrow_table_set_property(GObject *object,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ auto priv = GARROW_TABLE_GET_PRIVATE(object);
+
+ switch (prop_id) {
+ case PROP_TABLE:
+ priv->table =
+ *static_cast<std::shared_ptr<arrow::Table> *>(g_value_get_pointer(value));
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+garrow_table_get_property(GObject *object,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ switch (prop_id) {
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+garrow_table_init(GArrowTable *object)
+{
+ auto priv = GARROW_TABLE_GET_PRIVATE(object);
+ new(&priv->table) std::shared_ptr<arrow::Table>;
+}
+
+static void
+garrow_table_class_init(GArrowTableClass *klass)
+{
+ GObjectClass *gobject_class;
+ GParamSpec *spec;
+
+ gobject_class = G_OBJECT_CLASS(klass);
+
+ gobject_class->finalize = garrow_table_finalize;
+ gobject_class->set_property = garrow_table_set_property;
+ gobject_class->get_property = garrow_table_get_property;
+
+ spec = g_param_spec_pointer("table",
+ "Table",
+ "The raw std::shared_ptr<arrow::Table> *",
+ static_cast<GParamFlags>(G_PARAM_WRITABLE |
+ G_PARAM_CONSTRUCT_ONLY));
+ g_object_class_install_property(gobject_class, PROP_TABLE, spec);
+}
+
+/**
+ * garrow_table_new_values: (skip)
+ * @schema: The schema of the table.
+ * @values: The values of the table. All values must be instance of
+ * the same class. Available classes are #GArrowChunkedArray,
+ * #GArrowArray and #GArrowRecordBatch.
+ * @error: (nullable): Return location for a #GError or %NULL.
+ *
+ * Returns: (nullable): A newly created #GArrowTable or %NULL on error.
+ *
+ * Since: 0.12.0
+ */
+GArrowTable *
+garrow_table_new_values(GArrowSchema *schema,
+ GList *values,
+ GError **error)
+{
+ const auto context = "[table][new][values]";
+ auto arrow_schema = garrow_schema_get_raw(schema);
+ std::vector<std::shared_ptr<arrow::ChunkedArray>> arrow_chunked_arrays;
+ std::vector<std::shared_ptr<arrow::Array>> arrow_arrays;
+ std::vector<std::shared_ptr<arrow::RecordBatch>> arrow_record_batches;
+ for (GList *node = values; node; node = node->next) {
+ if (GARROW_IS_CHUNKED_ARRAY(node->data)) {
+ auto chunked_array = GARROW_CHUNKED_ARRAY(node->data);
+ arrow_chunked_arrays.push_back(garrow_chunked_array_get_raw(chunked_array));
+ } else if (GARROW_IS_ARRAY(node->data)) {
+ auto array = GARROW_ARRAY(node->data);
+ arrow_arrays.push_back(garrow_array_get_raw(array));
+ } else if (GARROW_IS_RECORD_BATCH(node->data)) {
+ auto record_batch = GARROW_RECORD_BATCH(node->data);
+ arrow_record_batches.push_back(garrow_record_batch_get_raw(record_batch));
+ } else {
+ g_set_error(error,
+ GARROW_ERROR,
+ GARROW_ERROR_INVALID,
+ "%s: %s",
+ context,
+ "value must be one of "
+ "GArrowChunkedArray, GArrowArray and GArrowRecordBatch");
+ return NULL;
+ }
+ }
+
+ size_t n_types = 0;
+ if (!arrow_chunked_arrays.empty()) {
+ ++n_types;
+ }
+ if (!arrow_arrays.empty()) {
+ ++n_types;
+ }
+ if (!arrow_record_batches.empty()) {
+ ++n_types;
+ }
+ if (n_types > 1) {
+ g_set_error(error,
+ GARROW_ERROR,
+ GARROW_ERROR_INVALID,
+ "%s: %s",
+ context,
+ "all values must be the same objects of "
+ "GArrowChunkedArray, GArrowArray or GArrowRecordBatch");
+ return NULL;
+ }
+
+ if (!arrow_chunked_arrays.empty()) {
+ auto arrow_table = arrow::Table::Make(arrow_schema,
+ std::move(arrow_chunked_arrays));
+ auto status = arrow_table->Validate();
+ if (garrow_error_check(error, status, context)) {
+ return garrow_table_new_raw(&arrow_table);
+ } else {
+ return NULL;
+ }
+ } else if (!arrow_arrays.empty()) {
+ auto arrow_table = arrow::Table::Make(arrow_schema, std::move(arrow_arrays));
+ auto status = arrow_table->Validate();
+ if (garrow_error_check(error, status, context)) {
+ return garrow_table_new_raw(&arrow_table);
+ } else {
+ return NULL;
+ }
+ } else {
+ auto maybe_table = arrow::Table::FromRecordBatches(
+ arrow_schema, std::move(arrow_record_batches));
+ if (garrow::check(error, maybe_table, context)) {
+ return garrow_table_new_raw(&(*maybe_table));
+ } else {
+ return NULL;
+ }
+ }
+}
+
+/**
+ * garrow_table_new_chunked_arrays:
+ * @schema: The schema of the table.
+ * @chunked_arrays: (array length=n_chunked_arrays): The chunked arrays of
+ * the table.
+ * @n_chunked_arrays: The number of chunked arrays.
+ * @error: (nullable): Return location for a #GError or %NULL.
+ *
+ * Returns: (nullable): A newly created #GArrowTable or %NULL on error.
+ *
+ * Since: 0.15.0
+ */
+GArrowTable *
+garrow_table_new_chunked_arrays(GArrowSchema *schema,
+ GArrowChunkedArray **chunked_arrays,
+ gsize n_chunked_arrays,
+ GError **error)
+{
+ auto arrow_schema = garrow_schema_get_raw(schema);
+ std::vector<std::shared_ptr<arrow::ChunkedArray>> arrow_chunked_arrays;
+ for (gsize i = 0; i < n_chunked_arrays; ++i) {
+ auto arrow_chunked_array = garrow_chunked_array_get_raw(chunked_arrays[i]);
+ arrow_chunked_arrays.push_back(arrow_chunked_array);
+ }
+
+ auto arrow_table = arrow::Table::Make(arrow_schema, arrow_chunked_arrays);
+ auto status = arrow_table->Validate();
+ if (garrow_error_check(error, status, "[table][new][chunked-arrays]")) {
+ return garrow_table_new_raw(&arrow_table);
+ } else {
+ return NULL;
+ }
+}
+
+/**
+ * garrow_table_new_arrays:
+ * @schema: The schema of the table.
+ * @arrays: (array length=n_arrays): The arrays of the table.
+ * @n_arrays: The number of arrays.
+ * @error: (nullable): Return location for a #GError or %NULL.
+ *
+ * Returns: (nullable): A newly created #GArrowTable or %NULL on error.
+ *
+ * Since: 0.12.0
+ */
+GArrowTable *
+garrow_table_new_arrays(GArrowSchema *schema,
+ GArrowArray **arrays,
+ gsize n_arrays,
+ GError **error)
+{
+ auto arrow_schema = garrow_schema_get_raw(schema);
+ std::vector<std::shared_ptr<arrow::Array>> arrow_arrays;
+ for (gsize i = 0; i < n_arrays; ++i) {
+ arrow_arrays.push_back(garrow_array_get_raw(arrays[i]));
+ }
+
+ auto arrow_table = arrow::Table::Make(arrow_schema, arrow_arrays);
+ auto status = arrow_table->Validate();
+ if (garrow_error_check(error, status, "[table][new][arrays]")) {
+ return garrow_table_new_raw(&arrow_table);
+ } else {
+ return NULL;
+ }
+}
+
+/**
+ * garrow_table_new_record_batches:
+ * @schema: The schema of the table.
+ * @record_batches: (array length=n_record_batches): The record batches
+ * that have data for the table.
+ * @n_record_batches: The number of record batches.
+ * @error: (nullable): Return location for a #GError or %NULL.
+ *
+ * Returns: (nullable): A newly created #GArrowTable or %NULL on error.
+ *
+ * Since: 0.12.0
+ */
+GArrowTable *
+garrow_table_new_record_batches(GArrowSchema *schema,
+ GArrowRecordBatch **record_batches,
+ gsize n_record_batches,
+ GError **error)
+{
+ auto arrow_schema = garrow_schema_get_raw(schema);
+ std::vector<std::shared_ptr<arrow::RecordBatch>> arrow_record_batches;
+ for (gsize i = 0; i < n_record_batches; ++i) {
+ auto arrow_record_batch = garrow_record_batch_get_raw(record_batches[i]);
+ arrow_record_batches.push_back(arrow_record_batch);
+ }
+
+ auto maybe_table = arrow::Table::FromRecordBatches(arrow_schema,
+ arrow_record_batches);
+ if (garrow::check(error, maybe_table, "[table][new][record-batches]")) {
+ return garrow_table_new_raw(&(*maybe_table));
+ } else {
+ return NULL;
+ }
+}
+
+/**
+ * garrow_table_equal:
+ * @table: A #GArrowTable.
+ * @other_table: A #GArrowTable to be compared.
+ *
+ * Returns: %TRUE if both of them have the same data, %FALSE
+ * otherwise.
+ *
+ * Since: 0.4.0
+ */
+gboolean
+garrow_table_equal(GArrowTable *table, GArrowTable *other_table)
+{
+ const auto arrow_table = garrow_table_get_raw(table);
+ const auto arrow_other_table = garrow_table_get_raw(other_table);
+ return arrow_table->Equals(*arrow_other_table);
+}
+
+/**
+ * garrow_table_equal_metadata:
+ * @table: A #GArrowTable.
+ * @other_table: A #GArrowTable to be compared.
+ * @check_metadata: Whether to compare metadata.
+ *
+ * Returns: %TRUE if both of them have the same data, %FALSE
+ * otherwise.
+ *
+ * Since: 0.17.0
+ */
+gboolean
+garrow_table_equal_metadata(GArrowTable *table,
+ GArrowTable *other_table,
+ gboolean check_metadata)
+{
+ const auto arrow_table = garrow_table_get_raw(table);
+ const auto arrow_other_table = garrow_table_get_raw(other_table);
+ return arrow_table->Equals(*arrow_other_table, check_metadata);
+}
+
+/**
+ * garrow_table_get_schema:
+ * @table: A #GArrowTable.
+ *
+ * Returns: (transfer full): The schema of the table.
+ */
+GArrowSchema *
+garrow_table_get_schema(GArrowTable *table)
+{
+ const auto arrow_table = garrow_table_get_raw(table);
+ auto arrow_schema = arrow_table->schema();
+ return garrow_schema_new_raw(&arrow_schema);
+}
+
+/**
+ * garrow_table_get_column_data:
+ * @table: A #GArrowTable.
+ * @i: The index of the target column. If it's negative, index is
+ * counted backward from the end of the columns. `-1` means the last
+ * column.
+ *
+ * Returns: (nullable) (transfer full): The i-th column's data in the table.
+ *
+ * Since: 0.15.0
+ */
+GArrowChunkedArray *
+garrow_table_get_column_data(GArrowTable *table,
+ gint i)
+{
+ const auto &arrow_table = garrow_table_get_raw(table);
+ if (!garrow_internal_index_adjust(i, arrow_table->num_columns())) {
+ return NULL;
+ }
+ auto arrow_column = arrow_table->column(i);
+ return garrow_chunked_array_new_raw(&arrow_column);
+}
+
+/**
+ * garrow_table_get_n_columns:
+ * @table: A #GArrowTable.
+ *
+ * Returns: The number of columns in the table.
+ */
+guint
+garrow_table_get_n_columns(GArrowTable *table)
+{
+ const auto arrow_table = garrow_table_get_raw(table);
+ return arrow_table->num_columns();
+}
+
+/**
+ * garrow_table_get_n_rows:
+ * @table: A #GArrowTable.
+ *
+ * Returns: The number of rows in the table.
+ */
+guint64
+garrow_table_get_n_rows(GArrowTable *table)
+{
+ const auto arrow_table = garrow_table_get_raw(table);
+ return arrow_table->num_rows();
+}
+
+/**
+ * garrow_table_add_column:
+ * @table: A #GArrowTable.
+ * @i: The index of the new column.
+ * @field: The field for the column to be added.
+ * @chunked_array: The column data to be added.
+ * @error: (nullable): Return location for a #GError or %NULL.
+ *
+ * Returns: (nullable) (transfer full): The newly allocated
+ * #GArrowTable that has a new column or %NULL on error.
+ *
+ * Since: 0.15.0
+ */
+GArrowTable *
+garrow_table_add_column(GArrowTable *table,
+ guint i,
+ GArrowField *field,
+ GArrowChunkedArray *chunked_array,
+ GError **error)
+{
+ const auto arrow_table = garrow_table_get_raw(table);
+ const auto arrow_field = garrow_field_get_raw(field);
+ const auto arrow_chunked_array = garrow_chunked_array_get_raw(chunked_array);
+ auto maybe_new_table = arrow_table->AddColumn(i,
+ arrow_field,
+ arrow_chunked_array);
+ if (garrow::check(error, maybe_new_table, "[table][add-column]")) {
+ return garrow_table_new_raw(&(*maybe_new_table));
+ } else {
+ return NULL;
+ }
+}
+
+/**
+ * garrow_table_remove_column:
+ * @table: A #GArrowTable.
+ * @i: The index of the column to be removed.
+ * @error: (nullable): Return location for a #GError or %NULL.
+ *
+ * Returns: (nullable) (transfer full): The newly allocated
+ * #GArrowTable that doesn't have the column or %NULL on error.
+ *
+ * Since: 0.3.0
+ */
+GArrowTable *
+garrow_table_remove_column(GArrowTable *table,
+ guint i,
+ GError **error)
+{
+ const auto arrow_table = garrow_table_get_raw(table);
+ auto maybe_new_table = arrow_table->RemoveColumn(i);
+ if (garrow::check(error, maybe_new_table, "[table][remove-column]")) {
+ return garrow_table_new_raw(&(*maybe_new_table));
+ } else {
+ return NULL;
+ }
+}
+
+/**
+ * garrow_table_replace_column:
+ * @table: A #GArrowTable.
+ * @i: The index of the column to be replaced.
+ * @field: The field for the new column.
+ * @chunked_array: The newly added column data.
+ * @error: (nullable): Return location for a #GError or %NULL.
+ *
+ * Returns: (nullable) (transfer full): The newly allocated
+ * #GArrowTable that has @column as the @i-th column or %NULL on
+ * error.
+ *
+ * Since: 0.15.0
+ */
+GArrowTable *
+garrow_table_replace_column(GArrowTable *table,
+ guint i,
+ GArrowField *field,
+ GArrowChunkedArray *chunked_array,
+ GError **error)
+{
+ const auto arrow_table = garrow_table_get_raw(table);
+ const auto arrow_field = garrow_field_get_raw(field);
+ const auto arrow_chunked_array = garrow_chunked_array_get_raw(chunked_array);
+ auto maybe_new_table = arrow_table->SetColumn(i,
+ arrow_field,
+ arrow_chunked_array);
+ if (garrow::check(error, maybe_new_table, "[table][replace-column]")) {
+ return garrow_table_new_raw(&(*maybe_new_table));
+ } else {
+ return NULL;
+ }
+}
+
+/**
+ * garrow_table_to_string:
+ * @table: A #GArrowTable.
+ * @error: (nullable): Return location for a #GError or %NULL.
+ *
+ * Returns: (nullable):
+ * The formatted table content or %NULL on error.
+ *
+ * It should be freed with g_free() when no longer needed.
+ *
+ * Since: 0.12.0
+ */
+gchar *
+garrow_table_to_string(GArrowTable *table, GError **error)
+{
+ const auto arrow_table = garrow_table_get_raw(table);
+ return g_strdup(arrow_table->ToString().c_str());
+}
+
+/**
+ * garrow_table_concatenate:
+ * @table: A #GArrowTable.
+ * @other_tables: (element-type GArrowTable): The tables to be concatenated.
+ * @options: (nullable): The options to customize concatenation.
+ * @error: (nullable): Return location for a #GError or %NULL.
+ *
+ * Returns: (nullable) (transfer full): The table concatenated vertically.
+ *
+ * Since: 0.14.0
+ */
+GArrowTable *
+garrow_table_concatenate(GArrowTable *table,
+ GList *other_tables,
+ GArrowTableConcatenateOptions *options,
+ GError **error)
+{
+ auto arrow_table = garrow_table_get_raw(table);
+ std::vector<std::shared_ptr<arrow::Table>> arrow_tables = { arrow_table };
+ for (auto node = other_tables; node; node = g_list_next(node)) {
+ auto arrow_other_table = garrow_table_get_raw(GARROW_TABLE(node->data));
+ arrow_tables.push_back(arrow_other_table);
+ }
+ auto arrow_options = arrow::ConcatenateTablesOptions::Defaults();
+ if (options) {
+ auto options_priv = GARROW_TABLE_CONCATENATE_OPTIONS_GET_PRIVATE(options);
+ arrow_options = options_priv->options;
+ }
+ auto arrow_concatenated_table_result =
+ arrow::ConcatenateTables(arrow_tables, arrow_options);
+ if (garrow::check(error,
+ arrow_concatenated_table_result,
+ "[table][concatenate]")) {
+ auto arrow_concatenated_table = std::move(*arrow_concatenated_table_result);
+ return garrow_table_new_raw(&arrow_concatenated_table);
+ } else {
+ return NULL;
+ }
+}
+
+/**
+ * garrow_table_slice:
+ * @table: A #GArrowTable.
+ * @offset: The offset of sub #GArrowTable. If the offset is negative,
+ * the offset is counted from the last.
+ * @length: The length of sub #GArrowTable.
+ *
+ * Returns: (transfer full): The sub #GArrowTable. It covers
+ * only from `offset` to `offset + length` range. The sub
+ * #GArrowTable shares values with the base
+ * #GArrowTable.
+ *
+ * Since: 0.14.0
+ */
+GArrowTable *
+garrow_table_slice(GArrowTable *table,
+ gint64 offset,
+ gint64 length)
+{
+ const auto arrow_table = garrow_table_get_raw(table);
+ if (offset < 0) {
+ offset += arrow_table->num_rows();
+ }
+ auto arrow_sub_table = arrow_table->Slice(offset, length);
+ return garrow_table_new_raw(&arrow_sub_table);
+}
+
+/**
+ * garrow_table_combine_chunks:
+ * @table: A #GArrowTable.
+ * @error: (nullable): Return location for a #GError or %NULL.
+ *
+ * Returns: (nullable) (transfer full): The #GArrowTable with
+ * chunks combined, or %NULL on error.
+ *
+ * Since: 0.16.0
+ */
+GArrowTable *
+garrow_table_combine_chunks(GArrowTable *table,
+ GError **error)
+{
+ const auto arrow_table = garrow_table_get_raw(table);
+
+ auto maybe_new_table = arrow_table->CombineChunks();
+ if (garrow::check(error, maybe_new_table, "[table][combine-chunks]")) {
+ return garrow_table_new_raw(&(*maybe_new_table));
+ } else {
+ return NULL;
+ }
+}
+
+
+typedef struct GArrowFeatherWritePropertiesPrivate_ {
+ arrow::ipc::feather::WriteProperties properties;
+} GArrowFeatherWritePropertiesPrivate;
+
+enum {
+ PROP_VERSION = 1,
+ PROP_CHUNK_SIZE,
+ PROP_COMPRESSION,
+ PROP_COMPRESSION_LEVEL,
+};
+
+G_DEFINE_TYPE_WITH_PRIVATE(GArrowFeatherWriteProperties,
+ garrow_feather_write_properties,
+ G_TYPE_OBJECT)
+
+#define GARROW_FEATHER_WRITE_PROPERTIES_GET_PRIVATE(obj) \
+ static_cast<GArrowFeatherWritePropertiesPrivate *>( \
+ garrow_feather_write_properties_get_instance_private( \
+ GARROW_FEATHER_WRITE_PROPERTIES(obj)))
+
+static void
+garrow_feather_write_properties_finalize(GObject *object)
+{
+ auto priv = GARROW_FEATHER_WRITE_PROPERTIES_GET_PRIVATE(object);
+
+ priv->properties.~WriteProperties();
+
+ G_OBJECT_CLASS(garrow_feather_write_properties_parent_class)->finalize(object);
+}
+
+static void
+garrow_feather_write_properties_set_property(GObject *object,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ auto priv = GARROW_FEATHER_WRITE_PROPERTIES_GET_PRIVATE(object);
+
+ switch (prop_id) {
+ case PROP_COMPRESSION:
+ priv->properties.compression =
+ static_cast<arrow::Compression::type>(g_value_get_enum(value));
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+garrow_feather_write_properties_get_property(GObject *object,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ auto priv = GARROW_FEATHER_WRITE_PROPERTIES_GET_PRIVATE(object);
+
+ switch (prop_id) {
+ case PROP_COMPRESSION:
+ g_value_set_enum(value, priv->properties.compression);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+garrow_feather_write_properties_init(GArrowFeatherWriteProperties *object)
+{
+ auto priv = GARROW_FEATHER_WRITE_PROPERTIES_GET_PRIVATE(object);
+ new(&priv->properties) arrow::ipc::feather::WriteProperties;
+ priv->properties = arrow::ipc::feather::WriteProperties::Defaults();
+}
+
+static void
+garrow_feather_write_properties_class_init(GArrowFeatherWritePropertiesClass *klass)
+{
+ auto gobject_class = G_OBJECT_CLASS(klass);
+
+ gobject_class->finalize = garrow_feather_write_properties_finalize;
+ gobject_class->set_property = garrow_feather_write_properties_set_property;
+ gobject_class->get_property = garrow_feather_write_properties_get_property;
+
+ auto properties = arrow::ipc::feather::WriteProperties::Defaults();
+ GParamSpec *spec;
+ // TODO: version
+ // TODO: chunk_size
+
+ /**
+ * GArrowFeatherWriteProperties:compression:
+ *
+ * Compression type to use. Only
+ * %GARROW_COMPRESSION_TYPE_UNCOMPRESSED,
+ * %GARROW_COMPRESSION_TYPE_LZ4 and %GARROW_COMPRESSION_TYPE_ZSTD
+ * are supported. The default compression is
+ * %GARROW_COMPRESSION_TYPE_LZ4 if Apache Arrow C++ is built with
+ * support for it, otherwise %GARROW_COMPRESSION_TYPE_UNCOMPRESSED.
+ * %GARROW_COMPRESSION_TYPE_UNCOMPRESSED is set as the object
+ * default here.
+ *
+ * Since: 0.17.0
+ */
+ spec = g_param_spec_enum("compression",
+ "Compression",
+ "The compression type to use",
+ GARROW_TYPE_COMPRESSION_TYPE,
+ properties.compression,
+ static_cast<GParamFlags>(G_PARAM_READWRITE));
+ g_object_class_install_property(gobject_class, PROP_COMPRESSION, spec);
+ // TODO: compression_level
+}
+
+/**
+ * garrow_feather_write_properties_new:
+ *
+ * Returns: A newly created #GArrowFeatherWriteProperties.
+ *
+ * Since: 0.17.0
+ */
+GArrowFeatherWriteProperties *
+garrow_feather_write_properties_new(void)
+{
+ auto properties = g_object_new(GARROW_TYPE_FEATHER_WRITE_PROPERTIES, NULL);
+ return GARROW_FEATHER_WRITE_PROPERTIES(properties);
+}
+
+/**
+ * garrow_table_write_as_feather:
+ * @table: A #GArrowTable.
+ * @sink: The output.
+ * @properties: (nullable): The properties for this write.
+ * @error: (nullable): Return location for a #GError or %NULL.
+ *
+ * Writes the @table as Feather format data to the @sink.
+ *
+ * Returns: %TRUE on success, %FALSE if there was an error.
+ *
+ * Since: 0.17.0
+ */
+gboolean
+garrow_table_write_as_feather(GArrowTable *table,
+ GArrowOutputStream *sink,
+ GArrowFeatherWriteProperties *properties,
+ GError **error)
+{
+ auto arrow_table = garrow_table_get_raw(table);
+ auto arrow_sink = garrow_output_stream_get_raw(sink);
+ arrow::Status status;
+ if (properties) {
+ auto arrow_properties = garrow_feather_write_properties_get_raw(properties);
+ status = arrow::ipc::feather::WriteTable(*arrow_table,
+ arrow_sink.get(),
+ *arrow_properties);
+ } else {
+ status = arrow::ipc::feather::WriteTable(*arrow_table, arrow_sink.get());
+ }
+ return garrow::check(error, status, "[feather-write-file]");
+}
+
+G_END_DECLS
+
+GArrowTable *
+garrow_table_new_raw(std::shared_ptr<arrow::Table> *arrow_table)
+{
+ auto table = GARROW_TABLE(g_object_new(GARROW_TYPE_TABLE,
+ "table", arrow_table,
+ NULL));
+ return table;
+}
+
+std::shared_ptr<arrow::Table>
+garrow_table_get_raw(GArrowTable *table)
+{
+ auto priv = GARROW_TABLE_GET_PRIVATE(table);
+ return priv->table;
+}
+
+arrow::ipc::feather::WriteProperties *
+garrow_feather_write_properties_get_raw(GArrowFeatherWriteProperties *properties)
+{
+ auto priv = GARROW_FEATHER_WRITE_PROPERTIES_GET_PRIVATE(properties);
+ return &(priv->properties);
+}