Re: [389-devel] Review of plugin code

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



> > 
> > Say for example:
> > 
> > begin
> > add some object Y
> > read Y
> > commit
> > 
> > Does read Y see the object within the transaction?
> 
> Yes.
> 
> > Is there a way to make the
> > search happen so that it occurs outside the transaction, IE it doesn't see 
> > Y?
> 
> Not a nested search operation.  A nested search operation will always 
> use the parent/context transaction.
> 

Makes sense. Just wanted to clarify so that I understood how different apis etc
will behave within a betxn context.


> > 
> > Is there a piece of documentation (perhaps the plugin guide) that lists the
> > order in which these operations are called?
> 
> Not sure, but in general it is:
> 
> incoming operation from client
> front end processing
> preoperation
> call backend
> bepreoperation
> start transaction
> betxnpreoperation
> do operation in the database
> betxnpostoperation
> end transaction
> bepostoperation
> return from backend
> send result to client
> postoperation
> 

I found this list in the Redhat Directory Services Plugin Development guide, but
thank you for it as well.

So with the transaction, there is only one transaction that covers all betxn
plugins? And I also wouldn't need to do anything to start / end the transaction
within a plugin either do I?

> > > The toughest part is the deadlock prevention.  At the start transaction,
> > > it holds a DB lock.  And most plug-ins maintain its own mutex to protect
> > > its resource.  It'd easily cause deadlock situation especially when
> > > multiple plug-ins are enabled (which is common :). So, please be careful
> > > not to acquire/free locks in the wrong order...
> > Of course. This is always an issue in multi-threaded code and anything with
> > locking. Stress tests are probably good to find these deadlocks, no?
> 
> Yes.  There is some code in dblayer.c that will stress the transaction 
> code by locking/unlocking many db pages concurrently with external 
> operations.
> https://git.fedorahosted.org/cgit/389/ds.git/tree/ldap/servers/slapd/back-ldbm
> /dblayer.c#n210
> https://git.fedorahosted.org/cgit/389/ds.git/tree/ldap/servers/slapd/back-ldbm
> /dblayer.c#n4131

Are there some unit tests hooked up that call and test this?

> > That answers some of my question. I guess the larger part of the question is 
> > how
> > the plugin subsystem treats each pluginType differently and the value of 
> > having
> > a plugin register to more than one pluginType. Are there some documents you 
> > can
> > point me to about this?
> 
> I'm not sure if there are docs which answer your specific question. But 
> a plugin may want to perform tasks at different points in the 
> operation.  For example, DNA may want to generate a unique value in the 
> pretxn phase, then roll back that value in the posttxn phase if the 
> operation failed.  Replication wants to many tasks at many different 
> points in the operation.

That makes sense.

Again, I found some documents about this in the plugin developers guide, and
I've re-read some other plugin code so I have a better grasp of this now. 

> 
> > 
> > Additionally, with betxn, this seems quite black-or-white. It's either on a 
> > ds
> > instance that has betxn support, so every update will be betxn capable, or 
> > it's
> > not on such a system so you fall back to other methods. Is this correct? 
> > With
> > new plugins is it even worth writing them without betxn support?
> 
> Correct.  It is not even worth writing a new plugin without betxn 
> support, if it does any update to the database that depends on other 
> updates to the database triggered by the incoming operation from the client.
> 

Fantastic. Thanks for the clarification.


For reference here is the plugin that this message references. It was a simple
demo to just get my head into plugin writing from scratch. I'm happy to accept
comments about it. I know that as a feature it's useless because
nsDS5ReplicaType already covers the use case, but it's a nice simple exercise
none the less.

Thanks for your time.

-- 
William Brown <william@xxxxxxxxxxxxxxxx>
/** BEGIN COPYRIGHT BLOCK
 * Copyright (C) 2010 Red Hat, Inc.
 * All rights reserved.
 *
 * License: GPL (version 3 or any later version).
 * See LICENSE for details. 
 * END COPYRIGHT BLOCK **/

#ifdef HAVE_CONFIG_H
#  include <config.h>
#endif

/*
 * Read Only Plug-in
 */
#include "read_only.h"
#include "slapi-private.h"
#include "slapi-plugin.h"

#define PLUGIN_NAME "readonly-plugin"
#define PLUGIN_DESC "prevent write operations from clients"

static Slapi_PluginDesc expdesc = { PLUGIN_NAME, VENDOR, DS_PACKAGE_VERSION, PLUGIN_DESC };

static int plugin_is_betxn = 0;
static void *_PluginID = NULL;

int read_only_init( Slapi_PBlock *pb );

static int readonly_pre_op( Slapi_PBlock *pb );
static int readonly_is_repl( Slapi_PBlock *pb );

/*
 * Plugin identity functions
 */
static void
read_only_set_plugin_id(void *pluginID)
{
    _PluginID = pluginID;
}

static void *
read_only_get_plugin_id()
{
    return _PluginID;
}


static int
readonly_is_repl_or_internal( Slapi_PBlock *pb)
{
    int is_repl = 0;
    int is_internal = 0;
    Slapi_Operation *operation = NULL;
    slapi_pblock_get(pb, SLAPI_OPERATION, &operation);
    slapi_pblock_get(pb, SLAPI_IS_REPLICATED_OPERATION, &is_repl);
    is_internal = slapi_operation_is_flag_set(operation, SLAPI_OP_FLAG_INTERNAL);
    if (is_repl != 0) {
        slapi_log_error(SLAPI_LOG_TRACE, PLUGIN_NAME, "readonly: is replicated operation\n");
    }
    if (is_internal != 0) {
        slapi_log_error(SLAPI_LOG_TRACE, PLUGIN_NAME, "readonly: is internal operation\n");
    }
    return is_repl + is_internal;

}

