| 
									
										
										
										
											2011-02-08 15:36:21 -08:00
										 |  |  |  | /* Copyright (c) 2009, 2010, 2011 Nicira Networks
 | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * 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 <config.h>
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #include "file.h"
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #include <assert.h>
 | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  | #include <errno.h>
 | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | #include <fcntl.h>
 | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  | #include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-11 17:14:31 -08:00
										 |  |  |  | #include "bitmap.h"
 | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | #include "column.h"
 | 
					
						
							|  |  |  |  | #include "log.h"
 | 
					
						
							|  |  |  |  | #include "json.h"
 | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  | #include "lockfile.h"
 | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | #include "ovsdb.h"
 | 
					
						
							|  |  |  |  | #include "ovsdb-error.h"
 | 
					
						
							|  |  |  |  | #include "row.h"
 | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  | #include "socket-util.h"
 | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | #include "table.h"
 | 
					
						
							| 
									
										
										
										
											2009-12-16 13:30:53 -08:00
										 |  |  |  | #include "timeval.h"
 | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | #include "transaction.h"
 | 
					
						
							|  |  |  |  | #include "uuid.h"
 | 
					
						
							|  |  |  |  | #include "util.h"
 | 
					
						
							|  |  |  |  | #include "vlog.h"
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-19 14:47:01 -07:00
										 |  |  |  | VLOG_DEFINE_THIS_MODULE(ovsdb_file); | 
					
						
							| 
									
										
										
										
											2010-07-16 11:02:49 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  | /* Minimum number of milliseconds between database compactions. */ | 
					
						
							|  |  |  |  | #define COMPACT_MIN_MSEC        (10 * 60 * 1000) /* 10 minutes. */
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Minimum number of milliseconds between trying to compact the database if
 | 
					
						
							|  |  |  |  |  * compacting fails. */ | 
					
						
							|  |  |  |  | #define COMPACT_RETRY_MSEC      (60 * 1000)      /* 1 minute. */
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  | /* A transaction being converted to JSON for writing to a file. */ | 
					
						
							|  |  |  |  | struct ovsdb_file_txn { | 
					
						
							|  |  |  |  |     struct json *json;          /* JSON for the whole transaction. */ | 
					
						
							|  |  |  |  |     struct json *table_json;    /* JSON for 'table''s transaction. */ | 
					
						
							|  |  |  |  |     struct ovsdb_table *table;  /* Table described in 'table_json'.  */ | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void ovsdb_file_txn_init(struct ovsdb_file_txn *); | 
					
						
							|  |  |  |  | static void ovsdb_file_txn_add_row(struct ovsdb_file_txn *, | 
					
						
							|  |  |  |  |                                    const struct ovsdb_row *old, | 
					
						
							| 
									
										
										
										
											2010-03-11 17:14:31 -08:00
										 |  |  |  |                                    const struct ovsdb_row *new, | 
					
						
							|  |  |  |  |                                    const unsigned long int *changed); | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  | static struct ovsdb_error *ovsdb_file_txn_commit(struct json *, | 
					
						
							|  |  |  |  |                                                  const char *comment, | 
					
						
							|  |  |  |  |                                                  bool durable, | 
					
						
							|  |  |  |  |                                                  struct ovsdb_log *); | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct ovsdb_error *ovsdb_file_open__(const char *file_name, | 
					
						
							|  |  |  |  |                                              const struct ovsdb_schema *, | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |                                              bool read_only, struct ovsdb **, | 
					
						
							|  |  |  |  |                                              struct ovsdb_file **); | 
					
						
							|  |  |  |  | static struct ovsdb_error *ovsdb_file_txn_from_json( | 
					
						
							|  |  |  |  |     struct ovsdb *, const struct json *, bool converting, | 
					
						
							|  |  |  |  |     long long int *date, struct ovsdb_txn **); | 
					
						
							|  |  |  |  | static struct ovsdb_error *ovsdb_file_create(struct ovsdb *, | 
					
						
							|  |  |  |  |                                              struct ovsdb_log *, | 
					
						
							|  |  |  |  |                                              const char *file_name, | 
					
						
							|  |  |  |  |                                              long long int oldest_commit, | 
					
						
							|  |  |  |  |                                              unsigned int n_transactions, | 
					
						
							|  |  |  |  |                                              struct ovsdb_file **filep); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  | /* Opens database 'file_name' and stores a pointer to the new database in
 | 
					
						
							|  |  |  |  |  * '*dbp'.  If 'read_only' is false, then the database will be locked and | 
					
						
							|  |  |  |  |  * changes to the database will be written to disk.  If 'read_only' is true, | 
					
						
							|  |  |  |  |  * the database will not be locked and changes to the database will persist | 
					
						
							|  |  |  |  |  * only as long as the "struct ovsdb". | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |  * If 'filep' is nonnull and 'read_only' is false, then on success sets | 
					
						
							|  |  |  |  |  * '*filep' to an ovsdb_file that represents the open file.  This ovsdb_file | 
					
						
							|  |  |  |  |  * persists until '*dbp' is destroyed. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  |  * On success, returns NULL.  On failure, returns an ovsdb_error (which the | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |  * caller must destroy) and sets '*dbp' and '*filep' to NULL. */ | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | struct ovsdb_error * | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  | ovsdb_file_open(const char *file_name, bool read_only, | 
					
						
							|  |  |  |  |                 struct ovsdb **dbp, struct ovsdb_file **filep) | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |     return ovsdb_file_open__(file_name, NULL, read_only, dbp, filep); | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Opens database 'file_name' with an alternate schema.  The specified 'schema'
 | 
					
						
							|  |  |  |  |  * is used to interpret the data in 'file_name', ignoring the schema actually | 
					
						
							|  |  |  |  |  * stored in the file.  Data in the file for tables or columns that do not | 
					
						
							|  |  |  |  |  * exist in 'schema' are ignored, but the ovsdb file format must otherwise be | 
					
						
							|  |  |  |  |  * observed, including column constraints. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * This function can be useful for upgrading or downgrading databases to | 
					
						
							|  |  |  |  |  * "almost-compatible" formats. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * The database will not be locked.  Changes to the database will persist only | 
					
						
							|  |  |  |  |  * as long as the "struct ovsdb". | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * On success, stores a pointer to the new database in '*dbp' and returns a | 
					
						
							|  |  |  |  |  * null pointer.  On failure, returns an ovsdb_error (which the caller must | 
					
						
							|  |  |  |  |  * destroy) and sets '*dbp' to NULL. */ | 
					
						
							|  |  |  |  | struct ovsdb_error * | 
					
						
							|  |  |  |  | ovsdb_file_open_as_schema(const char *file_name, | 
					
						
							|  |  |  |  |                           const struct ovsdb_schema *schema, | 
					
						
							|  |  |  |  |                           struct ovsdb **dbp) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |     return ovsdb_file_open__(file_name, schema, true, dbp, NULL); | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct ovsdb_error * | 
					
						
							| 
									
										
										
										
											2011-02-08 15:36:21 -08:00
										 |  |  |  | ovsdb_file_open_log(const char *file_name, enum ovsdb_log_open_mode open_mode, | 
					
						
							|  |  |  |  |                     struct ovsdb_log **logp, struct ovsdb_schema **schemap) | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |     struct ovsdb_schema *schema = NULL; | 
					
						
							|  |  |  |  |     struct ovsdb_log *log = NULL; | 
					
						
							| 
									
										
										
										
											2011-02-08 15:36:21 -08:00
										 |  |  |  |     struct ovsdb_error *error; | 
					
						
							|  |  |  |  |     struct json *json = NULL; | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-08 15:36:21 -08:00
										 |  |  |  |     assert(logp || schemap); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 11:31:32 -08:00
										 |  |  |  |     error = ovsdb_log_open(file_name, open_mode, -1, &log); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |     if (error) { | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     error = ovsdb_log_read(log, &json); | 
					
						
							|  |  |  |  |     if (error) { | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |     } else if (!json) { | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |         error = ovsdb_io_error(EOF, "%s: database file contains no schema", | 
					
						
							|  |  |  |  |                                file_name); | 
					
						
							|  |  |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-08 15:36:21 -08:00
										 |  |  |  |     if (schemap) { | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  |         error = ovsdb_schema_from_json(json, &schema); | 
					
						
							|  |  |  |  |         if (error) { | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |             error = ovsdb_wrap_error(error, | 
					
						
							|  |  |  |  |                                      "failed to parse \"%s\" as ovsdb schema", | 
					
						
							|  |  |  |  |                                      file_name); | 
					
						
							|  |  |  |  |             goto error; | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     json_destroy(json); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-08 15:36:21 -08:00
										 |  |  |  |     if (logp) { | 
					
						
							|  |  |  |  |         *logp = log; | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         ovsdb_log_close(log); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     if (schemap) { | 
					
						
							|  |  |  |  |         *schemap = schema; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | error: | 
					
						
							|  |  |  |  |     ovsdb_log_close(log); | 
					
						
							|  |  |  |  |     json_destroy(json); | 
					
						
							|  |  |  |  |     if (logp) { | 
					
						
							|  |  |  |  |         *logp = NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     if (schemap) { | 
					
						
							|  |  |  |  |         *schemap = NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return error; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct ovsdb_error * | 
					
						
							|  |  |  |  | ovsdb_file_open__(const char *file_name, | 
					
						
							|  |  |  |  |                   const struct ovsdb_schema *alternate_schema, | 
					
						
							|  |  |  |  |                   bool read_only, struct ovsdb **dbp, | 
					
						
							|  |  |  |  |                   struct ovsdb_file **filep) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     enum ovsdb_log_open_mode open_mode; | 
					
						
							|  |  |  |  |     long long int oldest_commit; | 
					
						
							|  |  |  |  |     unsigned int n_transactions; | 
					
						
							|  |  |  |  |     struct ovsdb_schema *schema = NULL; | 
					
						
							|  |  |  |  |     struct ovsdb_error *error; | 
					
						
							|  |  |  |  |     struct ovsdb_log *log; | 
					
						
							|  |  |  |  |     struct json *json; | 
					
						
							|  |  |  |  |     struct ovsdb *db = NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* In read-only mode there is no ovsdb_file so 'filep' must be null. */ | 
					
						
							|  |  |  |  |     assert(!(read_only && filep)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     open_mode = read_only ? OVSDB_LOG_READ_ONLY : OVSDB_LOG_READ_WRITE; | 
					
						
							|  |  |  |  |     error = ovsdb_file_open_log(file_name, open_mode, &log, | 
					
						
							|  |  |  |  |                                 alternate_schema ? NULL : &schema); | 
					
						
							|  |  |  |  |     if (error) { | 
					
						
							|  |  |  |  |         goto error; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     db = ovsdb_create(schema ? schema : ovsdb_schema_clone(alternate_schema)); | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     oldest_commit = LLONG_MAX; | 
					
						
							|  |  |  |  |     n_transactions = 0; | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |     while ((error = ovsdb_log_read(log, &json)) == NULL && json) { | 
					
						
							|  |  |  |  |         struct ovsdb_txn *txn; | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |         long long int date; | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  |         error = ovsdb_file_txn_from_json(db, json, alternate_schema != NULL, | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |                                          &date, &txn); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |         json_destroy(json); | 
					
						
							|  |  |  |  |         if (error) { | 
					
						
							| 
									
										
										
										
											2011-03-28 13:05:40 -07:00
										 |  |  |  |             ovsdb_log_unread(log); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |             break; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |         n_transactions++; | 
					
						
							|  |  |  |  |         if (date < oldest_commit) { | 
					
						
							|  |  |  |  |             oldest_commit = date; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-28 13:05:40 -07:00
										 |  |  |  |         error = ovsdb_txn_commit(txn, false); | 
					
						
							|  |  |  |  |         if (error) { | 
					
						
							|  |  |  |  |             ovsdb_log_unread(log); | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     if (error) { | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |         /* Log error but otherwise ignore it.  Probably the database just got
 | 
					
						
							|  |  |  |  |          * truncated due to power failure etc. and we should use its current | 
					
						
							|  |  |  |  |          * contents. */ | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |         char *msg = ovsdb_error_to_string(error); | 
					
						
							| 
									
										
										
										
											2011-03-28 12:59:18 -07:00
										 |  |  |  |         VLOG_ERR("%s", msg); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |         free(msg); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         ovsdb_error_destroy(error); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!read_only) { | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |         struct ovsdb_file *file; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         error = ovsdb_file_create(db, log, file_name, oldest_commit, | 
					
						
							|  |  |  |  |                                   n_transactions, &file); | 
					
						
							|  |  |  |  |         if (error) { | 
					
						
							|  |  |  |  |             goto error; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         if (filep) { | 
					
						
							|  |  |  |  |             *filep = file; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |     } else { | 
					
						
							|  |  |  |  |         ovsdb_log_close(log); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     *dbp = db; | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | error: | 
					
						
							|  |  |  |  |     *dbp = NULL; | 
					
						
							|  |  |  |  |     if (filep) { | 
					
						
							|  |  |  |  |         *filep = NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     ovsdb_destroy(db); | 
					
						
							|  |  |  |  |     ovsdb_log_close(log); | 
					
						
							|  |  |  |  |     return error; | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  | static struct ovsdb_error * | 
					
						
							|  |  |  |  | ovsdb_file_update_row_from_json(struct ovsdb_row *row, bool converting, | 
					
						
							|  |  |  |  |                                 const struct json *json) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct ovsdb_table_schema *schema = row->table->schema; | 
					
						
							|  |  |  |  |     struct ovsdb_error *error; | 
					
						
							|  |  |  |  |     struct shash_node *node; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (json->type != JSON_OBJECT) { | 
					
						
							|  |  |  |  |         return ovsdb_syntax_error(json, NULL, "row must be JSON object"); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     SHASH_FOR_EACH (node, json_object(json)) { | 
					
						
							|  |  |  |  |         const char *column_name = node->name; | 
					
						
							|  |  |  |  |         const struct ovsdb_column *column; | 
					
						
							|  |  |  |  |         struct ovsdb_datum datum; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         column = ovsdb_table_schema_get_column(schema, column_name); | 
					
						
							|  |  |  |  |         if (!column) { | 
					
						
							|  |  |  |  |             if (converting) { | 
					
						
							|  |  |  |  |                 continue; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             return ovsdb_syntax_error(json, "unknown column", | 
					
						
							|  |  |  |  |                                       "No column %s in table %s.", | 
					
						
							|  |  |  |  |                                       column_name, schema->name); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         error = ovsdb_datum_from_json(&datum, &column->type, node->data, NULL); | 
					
						
							|  |  |  |  |         if (error) { | 
					
						
							|  |  |  |  |             return error; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         ovsdb_datum_swap(&row->fields[column->index], &datum); | 
					
						
							|  |  |  |  |         ovsdb_datum_destroy(&datum, &column->type); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | static struct ovsdb_error * | 
					
						
							|  |  |  |  | ovsdb_file_txn_row_from_json(struct ovsdb_txn *txn, struct ovsdb_table *table, | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  |                              bool converting, | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |                              const struct uuid *row_uuid, struct json *json) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     const struct ovsdb_row *row = ovsdb_table_get_row(table, row_uuid); | 
					
						
							|  |  |  |  |     if (json->type == JSON_NULL) { | 
					
						
							|  |  |  |  |         if (!row) { | 
					
						
							|  |  |  |  |             return ovsdb_syntax_error(NULL, NULL, "transaction deletes " | 
					
						
							|  |  |  |  |                                       "row "UUID_FMT" that does not exist", | 
					
						
							|  |  |  |  |                                       UUID_ARGS(row_uuid)); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         ovsdb_txn_row_delete(txn, row); | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  |     } else if (row) { | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  |         return ovsdb_file_update_row_from_json(ovsdb_txn_row_modify(txn, row), | 
					
						
							|  |  |  |  |                                                converting, json); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |     } else { | 
					
						
							|  |  |  |  |         struct ovsdb_error *error; | 
					
						
							|  |  |  |  |         struct ovsdb_row *new; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         new = ovsdb_row_create(table); | 
					
						
							|  |  |  |  |         *ovsdb_row_get_uuid_rw(new) = *row_uuid; | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  |         error = ovsdb_file_update_row_from_json(new, converting, json); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |         if (error) { | 
					
						
							|  |  |  |  |             ovsdb_row_destroy(new); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         ovsdb_txn_row_insert(txn, new); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         return error; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct ovsdb_error * | 
					
						
							|  |  |  |  | ovsdb_file_txn_table_from_json(struct ovsdb_txn *txn, | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  |                                struct ovsdb_table *table, | 
					
						
							|  |  |  |  |                                bool converting, struct json *json) | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct shash_node *node; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (json->type != JSON_OBJECT) { | 
					
						
							|  |  |  |  |         return ovsdb_syntax_error(json, NULL, "object expected"); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     SHASH_FOR_EACH (node, json->u.object) { | 
					
						
							|  |  |  |  |         const char *uuid_string = node->name; | 
					
						
							|  |  |  |  |         struct json *txn_row_json = node->data; | 
					
						
							|  |  |  |  |         struct ovsdb_error *error; | 
					
						
							|  |  |  |  |         struct uuid row_uuid; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (!uuid_from_string(&row_uuid, uuid_string)) { | 
					
						
							|  |  |  |  |             return ovsdb_syntax_error(json, NULL, "\"%s\" is not a valid UUID", | 
					
						
							|  |  |  |  |                                       uuid_string); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  |         error = ovsdb_file_txn_row_from_json(txn, table, converting, | 
					
						
							|  |  |  |  |                                              &row_uuid, txn_row_json); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |         if (error) { | 
					
						
							|  |  |  |  |             return error; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  | /* Converts 'json' to an ovsdb_txn for 'db', storing the new transaction in
 | 
					
						
							|  |  |  |  |  * '*txnp'.  Returns NULL if successful, otherwise an error. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If 'converting' is true, then unknown table and column names are ignored | 
					
						
							|  |  |  |  |  * (which can ease upgrading and downgrading schemas); otherwise, they are | 
					
						
							|  |  |  |  |  * treated as errors. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If successful, the date associated with the transaction, as the number of | 
					
						
							|  |  |  |  |  * milliseconds since the epoch, is stored in '*date'.  If the transaction does | 
					
						
							|  |  |  |  |  * not include a date, LLONG_MAX is stored. */ | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | static struct ovsdb_error * | 
					
						
							|  |  |  |  | ovsdb_file_txn_from_json(struct ovsdb *db, const struct json *json, | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |                          bool converting, long long int *date, | 
					
						
							|  |  |  |  |                          struct ovsdb_txn **txnp) | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct ovsdb_error *error; | 
					
						
							|  |  |  |  |     struct shash_node *node; | 
					
						
							|  |  |  |  |     struct ovsdb_txn *txn; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     *txnp = NULL; | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |     *date = LLONG_MAX; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |     if (json->type != JSON_OBJECT) { | 
					
						
							|  |  |  |  |         return ovsdb_syntax_error(json, NULL, "object expected"); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     txn = ovsdb_txn_create(db); | 
					
						
							|  |  |  |  |     SHASH_FOR_EACH (node, json->u.object) { | 
					
						
							|  |  |  |  |         const char *table_name = node->name; | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |         struct json *node_json = node->data; | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |         struct ovsdb_table *table; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         table = shash_find_data(&db->tables, table_name); | 
					
						
							|  |  |  |  |         if (!table) { | 
					
						
							| 
									
										
										
										
											2009-12-16 13:30:53 -08:00
										 |  |  |  |             if (!strcmp(table_name, "_date") | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |                 && node_json->type == JSON_INTEGER) { | 
					
						
							| 
									
										
										
										
											2011-02-21 17:02:35 -08:00
										 |  |  |  |                 *date = json_integer(node_json); | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |                 continue; | 
					
						
							|  |  |  |  |             } else if (!strcmp(table_name, "_comment") || converting) { | 
					
						
							| 
									
										
										
										
											2009-12-16 13:30:53 -08:00
										 |  |  |  |                 continue; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |             error = ovsdb_syntax_error(json, "unknown table", | 
					
						
							|  |  |  |  |                                        "No table named %s.", table_name); | 
					
						
							|  |  |  |  |             goto error; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  |         error = ovsdb_file_txn_table_from_json(txn, table, converting, | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |                                                node_json); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |         if (error) { | 
					
						
							|  |  |  |  |             goto error; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     *txnp = txn; | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | error: | 
					
						
							|  |  |  |  |     ovsdb_txn_abort(txn); | 
					
						
							|  |  |  |  |     return error; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  | static struct ovsdb_error * | 
					
						
							|  |  |  |  | ovsdb_file_save_copy__(const char *file_name, int locking, | 
					
						
							|  |  |  |  |                        const char *comment, const struct ovsdb *db, | 
					
						
							|  |  |  |  |                        struct ovsdb_log **logp) | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     const struct shash_node *node; | 
					
						
							|  |  |  |  |     struct ovsdb_file_txn ftxn; | 
					
						
							|  |  |  |  |     struct ovsdb_error *error; | 
					
						
							|  |  |  |  |     struct ovsdb_log *log; | 
					
						
							|  |  |  |  |     struct json *json; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     error = ovsdb_log_open(file_name, OVSDB_LOG_CREATE, locking, &log); | 
					
						
							|  |  |  |  |     if (error) { | 
					
						
							|  |  |  |  |         return error; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Write schema. */ | 
					
						
							|  |  |  |  |     json = ovsdb_schema_to_json(db->schema); | 
					
						
							|  |  |  |  |     error = ovsdb_log_write(log, json); | 
					
						
							|  |  |  |  |     json_destroy(json); | 
					
						
							|  |  |  |  |     if (error) { | 
					
						
							|  |  |  |  |         goto exit; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Write data. */ | 
					
						
							|  |  |  |  |     ovsdb_file_txn_init(&ftxn); | 
					
						
							|  |  |  |  |     SHASH_FOR_EACH (node, &db->tables) { | 
					
						
							|  |  |  |  |         const struct ovsdb_table *table = node->data; | 
					
						
							|  |  |  |  |         const struct ovsdb_row *row; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-17 10:33:10 -07:00
										 |  |  |  |         HMAP_FOR_EACH (row, hmap_node, &table->rows) { | 
					
						
							| 
									
										
										
										
											2010-03-11 17:14:31 -08:00
										 |  |  |  |             ovsdb_file_txn_add_row(&ftxn, NULL, row, NULL); | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     error = ovsdb_file_txn_commit(ftxn.json, comment, true, log); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | exit: | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |     if (logp) { | 
					
						
							|  |  |  |  |         if (!error) { | 
					
						
							|  |  |  |  |             *logp = log; | 
					
						
							|  |  |  |  |             log = NULL; | 
					
						
							|  |  |  |  |         } else { | 
					
						
							|  |  |  |  |             *logp = NULL; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-12 11:26:54 -08:00
										 |  |  |  |     ovsdb_log_close(log); | 
					
						
							|  |  |  |  |     if (error) { | 
					
						
							|  |  |  |  |         remove(file_name); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return error; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Saves a snapshot of 'db''s current contents as 'file_name'.  If 'comment' is
 | 
					
						
							|  |  |  |  |  * nonnull, then it is added along with the data contents and can be viewed | 
					
						
							|  |  |  |  |  * with "ovsdb-tool show-log". | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * 'locking' is passed along to ovsdb_log_open() untouched. */ | 
					
						
							|  |  |  |  | struct ovsdb_error * | 
					
						
							|  |  |  |  | ovsdb_file_save_copy(const char *file_name, int locking, | 
					
						
							|  |  |  |  |                      const char *comment, const struct ovsdb *db) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     return ovsdb_file_save_copy__(file_name, locking, comment, db, NULL); | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-02-08 15:36:21 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Opens database 'file_name', reads its schema, and closes it.  On success,
 | 
					
						
							|  |  |  |  |  * stores the schema into '*schemap' and returns NULL; the caller then owns the | 
					
						
							|  |  |  |  |  * schema.  On failure, returns an ovsdb_error (which the caller must destroy) | 
					
						
							|  |  |  |  |  * and sets '*dbp' to NULL. */ | 
					
						
							|  |  |  |  | struct ovsdb_error * | 
					
						
							|  |  |  |  | ovsdb_file_read_schema(const char *file_name, struct ovsdb_schema **schemap) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     assert(schemap != NULL); | 
					
						
							|  |  |  |  |     return ovsdb_file_open_log(file_name, OVSDB_LOG_READ_ONLY, NULL, schemap); | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |  | 
					
						
							|  |  |  |  | /* Replica implementation. */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-16 15:13:42 -07:00
										 |  |  |  | struct ovsdb_file { | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |     struct ovsdb_replica replica; | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |     struct ovsdb *db; | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |     struct ovsdb_log *log; | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |     char *file_name; | 
					
						
							|  |  |  |  |     long long int oldest_commit; | 
					
						
							|  |  |  |  |     long long int next_compact; | 
					
						
							|  |  |  |  |     unsigned int n_transactions; | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-16 15:13:42 -07:00
										 |  |  |  | static const struct ovsdb_replica_class ovsdb_file_class; | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  | static struct ovsdb_error * | 
					
						
							|  |  |  |  | ovsdb_file_create(struct ovsdb *db, struct ovsdb_log *log, | 
					
						
							|  |  |  |  |                   const char *file_name, | 
					
						
							|  |  |  |  |                   long long int oldest_commit, | 
					
						
							|  |  |  |  |                   unsigned int n_transactions, | 
					
						
							|  |  |  |  |                   struct ovsdb_file **filep) | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |     long long int now = time_msec(); | 
					
						
							|  |  |  |  |     struct ovsdb_file *file; | 
					
						
							|  |  |  |  |     char *abs_name; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Use the absolute name of the file because ovsdb-server opens its
 | 
					
						
							|  |  |  |  |      * database before daemonize() chdirs to "/". */ | 
					
						
							|  |  |  |  |     abs_name = abs_file_name(NULL, file_name); | 
					
						
							|  |  |  |  |     if (!abs_name) { | 
					
						
							|  |  |  |  |         *filep = NULL; | 
					
						
							|  |  |  |  |         return ovsdb_io_error(0, "could not determine current " | 
					
						
							|  |  |  |  |                               "working directory"); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     file = xmalloc(sizeof *file); | 
					
						
							| 
									
										
										
										
											2010-03-16 15:13:42 -07:00
										 |  |  |  |     ovsdb_replica_init(&file->replica, &ovsdb_file_class); | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |     file->db = db; | 
					
						
							| 
									
										
										
										
											2010-03-16 15:13:42 -07:00
										 |  |  |  |     file->log = log; | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |     file->file_name = abs_name; | 
					
						
							|  |  |  |  |     file->oldest_commit = MIN(oldest_commit, now); | 
					
						
							|  |  |  |  |     file->next_compact = file->oldest_commit + COMPACT_MIN_MSEC; | 
					
						
							|  |  |  |  |     file->n_transactions = n_transactions; | 
					
						
							| 
									
										
										
										
											2010-03-16 15:13:42 -07:00
										 |  |  |  |     ovsdb_add_replica(db, &file->replica); | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     *filep = file; | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-16 15:13:42 -07:00
										 |  |  |  | static struct ovsdb_file * | 
					
						
							|  |  |  |  | ovsdb_file_cast(struct ovsdb_replica *replica) | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-03-16 15:13:42 -07:00
										 |  |  |  |     assert(replica->class == &ovsdb_file_class); | 
					
						
							|  |  |  |  |     return CONTAINER_OF(replica, struct ovsdb_file, replica); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static bool | 
					
						
							| 
									
										
										
										
											2010-03-16 15:13:42 -07:00
										 |  |  |  | ovsdb_file_change_cb(const struct ovsdb_row *old, | 
					
						
							|  |  |  |  |                      const struct ovsdb_row *new, | 
					
						
							|  |  |  |  |                      const unsigned long int *changed, | 
					
						
							|  |  |  |  |                      void *ftxn_) | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct ovsdb_file_txn *ftxn = ftxn_; | 
					
						
							| 
									
										
										
										
											2010-03-11 17:14:31 -08:00
										 |  |  |  |     ovsdb_file_txn_add_row(ftxn, old, new, changed); | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  |     return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct ovsdb_error * | 
					
						
							| 
									
										
										
										
											2010-03-16 15:13:42 -07:00
										 |  |  |  | ovsdb_file_commit(struct ovsdb_replica *replica, | 
					
						
							|  |  |  |  |                   const struct ovsdb_txn *txn, bool durable) | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-03-16 15:13:42 -07:00
										 |  |  |  |     struct ovsdb_file *file = ovsdb_file_cast(replica); | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  |     struct ovsdb_file_txn ftxn; | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |     struct ovsdb_error *error; | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovsdb_file_txn_init(&ftxn); | 
					
						
							| 
									
										
										
										
											2010-03-16 15:13:42 -07:00
										 |  |  |  |     ovsdb_txn_for_each_change(txn, ovsdb_file_change_cb, &ftxn); | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  |     if (!ftxn.json) { | 
					
						
							|  |  |  |  |         /* Nothing to commit. */ | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |     error = ovsdb_file_txn_commit(ftxn.json, ovsdb_txn_get_comment(txn), | 
					
						
							|  |  |  |  |                                   durable, file->log); | 
					
						
							|  |  |  |  |     if (error) { | 
					
						
							|  |  |  |  |         return error; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     file->n_transactions++; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* If it has been at least COMPACT_MIN_MSEC millseconds since the last time
 | 
					
						
							|  |  |  |  |      * we compacted (or at least COMPACT_RETRY_MSEC since the last time we | 
					
						
							|  |  |  |  |      * tried), and if there are at least 100 transactions in the database, and | 
					
						
							|  |  |  |  |      * if the database is at least 1 MB, then compact the database. */ | 
					
						
							|  |  |  |  |     if (time_msec() >= file->next_compact | 
					
						
							|  |  |  |  |         && file->n_transactions >= 100 | 
					
						
							|  |  |  |  |         && ovsdb_log_get_offset(file->log) >= 10 * 1024 * 1024) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         error = ovsdb_file_compact(file); | 
					
						
							|  |  |  |  |         if (error) { | 
					
						
							|  |  |  |  |             char *s = ovsdb_error_to_string(error); | 
					
						
							|  |  |  |  |             ovsdb_error_destroy(error); | 
					
						
							|  |  |  |  |             VLOG_WARN("%s: compacting database failed (%s), retrying in " | 
					
						
							|  |  |  |  |                       "60 seconds", file->file_name, s); | 
					
						
							|  |  |  |  |             free(s); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             file->next_compact = time_msec() + COMPACT_RETRY_MSEC; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | struct ovsdb_error * | 
					
						
							|  |  |  |  | ovsdb_file_compact(struct ovsdb_file *file) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct ovsdb_log *new_log = NULL; | 
					
						
							|  |  |  |  |     struct lockfile *tmp_lock = NULL; | 
					
						
							|  |  |  |  |     struct ovsdb_error *error; | 
					
						
							|  |  |  |  |     char *tmp_name = NULL; | 
					
						
							|  |  |  |  |     char *comment = NULL; | 
					
						
							|  |  |  |  |     int retval; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     comment = xasprintf("compacting database online " | 
					
						
							|  |  |  |  |                         "(%.3f seconds old, %u transactions, %llu bytes)", | 
					
						
							|  |  |  |  |                         (time_msec() - file->oldest_commit) / 1000.0, | 
					
						
							|  |  |  |  |                         file->n_transactions, | 
					
						
							|  |  |  |  |                         (unsigned long long) ovsdb_log_get_offset(file->log)); | 
					
						
							|  |  |  |  |     VLOG_INFO("%s: %s", file->file_name, comment); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Commit the old version, so that we can be assured that we'll eventually
 | 
					
						
							|  |  |  |  |      * have either the old or the new version. */ | 
					
						
							|  |  |  |  |     error = ovsdb_log_commit(file->log); | 
					
						
							|  |  |  |  |     if (error) { | 
					
						
							|  |  |  |  |         goto exit; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Lock temporary file. */ | 
					
						
							|  |  |  |  |     tmp_name = xasprintf("%s.tmp", file->file_name); | 
					
						
							|  |  |  |  |     retval = lockfile_lock(tmp_name, 0, &tmp_lock); | 
					
						
							|  |  |  |  |     if (retval) { | 
					
						
							|  |  |  |  |         error = ovsdb_io_error(retval, "could not get lock on %s", tmp_name); | 
					
						
							|  |  |  |  |         goto exit; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Remove temporary file.  (It might not exist.) */ | 
					
						
							|  |  |  |  |     if (unlink(tmp_name) < 0 && errno != ENOENT) { | 
					
						
							|  |  |  |  |         error = ovsdb_io_error(errno, "failed to remove %s", tmp_name); | 
					
						
							|  |  |  |  |         goto exit; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Save a copy. */ | 
					
						
							|  |  |  |  |     error = ovsdb_file_save_copy__(tmp_name, false, comment, file->db, | 
					
						
							|  |  |  |  |                                    &new_log); | 
					
						
							|  |  |  |  |     if (error) { | 
					
						
							|  |  |  |  |         goto exit; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Replace original by temporary. */ | 
					
						
							|  |  |  |  |     if (rename(tmp_name, file->file_name)) { | 
					
						
							|  |  |  |  |         error = ovsdb_io_error(errno, "failed to rename \"%s\" to \"%s\"", | 
					
						
							|  |  |  |  |                                tmp_name, file->file_name); | 
					
						
							|  |  |  |  |         goto exit; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     fsync_parent_dir(file->file_name); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | exit: | 
					
						
							|  |  |  |  |     if (!error) { | 
					
						
							|  |  |  |  |         ovsdb_log_close(file->log); | 
					
						
							|  |  |  |  |         file->log = new_log; | 
					
						
							|  |  |  |  |         file->oldest_commit = time_msec(); | 
					
						
							|  |  |  |  |         file->next_compact = file->oldest_commit + COMPACT_MIN_MSEC; | 
					
						
							|  |  |  |  |         file->n_transactions = 1; | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         ovsdb_log_close(new_log); | 
					
						
							|  |  |  |  |         if (tmp_lock) { | 
					
						
							|  |  |  |  |             unlink(tmp_name); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     lockfile_unlock(tmp_lock); | 
					
						
							|  |  |  |  |     free(tmp_name); | 
					
						
							|  |  |  |  |     free(comment); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return error; | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2010-03-16 15:13:42 -07:00
										 |  |  |  | ovsdb_file_destroy(struct ovsdb_replica *replica) | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-03-16 15:13:42 -07:00
										 |  |  |  |     struct ovsdb_file *file = ovsdb_file_cast(replica); | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-16 15:13:42 -07:00
										 |  |  |  |     ovsdb_log_close(file->log); | 
					
						
							| 
									
										
										
										
											2010-03-18 11:24:55 -07:00
										 |  |  |  |     free(file->file_name); | 
					
						
							| 
									
										
										
										
											2010-03-16 15:13:42 -07:00
										 |  |  |  |     free(file); | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-16 15:13:42 -07:00
										 |  |  |  | static const struct ovsdb_replica_class ovsdb_file_class = { | 
					
						
							|  |  |  |  |     ovsdb_file_commit, | 
					
						
							|  |  |  |  |     ovsdb_file_destroy | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  |  | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | ovsdb_file_txn_init(struct ovsdb_file_txn *ftxn) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     ftxn->json = NULL; | 
					
						
							|  |  |  |  |     ftxn->table_json = NULL; | 
					
						
							|  |  |  |  |     ftxn->table = NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | ovsdb_file_txn_add_row(struct ovsdb_file_txn *ftxn, | 
					
						
							|  |  |  |  |                        const struct ovsdb_row *old, | 
					
						
							| 
									
										
										
										
											2010-03-11 17:14:31 -08:00
										 |  |  |  |                        const struct ovsdb_row *new, | 
					
						
							|  |  |  |  |                        const unsigned long int *changed) | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct json *row; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!new) { | 
					
						
							|  |  |  |  |         row = json_null_create(); | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         struct shash_node *node; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-12 11:13:24 -08:00
										 |  |  |  |         row = old ? NULL : json_object_create(); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |         SHASH_FOR_EACH (node, &new->table->schema->columns) { | 
					
						
							|  |  |  |  |             const struct ovsdb_column *column = node->data; | 
					
						
							|  |  |  |  |             const struct ovsdb_type *type = &column->type; | 
					
						
							|  |  |  |  |             unsigned int idx = column->index; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (idx != OVSDB_COL_UUID && column->persistent | 
					
						
							| 
									
										
										
										
											2010-01-11 13:14:54 -08:00
										 |  |  |  |                 && (old | 
					
						
							| 
									
										
										
										
											2010-03-11 17:14:31 -08:00
										 |  |  |  |                     ? bitmap_is_set(changed, idx) | 
					
						
							| 
									
										
										
										
											2010-01-11 13:14:54 -08:00
										 |  |  |  |                     : !ovsdb_datum_is_default(&new->fields[idx], type))) | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |             { | 
					
						
							|  |  |  |  |                 if (!row) { | 
					
						
							|  |  |  |  |                     row = json_object_create(); | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |                 json_object_put(row, column->name, | 
					
						
							|  |  |  |  |                                 ovsdb_datum_to_json(&new->fields[idx], type)); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (row) { | 
					
						
							|  |  |  |  |         struct ovsdb_table *table = new ? new->table : old->table; | 
					
						
							|  |  |  |  |         char uuid[UUID_LEN + 1]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  |         if (table != ftxn->table) { | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |             /* Create JSON object for transaction overall. */ | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  |             if (!ftxn->json) { | 
					
						
							|  |  |  |  |                 ftxn->json = json_object_create(); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             /* Create JSON object for transaction on this table. */ | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  |             ftxn->table_json = json_object_create(); | 
					
						
							|  |  |  |  |             ftxn->table = table; | 
					
						
							|  |  |  |  |             json_object_put(ftxn->json, table->schema->name, ftxn->table_json); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         /* Add row to transaction for this table. */ | 
					
						
							|  |  |  |  |         snprintf(uuid, sizeof uuid, | 
					
						
							|  |  |  |  |                  UUID_FMT, UUID_ARGS(ovsdb_row_get_uuid(new ? new : old))); | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  |         json_object_put(ftxn->table_json, uuid, row); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct ovsdb_error * | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  | ovsdb_file_txn_commit(struct json *json, const char *comment, | 
					
						
							|  |  |  |  |                       bool durable, struct ovsdb_log *log) | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct ovsdb_error *error; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  |     if (!json) { | 
					
						
							|  |  |  |  |         json = json_object_create(); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-12-16 13:30:53 -08:00
										 |  |  |  |     if (comment) { | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  |         json_object_put_string(json, "_comment", comment); | 
					
						
							| 
									
										
										
										
											2009-12-16 13:30:53 -08:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-06-08 17:18:48 -07:00
										 |  |  |  |     json_object_put(json, "_date", json_integer_create(time_wall())); | 
					
						
							| 
									
										
										
										
											2009-12-16 13:30:53 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  |     error = ovsdb_log_write(log, json); | 
					
						
							|  |  |  |  |     json_destroy(json); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |     if (error) { | 
					
						
							|  |  |  |  |         return ovsdb_wrap_error(error, "writing transaction failed"); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (durable) { | 
					
						
							| 
									
										
										
										
											2010-02-10 16:35:24 -08:00
										 |  |  |  |         error = ovsdb_log_commit(log); | 
					
						
							| 
									
										
										
										
											2009-11-13 13:37:55 -08:00
										 |  |  |  |         if (error) { | 
					
						
							|  |  |  |  |             return ovsdb_wrap_error(error, "committing transaction failed"); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | } |