624 lines
11 KiB
C++
624 lines
11 KiB
C++
/*
|
|
* This file is a part of morm
|
|
* and is distributed under the 2-Clause BSD licence.
|
|
* Author: Tomasz Sowa <t.sowa@ttmath.org>
|
|
*/
|
|
|
|
/*
|
|
* Copyright (c) 2018-2019, Tomasz Sowa
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright notice,
|
|
* this list of conditions and the following disclaimer.
|
|
*
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
|
|
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
*/
|
|
|
|
#include "model.h"
|
|
|
|
|
|
namespace morm
|
|
{
|
|
|
|
Model::Model()
|
|
{
|
|
model_connector = nullptr;
|
|
model_data = nullptr;
|
|
doc_field_pointer = nullptr;
|
|
save_mode = DO_INSERT_ON_SAVE;
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_NONE;
|
|
query_result = nullptr;
|
|
}
|
|
|
|
|
|
Model::Model(const Model & m)
|
|
{
|
|
model_connector = m.model_connector;
|
|
save_mode = m.save_mode;
|
|
|
|
model_data = nullptr;
|
|
doc_field_pointer = nullptr; // does it need to be copied?
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_NONE;
|
|
query_result = nullptr;
|
|
}
|
|
|
|
|
|
Model::~Model()
|
|
{
|
|
}
|
|
|
|
void Model::set_save_mode(SaveMode save_mode)
|
|
{
|
|
this->save_mode = save_mode;
|
|
}
|
|
|
|
|
|
Model::SaveMode Model::get_save_mode()
|
|
{
|
|
return save_mode;
|
|
}
|
|
|
|
|
|
void Model::mark_to_delete()
|
|
{
|
|
save_mode = DO_DELETE_ON_SAVE;
|
|
}
|
|
|
|
void Model::mark_to_remove()
|
|
{
|
|
save_mode = DO_DELETE_ON_SAVE;
|
|
}
|
|
|
|
void Model::mark_to_insert()
|
|
{
|
|
save_mode = DO_INSERT_ON_SAVE;
|
|
}
|
|
|
|
|
|
void Model::mark_to_update()
|
|
{
|
|
save_mode = DO_UPDATE_ON_SAVE;
|
|
}
|
|
|
|
|
|
|
|
void Model::table_name(PT::TextStream & stream)
|
|
{
|
|
}
|
|
|
|
|
|
// the body can be moved to dbexpression
|
|
// the algorithm can be different depending on the database used
|
|
void Model::table_name_for_join_as(PT::TextStream & stream)
|
|
{
|
|
PT::TextStream str_temp;
|
|
table_name(str_temp);
|
|
|
|
// temporarily
|
|
std::string str;
|
|
str_temp.to_string(str);
|
|
|
|
size_t i = str.size();
|
|
|
|
while( i-- > 0 )
|
|
{
|
|
if( str[i] == '.' )
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
if( str[i] == '.' )
|
|
{
|
|
i += 1;
|
|
}
|
|
|
|
for( ; i < str.size() ; ++i)
|
|
{
|
|
stream << str[i];
|
|
}
|
|
}
|
|
|
|
|
|
void Model::set_connector(ModelConnector & connector)
|
|
{
|
|
set_connector(&connector);
|
|
}
|
|
|
|
|
|
void Model::set_connector(ModelConnector * connector)
|
|
{
|
|
model_connector = connector;
|
|
}
|
|
|
|
|
|
bool Model::object_exists()
|
|
{
|
|
return save_mode == DO_UPDATE_ON_SAVE;
|
|
}
|
|
|
|
|
|
bool Model::found()
|
|
{
|
|
return save_mode == DO_UPDATE_ON_SAVE;
|
|
}
|
|
|
|
|
|
void Model::to_text(PT::TextStream & stream, ModelData * model_data, bool clear_stream)
|
|
{
|
|
if( clear_stream )
|
|
{
|
|
stream.clear();
|
|
}
|
|
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_GENERATING_FLAT_STRING;
|
|
|
|
if( model_connector )
|
|
{
|
|
FlatConnector * flat_connector = model_connector->get_flat_connector();
|
|
|
|
if( flat_connector )
|
|
{
|
|
this->model_data = model_data;
|
|
flat_connector->to_text(stream, *this);
|
|
this->model_data = nullptr;
|
|
}
|
|
}
|
|
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_NONE;
|
|
}
|
|
|
|
|
|
void Model::to_text(PT::TextStream & stream, ModelData & model_data, bool clear_stream)
|
|
{
|
|
to_text(stream, &model_data, clear_stream);
|
|
}
|
|
|
|
|
|
void Model::to_text(PT::TextStream & stream, bool clear_stream)
|
|
{
|
|
to_text(stream, nullptr, clear_stream);
|
|
}
|
|
|
|
|
|
|
|
void Model::to_text(std::string & str, ModelData * model_data, bool clear_string)
|
|
{
|
|
if( model_connector )
|
|
{
|
|
// CHECK ME what if the stream is being used by something other?
|
|
PT::TextStream * out_stream = model_connector->get_stream();
|
|
|
|
if( out_stream )
|
|
{
|
|
to_text(*out_stream, model_data, true);
|
|
out_stream->to_string(str, clear_string);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void Model::to_text(std::string & str, ModelData & model_data, bool clear_string)
|
|
{
|
|
to_text(str, &model_data, clear_string);
|
|
}
|
|
|
|
|
|
void Model::to_text(std::string & str, bool clear_string)
|
|
{
|
|
to_text(str, nullptr, clear_string);
|
|
}
|
|
|
|
|
|
std::string Model::to_text()
|
|
{
|
|
std::string str;
|
|
|
|
to_text(str, false);
|
|
return str;
|
|
}
|
|
|
|
|
|
std::string Model::to_string()
|
|
{
|
|
return to_text();
|
|
}
|
|
|
|
|
|
|
|
void Model::generate_insert_query(PT::TextStream & stream)
|
|
{
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_GENERATING_DB_SQL;
|
|
|
|
if( model_connector )
|
|
{
|
|
DbConnector * db_connector = model_connector->get_db_connector();
|
|
|
|
if( db_connector )
|
|
{
|
|
db_connector->generate_insert_query(stream, *this);
|
|
}
|
|
}
|
|
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_NONE;
|
|
}
|
|
|
|
|
|
|
|
bool Model::insert(ModelData * model_data)
|
|
{
|
|
bool result = false;
|
|
|
|
if( model_connector )
|
|
{
|
|
this->model_data = model_data;
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_GENERATING_DB_SQL;
|
|
DbConnector * db_connector = model_connector->get_db_connector();
|
|
|
|
// CHECK ME what if the stream is being used by something other?
|
|
PT::TextStream * out_stream = model_connector->get_stream();
|
|
|
|
if( db_connector && out_stream )
|
|
{
|
|
before_insert();
|
|
out_stream->clear();
|
|
result = db_connector->insert(*out_stream, *this);
|
|
|
|
if( result )
|
|
{
|
|
save_mode = DO_UPDATE_ON_SAVE; // IMPROVE ME check if there is a primary key
|
|
after_insert();
|
|
}
|
|
else
|
|
{
|
|
after_insert_failure();
|
|
}
|
|
}
|
|
|
|
this->model_data = nullptr;
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_NONE;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
|
|
bool Model::insert(ModelData & model_data)
|
|
{
|
|
return insert(&model_data);
|
|
}
|
|
|
|
|
|
void Model::generate_update_query(PT::TextStream & stream)
|
|
{
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_GENERATING_DB_SQL;
|
|
|
|
if( model_connector )
|
|
{
|
|
DbConnector * db_connector = model_connector->get_db_connector();
|
|
|
|
if( db_connector )
|
|
{
|
|
db_connector->generate_update_query(stream, *this);
|
|
}
|
|
}
|
|
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_NONE;
|
|
}
|
|
|
|
|
|
bool Model::update(ModelData * model_data)
|
|
{
|
|
bool result = false;
|
|
|
|
if( model_connector )
|
|
{
|
|
this->model_data = model_data;
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_GENERATING_DB_SQL;
|
|
DbConnector * db_connector = model_connector->get_db_connector();
|
|
|
|
// CHECK ME what if the stream is being used by something other?
|
|
PT::TextStream * out_stream = model_connector->get_stream();
|
|
|
|
if( db_connector && out_stream )
|
|
{
|
|
before_update();
|
|
out_stream->clear();
|
|
result = db_connector->update(*out_stream, *this);
|
|
|
|
if( result )
|
|
after_update();
|
|
else
|
|
after_update_failure();
|
|
}
|
|
|
|
this->model_data = nullptr;
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_NONE;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
bool Model::update(ModelData & model_data)
|
|
{
|
|
return update(&model_data);
|
|
}
|
|
|
|
|
|
void Model::generate_remove_query(PT::TextStream & stream)
|
|
{
|
|
if( model_connector )
|
|
{
|
|
DbConnector * db_connector = model_connector->get_db_connector();
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_GENERATING_DB_SQL;
|
|
|
|
if( db_connector )
|
|
{
|
|
db_connector->generate_remove_query(stream, *this);
|
|
}
|
|
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_NONE;
|
|
}
|
|
}
|
|
|
|
|
|
bool Model::remove(ModelData * model_data)
|
|
{
|
|
bool result = false;
|
|
|
|
if( model_connector )
|
|
{
|
|
this->model_data = model_data;
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_GENERATING_DB_SQL;
|
|
DbConnector * db_connector = model_connector->get_db_connector();
|
|
|
|
// CHECK ME what if the stream is being used by something other?
|
|
PT::TextStream * out_stream = model_connector->get_stream();
|
|
|
|
if( db_connector && out_stream )
|
|
{
|
|
before_remove();
|
|
out_stream->clear();
|
|
result = db_connector->remove(*out_stream, *this);
|
|
|
|
if( result )
|
|
{
|
|
save_mode = DO_INSERT_ON_SAVE;
|
|
after_remove();
|
|
}
|
|
else
|
|
{
|
|
after_remove_failure();
|
|
}
|
|
}
|
|
|
|
this->model_data = nullptr;
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_NONE;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
bool Model::remove(ModelData & model_data)
|
|
{
|
|
return remove(&model_data);
|
|
}
|
|
|
|
|
|
bool Model::save(ModelData * model_data)
|
|
{
|
|
bool result = false;
|
|
|
|
switch( save_mode )
|
|
{
|
|
case DO_DELETE_ON_SAVE:
|
|
result = remove(model_data);
|
|
break;
|
|
|
|
case DO_INSERT_ON_SAVE:
|
|
result = insert(model_data);
|
|
break;
|
|
|
|
case DO_UPDATE_ON_SAVE:
|
|
result = update(model_data);
|
|
break;
|
|
|
|
case DO_NOTHING_ON_SAVE:
|
|
result = true;
|
|
break;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
bool Model::save(ModelData & model_data)
|
|
{
|
|
return save(&model_data);
|
|
}
|
|
|
|
|
|
void Model::generate_select_columns(PT::TextStream & stream, const std::string & column_prefix, ModelData * model_data)
|
|
{
|
|
if( model_connector )
|
|
{
|
|
this->model_data = model_data;
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_GENERATING_DB_SQL;
|
|
DbConnector * db_connector = model_connector->get_db_connector();
|
|
|
|
if( db_connector )
|
|
{
|
|
db_connector->generate_select_columns(stream, *this, column_prefix);
|
|
}
|
|
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_NONE;
|
|
this->model_data = nullptr;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void Model::generate_doc_for_flat(PT::TextStream & stream, bool clear_stream)
|
|
{
|
|
if( clear_stream )
|
|
{
|
|
stream.clear();
|
|
}
|
|
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_GENERATING_DOC_FOR_FLAT;
|
|
|
|
if( model_connector )
|
|
{
|
|
FlatConnector * flat_connector = model_connector->get_doc_connector(); // different connector will be here
|
|
|
|
if( flat_connector )
|
|
{
|
|
flat_connector->to_text(stream, *this);
|
|
}
|
|
}
|
|
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_NONE;
|
|
}
|
|
|
|
|
|
void Model::generate_doc_for_db(PT::TextStream & stream, bool clear_stream)
|
|
{
|
|
if( clear_stream )
|
|
{
|
|
stream.clear();
|
|
}
|
|
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_GENERATING_DOC_FOR_DB;
|
|
|
|
if( model_connector )
|
|
{
|
|
FlatConnector * flat_connector = model_connector->get_doc_connector(); // different connector will be here
|
|
|
|
if( flat_connector )
|
|
{
|
|
flat_connector->to_text(stream, *this);
|
|
}
|
|
}
|
|
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_NONE;
|
|
}
|
|
|
|
|
|
void Model::map_values_from_query(QueryResult * query_result)
|
|
{
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_READING_VALUE_FROM_DB_RESULTSET;
|
|
this->query_result = query_result;
|
|
|
|
map_fields();
|
|
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_NONE;
|
|
this->query_result = nullptr;
|
|
}
|
|
|
|
|
|
|
|
void Model::clear()
|
|
{
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_CLEARING_VALUE;
|
|
map_fields();
|
|
model_connector_mode = MORM_MODEL_CONNECTOR_MODE_NONE;
|
|
|
|
save_mode = DO_INSERT_ON_SAVE;
|
|
model_data = nullptr;
|
|
}
|
|
|
|
|
|
void Model::prepare_to_select()
|
|
{
|
|
table_index = 1;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void Model::before_select()
|
|
{
|
|
}
|
|
|
|
void Model::before_insert()
|
|
{
|
|
}
|
|
|
|
void Model::before_update()
|
|
{
|
|
}
|
|
|
|
void Model::before_remove()
|
|
{
|
|
}
|
|
|
|
void Model::after_select()
|
|
{
|
|
}
|
|
|
|
void Model::after_insert()
|
|
{
|
|
}
|
|
|
|
void Model::after_update()
|
|
{
|
|
}
|
|
|
|
void Model::after_remove()
|
|
{
|
|
}
|
|
|
|
void Model::after_select_failure()
|
|
{
|
|
}
|
|
|
|
void Model::after_insert_failure()
|
|
{
|
|
}
|
|
|
|
void Model::after_update_failure()
|
|
{
|
|
}
|
|
|
|
void Model::after_remove_failure()
|
|
{
|
|
}
|
|
|
|
int Model::get_connector_mode()
|
|
{
|
|
return model_connector_mode;
|
|
}
|
|
|
|
bool Model::is_empty_field(const wchar_t * value)
|
|
{
|
|
return (!value || *value == '\0');
|
|
}
|
|
|
|
|
|
} // namespace
|
|
|