~drizzle-trunk/drizzle/development

« back to all changes in this revision

Viewing changes to plugin/catalog/engine.cc

Merge catalog with current trunk.

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
/* - mode: c; c-basic-offset: 2; indent-tabs-mode: nil; -*-
 
2
 *  vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
 
3
 *
 
4
 *  Copyright (C) 2010 Brian Aker
 
5
 *
 
6
 *  This program is free software; you can redistribute it and/or modify
 
7
 *  it under the terms of the GNU General Public License as published by
 
8
 *  the Free Software Foundation; either version 2 of the License, or
 
9
 *  (at your option) any later version.
 
10
 *
 
11
 *  This program is distributed in the hope that it will be useful,
 
12
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 
13
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
14
 *  GNU General Public License for more details.
 
15
 *
 
16
 *  You should have received a copy of the GNU General Public License
 
17
 *  along with this program; if not, write to the Free Software
 
18
 *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 
19
 */
 
20
 
 
21
#include "config.h"
 
22
 
 
23
#include <fcntl.h>
 
24
#include <sys/stat.h>
 
25
#include <sys/types.h>
 
26
 
 
27
#include "drizzled/display.h"
 
28
#include <google/protobuf/io/zero_copy_stream.h>
 
29
#include <google/protobuf/io/zero_copy_stream_impl.h>
 
30
 
 
31
#include <iostream>
 
32
#include <fstream>
 
33
#include <string>
 
34
 
 
35
#include "drizzled/data_home.h"
 
36
#include "drizzled/cached_directory.h"
 
37
#include "plugin/catalog/module.h"
 
