summaryrefslogtreecommitdiff
path: root/examples/gridfs.c
diff options
context:
space:
mode:
Diffstat (limited to 'examples/gridfs.c')
-rw-r--r--examples/gridfs.c413
1 files changed, 413 insertions, 0 deletions
diff --git a/examples/gridfs.c b/examples/gridfs.c
new file mode 100644
index 0000000..2d19aee
--- /dev/null
+++ b/examples/gridfs.c
@@ -0,0 +1,413 @@
+/* gridfs.c - A GridFS utility; example application
+ * Copyright 2011, 2012 Gergely Nagy <algernon@balabit.hu>
+ *
+ * Licensed 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 <mongo.h>
+
+#include <unistd.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <errno.h>
+#include <string.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <fcntl.h>
+#include <sys/mman.h>
+
+typedef struct
+{
+ gchar *host;
+ gint port;
+ gchar *db;
+ gchar *coll;
+ gchar *ns;
+ gboolean verbose;
+ gboolean slaveok;
+ gboolean master_sync;
+} config_t;
+
+#define VLOG(...) { if (config->verbose) fprintf (stderr, __VA_ARGS__); }
+
+void
+mongo_gridfs_error (int e)
+{
+ fprintf (stderr, "Error encountered: %s\n", strerror (e));
+ exit (1);
+}
+
+mongo_sync_gridfs *
+mongo_gridfs_connect (config_t *config)
+{
+ mongo_sync_connection *conn;
+ mongo_sync_gridfs *gfs;
+
+ VLOG ("Connecting to %s:%d/%s.%s...\n", config->host, config->port,
+ config->db, config->coll);
+
+ conn = mongo_sync_connect (config->host, config->port, config->slaveok);
+ if (!conn)
+ mongo_gridfs_error (errno);
+
+ if (config->master_sync)
+ {
+ VLOG ("Syncing to master...\n");
+ conn = mongo_sync_reconnect (conn, TRUE);
+ if (!conn)
+ mongo_gridfs_error (errno);
+ }
+
+ gfs = mongo_sync_gridfs_new (conn, config->ns);
+ if (!gfs)
+ mongo_gridfs_error (errno);
+
+ return gfs;
+}
+
+void
+mongo_gridfs_get (config_t *config, gint argc, gchar *argv[])
+{
+ mongo_sync_gridfs *gfs;
+ mongo_sync_gridfs_chunked_file *gfile;
+ mongo_sync_cursor *cursor;
+ gint64 n = 0;
+ bson *query;
+ int fd;
+
+ gchar *gfn, *ofn;
+
+ if (argc < 4)
+ {
+ fprintf (stderr, "Usage: %s get GRIDFS_FILENAME OUTPUT_FILENAME\n",
+ argv[0]);
+ exit (1);
+ }
+ gfn = argv[2];
+ ofn = argv[3];
+
+ gfs = mongo_gridfs_connect (config);
+
+ VLOG ("Trying to find '%s'...\n", gfn);
+
+ query = bson_build (BSON_TYPE_STRING, "filename", gfn, -1,
+ BSON_TYPE_NONE);
+ bson_finish (query);
+ gfile = mongo_sync_gridfs_chunked_find (gfs, query);
+ if (!gfile)
+ mongo_gridfs_error (errno);
+ bson_free (query);
+
+ VLOG ("Opening output file '%s'...\n", ofn);
+ fd = open (ofn, O_RDWR | O_CREAT | O_TRUNC, 0600);
+ if (fd == -1)
+ {
+ fprintf (stderr, "Error opening output file '%s': %s\n",
+ ofn, strerror (errno));
+ exit (1);
+ }
+
+ VLOG ("Writing '%s' -> '%s' (%" G_GINT64_FORMAT " bytes in %" G_GINT64_FORMAT
+ " chunks)\n", gfn, ofn,
+ mongo_sync_gridfs_file_get_length (gfile),
+ mongo_sync_gridfs_file_get_chunks (gfile));
+
+ cursor = mongo_sync_gridfs_chunked_file_cursor_new (gfile, 0, 0);
+ if (!cursor)
+ mongo_gridfs_error (errno);
+
+ while (mongo_sync_cursor_next (cursor))
+ {
+ gint32 size;
+ guint8 *data;
+
+ VLOG ("\rWriting chunk %" G_GINT64_FORMAT "...", n++);
+
+ data = mongo_sync_gridfs_chunked_file_cursor_get_chunk (cursor, &size);
+ if (!data)
+ mongo_gridfs_error (errno);
+
+ if (write (fd, data, size) != size)
+ {
+ perror ("write()");
+ exit (1);
+ }
+ g_free (data);
+ }
+ mongo_sync_cursor_free (cursor);
+ mongo_sync_gridfs_chunked_file_free (gfile);
+
+ close (fd);
+
+ mongo_sync_gridfs_free (gfs, TRUE);
+
+ VLOG("\n");
+}
+
+void
+mongo_gridfs_put (config_t *config, gint argc, gchar *argv[])
+{
+ mongo_sync_gridfs *gfs;
+ mongo_sync_gridfs_chunked_file *gfile;
+ bson *meta;
+ int fd;
+ guint8 *data;
+ struct stat st;
+
+ gchar *gfn, *ifn, *oid_s;
+
+ if (argc < 4)
+ {
+ fprintf (stderr, "Usage: %s put INPUT_FILENAME GRIDFS_FILENAME\n",
+ argv[0]);
+ exit (1);
+ }
+ ifn = argv[2];
+ gfn = argv[3];
+
+ mongo_util_oid_init (0);
+
+ gfs = mongo_gridfs_connect (config);
+
+ VLOG ("Opening input file: '%s'...\n", ifn);
+ fd = open (ifn, O_RDONLY);
+ if (!fd)
+ {
+ fprintf (stderr, "Error opening input file: %s\n",
+ strerror (errno));
+ exit (1);
+ }
+ if (fstat (fd, &st) != 0)
+ {
+ fprintf (stderr, "Error stat'ing the input file: %s\n",
+ strerror (errno));
+ exit (1);
+ }
+
+ data = mmap (NULL, st.st_size, PROT_READ, MAP_PRIVATE, fd, 0);
+ if (data == MAP_FAILED)
+ {
+ fprintf (stderr, "Error mmapping the input file: %s\n",
+ strerror (errno));
+ }
+
+ meta = bson_build (BSON_TYPE_STRING, "filename", gfn, -1,
+ BSON_TYPE_NONE);
+ bson_finish (meta);
+
+ VLOG ("Uploading '%s' -> '%s'...\n", ifn, gfn);
+
+ gfile = mongo_sync_gridfs_chunked_file_new_from_buffer (gfs, meta,
+ data, st.st_size);
+ if (!gfile)
+ mongo_gridfs_error (errno);
+ bson_free (meta);
+ munmap (data, st.st_size);
+
+ oid_s = mongo_util_oid_as_string (mongo_sync_gridfs_file_get_id (gfile));
+ printf ("Uploaded file: %s (_id: %s; md5 = %s)\n", gfn,
+ oid_s,
+ mongo_sync_gridfs_file_get_md5 (gfile));
+
+ g_free (oid_s);
+ mongo_sync_gridfs_chunked_file_free (gfile);
+ mongo_sync_gridfs_free (gfs, TRUE);
+}
+
+void
+mongo_gridfs_list (config_t *config)
+{
+ mongo_sync_cursor *cursor;
+ mongo_sync_gridfs *gfs;
+
+ gfs = mongo_gridfs_connect (config);
+
+ cursor = mongo_sync_gridfs_list (gfs, NULL);
+
+ while (mongo_sync_cursor_next (cursor))
+ {
+ bson *meta = mongo_sync_cursor_get_data (cursor);
+ bson_cursor *c;
+ const guint8 oid[12];
+ gint32 i32, chunk_size;
+ gint64 length, date;
+ const gchar *md5, *filename = NULL;
+ gchar *oid_s;
+
+ c = bson_find (meta, "_id");
+ if (!bson_cursor_get_oid (c, (const guint8 **)&oid))
+ mongo_gridfs_error (errno);
+
+ bson_cursor_find (c, "length");
+ if (!bson_cursor_get_int32 (c, &i32))
+ {
+ if (!bson_cursor_get_int64 (c, &length))
+ mongo_gridfs_error (errno);
+ }
+ else
+ length = i32;
+
+ bson_cursor_find (c, "chunkSize");
+ if (!bson_cursor_get_int32 (c, &chunk_size))
+ mongo_gridfs_error (errno);
+
+ bson_cursor_find (c, "uploadDate");
+ if (!bson_cursor_get_utc_datetime (c, &date))
+ mongo_gridfs_error (errno);
+
+ bson_cursor_find (c, "md5");
+ if (!bson_cursor_get_string (c, &md5))
+ mongo_gridfs_error (errno);
+
+ bson_cursor_find (c, "filename");
+ bson_cursor_get_string (c, &filename);
+
+ bson_cursor_free (c);
+
+ oid_s = mongo_util_oid_as_string (oid);
+ printf ("{ _id: ObjectID(\"%s\"), length: %" G_GINT64_FORMAT
+ ", chunkSize: %i, uploadDate: %"
+ G_GINT64_FORMAT ", md5: \"%s\"",
+
+ oid_s, length, chunk_size, date, md5);
+ g_free (oid_s);
+
+ if (filename)
+ printf (", filename: \"%s\"", filename);
+ printf (" }\n");
+
+ if (config->verbose)
+ {
+ c = bson_cursor_new (meta);
+ printf ("\tExtra metadata: [ ");
+ while (bson_cursor_next (c))
+ {
+ if (strcmp (bson_cursor_key (c), "_id") &&
+ strcmp (bson_cursor_key (c), "length") &&
+ strcmp (bson_cursor_key (c), "chunkSize") &&
+ strcmp (bson_cursor_key (c), "uploadDate") &&
+ strcmp (bson_cursor_key (c), "md5") &&
+ strcmp (bson_cursor_key (c), "filename"))
+ {
+ printf ("%s (%s), ", bson_cursor_key (c),
+ bson_cursor_type_as_string (c));
+ }
+ }
+ bson_cursor_free (c);
+ printf ("]\n");
+ }
+ }
+
+ mongo_sync_gridfs_free (gfs, TRUE);
+}
+
+void
+mongo_gridfs_remove (config_t *config, gint argc, gchar *argv[])
+{
+ mongo_sync_gridfs *gfs;
+ bson *query;
+ gchar *fn;
+
+ if (argc < 3)
+ {
+ fprintf (stderr, "Usage: %s remove GRIDFS_FILENAME\n", argv[0]);
+ exit (1);
+ }
+ fn = argv[2];
+
+ gfs = mongo_gridfs_connect (config);
+
+ VLOG ("Deleting file: '%s'...\n", fn);
+
+ query = bson_build (BSON_TYPE_STRING, "filename", fn, -1,
+ BSON_TYPE_NONE);
+ bson_finish (query);
+
+ if (mongo_sync_gridfs_remove (gfs, query))
+ {
+ VLOG ("\tDeleted\n");
+ }
+ else
+ {
+ VLOG ("\tFailed: %s\n", strerror (errno));
+ }
+ bson_free (query);
+
+ mongo_sync_gridfs_free (gfs, TRUE);
+}
+
+int
+main (int argc, char *argv[])
+{
+ GError *error = NULL;
+ GOptionContext *context;
+ config_t config = {
+ NULL, 27017, NULL, NULL, NULL, FALSE, FALSE, FALSE
+ };
+
+ GOptionEntry entries[] =
+ {
+ { "host", 'h', 0, G_OPTION_ARG_STRING, &config.host,
+ "Host to connect to", "HOST" },
+ { "port", 'p', 0, G_OPTION_ARG_INT, &config.port, "Port", "PORT" },
+ { "db", 'd', 0, G_OPTION_ARG_STRING, &config.db, "Database", "DB" },
+ { "collection", 'c', 0, G_OPTION_ARG_STRING, &config.coll, "Collection",
+ "COLL" },
+ { "verbose", 'v', 0, G_OPTION_ARG_NONE, &config.verbose,
+ "Be verbose", NULL },
+ { "slave-ok", 's', 0, G_OPTION_ARG_NONE, &config.slaveok,
+ "Connecting to slaves is ok", NULL },
+ { "master-sync", 'm', 0, G_OPTION_ARG_NONE, &config.master_sync,
+ "Reconnect to the replica master", NULL },
+ { NULL, 0, 0, 0, NULL, NULL, NULL }
+ };
+
+ context = g_option_context_new ("- GridFS utility");
+ g_option_context_add_main_entries (context, entries, "mongo-dump");
+ if (!g_option_context_parse (context, &argc, &argv, &error))
+ {
+ g_print ("option parsing failed: %s\n", error->message);
+ exit (1);
+ }
+
+ if (!config.host || !config.port || !config.db ||
+ !config.coll || argc < 2)
+ {
+ gchar **nargv;
+ argc = 2;
+
+ nargv = g_new (gchar *, 3);
+ nargv[0] = argv[0];
+ nargv[1] = "--help";
+ nargv[2] = NULL;
+
+ g_option_context_parse (context, &argc, (gchar ***)&nargv, &error);
+
+ exit (1);
+ }
+ config.ns = g_strdup_printf ("%s.%s", config.db, config.coll);
+
+ if (g_ascii_strcasecmp (argv[1], "get") == 0)
+ mongo_gridfs_get (&config, argc, argv);
+ else if (g_ascii_strcasecmp (argv[1], "put") == 0)
+ mongo_gridfs_put (&config, argc, argv);
+ else if (g_ascii_strcasecmp (argv[1], "list") == 0)
+ mongo_gridfs_list (&config);
+ else if (g_ascii_strcasecmp (argv[1], "remove") == 0)
+ mongo_gridfs_remove (&config, argc, argv);
+
+ g_free (config.ns);
+ g_option_context_free (context);
+
+ return 0;
+}