static int
readonly_pre_op( Slapi_PBlock *pb)
{
    int ret = SLAPI_PLUGIN_SUCCESS;
    // How do we ignore replicas and internal changes
    if (readonly_is_repl_or_internal(pb) != 0) {
        // Replicated op, do nothing.
        slapi_log_error(SLAPI_LOG_FATAL, PLUGIN_NAME, "readonly: ignored replica or internal operation\n" );

    } else {
        ret = SLAPI_PLUGIN_FAILURE;
        int rc = LDAP_UNWILLING_TO_PERFORM;
        char errtxt[SLAPI_DSE_RETURNTEXT_SIZE];
        PR_snprintf(errtxt, SLAPI_DSE_RETURNTEXT_SIZE,
                "Cannot alter read only instance.\n");
        slapi_pblock_set(pb, SLAPI_PB_RESULT_TEXT, &errtxt);
        slapi_pblock_set(pb, SLAPI_RESULT_CODE, &rc);
        slapi_log_error(SLAPI_LOG_FATAL, PLUGIN_NAME, "readonly: intercepted operation\n" );
    }

    return ret;
}

int read_only_init( Slapi_PBlock *pb )
{
    Slapi_Entry *plugin_entry = NULL;
    char *plugin_identity = NULL;
    char *plugin_type = NULL;
    int status = 0;
    int preadd = SLAPI_PLUGIN_PRE_ADD_FN;
    int premod = SLAPI_PLUGIN_PRE_MODIFY_FN;
    int predel = SLAPI_PLUGIN_PRE_DELETE_FN;
    int premdn = SLAPI_PLUGIN_PRE_MODRDN_FN;

    slapi_pblock_get(pb, SLAPI_PLUGIN_IDENTITY, &plugin_identity);
    PR_ASSERT(plugin_identity);
    read_only_set_plugin_id(plugin_identity);

    if ((slapi_pblock_get(pb, SLAPI_PLUGIN_CONFIG_ENTRY, &plugin_entry) == 0) && plugin_entry &&
        (plugin_type = slapi_entry_attr_get_charptr(plugin_entry, "nsslapd-plugintype")) &&
        plugin_type && strstr(plugin_type, "betxn")) {
        plugin_is_betxn = 1;
        preadd = SLAPI_PLUGIN_BE_TXN_PRE_ADD_FN;
        premod = SLAPI_PLUGIN_BE_TXN_PRE_MODIFY_FN;
        predel = SLAPI_PLUGIN_BE_TXN_PRE_DELETE_FN;
        premdn = SLAPI_PLUGIN_BE_TXN_PRE_MODRDN_FN;
    }
    slapi_log_error( SLAPI_LOG_TRACE, PLUGIN_NAME, plugin_type);
    slapi_log_error( SLAPI_LOG_TRACE, PLUGIN_NAME, "\n");
    slapi_ch_free_string(&plugin_type);

	if ( slapi_pblock_set( pb, SLAPI_PLUGIN_VERSION, SLAPI_PLUGIN_VERSION_03 ) != 0 ||
		slapi_pblock_set( pb, SLAPI_PLUGIN_DESCRIPTION, ( void * )&expdesc ) != 0 ||
        slapi_pblock_set( pb, preadd, (void *) readonly_pre_op) != 0 ||
        slapi_pblock_set( pb, premod, (void *) readonly_pre_op) != 0 ||
        slapi_pblock_set( pb, predel, (void *) readonly_pre_op) != 0 ||
        slapi_pblock_set( pb, premdn, (void *) readonly_pre_op) != 0
)
	{
		slapi_log_error( SLAPI_LOG_FATAL, PLUGIN_NAME, "read_only: Failed to register plug-in.\n" );
		status = -1;
	}

    /*
    if (!status && !plugin_is_betxn &&
        slapi_register_plugin("internalpreoperation",
                                1,
                                "read_only_init",
                                read_only_init,
                                PLUGIN_DESC,
                                NULL,
                                plugin_identity
        )) {
        slapi_log_error(SLAPI_LOG_FATAL, PLUGIN_NAME, "Failed to register internal preoperation\n");
        status = -1;
    }

    if (!status) {
        plugin_type = plugin_is_betxn ? "betxnpreoperation" : "preoperation";
        if (slapi_register_plugin(plugin_type,
                                1,
                                "read_only_init",
                                read_only_init,
                                PLUGIN_DESC,
                                NULL,
                                plugin_identity
        )) {
            slapi_log_error(SLAPI_LOG_FATAL, PLUGIN_NAME, "Failed to register internal betxnpreoperation\n");
            status = -1;
        }
    }
    */

    slapi_log_error( SLAPI_LOG_FATAL, PLUGIN_NAME, "readonly registered\n" );
    return( status );
}

--
389-devel mailing list
389-devel@xxxxxxxxxxxxxxxxxxxxxxx
https://admin.fedoraproject.org/mailman/listinfo/389-devel

[Index of Archives]     [Fedora Directory Announce]     [Fedora Users]     [Older Fedora Users Mail]     [Fedora Advisory Board]     [Fedora Security]     [Fedora Devel Java]     [Fedora Desktop]     [ATA RAID]     [Fedora Marketing]     [Fedora Mentors]     [Fedora Package Review]     [Fedora Art]     [Fedora Music]     [Fedora Packaging]     [CentOS]     [Fedora SELinux]     [Big List of Linux Books]     [KDE Users]     [Fedora Art]     [Fedora Docs]

  Powered by Linux