38
 
 
39
namespace plugin {
 
40
namespace catalog {
 
41
 
 
42
static std::string CATALOG_OPT_EXT(".cat");
 
43
 
 
44
bool Engine::create(const drizzled::identifier::Catalog &identifier, drizzled::message::catalog::shared_ptr &message)
 
45
{
 
46
  if (mkdir(identifier.getPath().c_str(), 0777) == -1)
 
47
    return false;
 
48
 
 
49
  if (not writeFile(identifier, message))
 
50
  {
 
51
    rmdir(identifier.getPath().c_str());
 
52
 
 
53
    return false;
 
54
  }
 
55
  return false;
 
56
}
 
57
 
 
58
bool Engine::drop(const drizzled::identifier::Catalog &identifier)
 
59
{
 
60
  std::string file(identifier.getPath());
 
61
  file.append(1, FN_LIBCHAR);
 
62
  file.append(CATALOG_OPT_EXT);
 
63
 
 
64
  // No catalog file, no love from us.
 
65
  if (access(file.c_str(), F_OK))
 
66
  {
 
67
    perror(file.c_str());
 
68
    return false;
 
69
  }
 
70
 
 
71
  if (unlink(file.c_str()))
 
72
  {
 
73
    perror(file.c_str());
 
74
    return false;
 
75
  }
 
76
 
 
77
  if (rmdir(identifier.getPath().c_str()))
 
78
  {
 
79
    perror(identifier.getPath().c_str());
 
80
    //@todo If this happens, we want a report of it. For the moment I dump
 
81
    //to stderr so I can catch it in Hudson.
 
82
    drizzled::CachedDirectory dir(identifier.getPath());
 
83
  }
 
84
 
 
85
  return true;
 
86
}
 
87
 
 
88
void Engine::getMessages(drizzled::message::catalog::vector &messages)
 
89
{
 
90
  prime(messages);
 
91
}
 
92
 
 
93
void Engine::prime(drizzled::message::catalog::vector &messages)
 
94
{
 
95
  bool found_local= false;
 
96
  static drizzled::identifier::Catalog LOCAL_IDENTIFIER("local");
 
97
  drizzled::CachedDirectory directory(drizzled::getFullDataHome().file_string(), drizzled::CachedDirectory::DIRECTORY, true);
 
98
  drizzled::CachedDirectory::Entries files= directory.getEntries();
 
99
 
 
100
 
 
101
  for (drizzled::CachedDirectory::Entries::iterator fileIter= files.begin();
 
102
       fileIter != files.end(); fileIter++)
 
103
  {
 
104
    drizzled::CachedDirectory::Entry *entry= *fileIter;
 
105
    drizzled::message::catalog::shared_ptr message;
 
106
 
 
107
    if (not entry->filename.compare(GLOBAL_TEMPORARY_EXT))
 
108
      continue;
 
109
 
 
110
    drizzled::identifier::Catalog identifier(entry->filename);
 
111
 
 
112
    if (readFile(identifier, message))
 
113
    {
 
114
      messages.push_back(message);
 
115
 
 
116
      if (LOCAL_IDENTIFIER == identifier)
 
117
        found_local= true;
 
118
    }
 
119
  }
 
120
 
 
121
  if (not found_local)
 
122
  {
 
123
    messages.push_back(drizzled::message::catalog::make_shared(LOCAL_IDENTIFIER));
 
124
  }
 
125
}
 
126
 
 
127
bool Engine::writeFile(const drizzled::identifier::Catalog &identifier, drizzled::message::catalog::shared_ptr &message)
 
128
{
 
129
  char file_tmp[FN_REFLEN];
 
130
  std::string file(identifier.getPath());
 
131
 
 
132
 
 
133
  file.append(1, FN_LIBCHAR);
 
134
  file.append(CATALOG_OPT_EXT);
 
135
 
 
136
  snprintf(file_tmp, FN_REFLEN, "%sXXXXXX", file.c_str());
 
137
 
 
138
  int fd= mkstemp(file_tmp);
 
139
 
 
140
  if (fd == -1)
 
141
  {
 
142
    perror(file_tmp);
 
143
 
 
144
    return false;
 
145
  }
 
146
 
 
147
  bool success;
 
148
 
 
149
  try {
 
150
    success= message->SerializeToFileDescriptor(fd);
 
151
  }
 
152
  catch (...)
 
153
  {
 
154
    success= false;
 
155
  }
 
156
 
 
157
  if (not success)
 
158
  {
 
159
    drizzled::my_error(drizzled::ER_CORRUPT_CATALOG_DEFINITION, MYF(0), file.c_str(),
 
160
                       message->InitializationErrorString().empty() ? "unknown" :  message->InitializationErrorString().c_str());
 
161
 
 
162
    if (close(fd) == -1)
 
163
      perror(file_tmp);
 
164
 
 
165
    if (unlink(file_tmp))
 
166
      perror(file_tmp);
 
167
 
 
168
    return false;
 
169
  }
 
170
 
 
171
  if (close(fd) == -1)
 
172
  {
 
173
    perror(file_tmp);
 
174
 
 
175
    if (unlink(file_tmp))
 
176
      perror(file_tmp);
 
177
 
 
178
    return false;
 
179
  }
 
180
 
 
181
  if (rename(file_tmp, file.c_str()) == -1)
 
182
  {
 
183
    if (unlink(file_tmp))
 
184
      perror(file_tmp);
 
185
 
 
186
    return false;
 
187
  }
 
188
 
 
189
  return true;
 
190
}
 
191
 
 
192
 
 
193
bool Engine::readFile(const drizzled::identifier::Catalog &identifier, drizzled::message::catalog::shared_ptr &message)
 
194
{
 
195
  std::string path(identifier.getPath());
 
196
 
 
197
  /*
 
198
    Pass an empty file name, and the database options file name as extension
 
199
    to avoid table name to file name encoding.
 
200
  */
 
201
  path.append(1, FN_LIBCHAR);
 
202
  path.append(CATALOG_OPT_EXT);
 
203
 
 
204
  std::fstream input(path.c_str(), std::ios::in | std::ios::binary);
 
205
 
 
206
  if (input.good())
 
207
  {
 
208
    message= drizzled::message::catalog::make_shared(identifier);
 
209
    if (message->ParseFromIstream(&input))
 
210
    {
 
211
      return true;
 
212
    }
 
213
 
 
214
    drizzled::my_error(drizzled::ER_CORRUPT_CATALOG_DEFINITION, MYF(0), path.c_str(),
 
215
                       message->InitializationErrorString().empty() ? "unknown" :  message->InitializationErrorString().c_str());
 
216
  }
 
217
  else
 
218
  {
 
219
    perror(path.c_str());
 
220
  }
 
221
 
 
222
  return false;
 
223
}
 
224
 
 
225
} /* namespace catalog */
 
226
} /* namespace plugin */