scality/Arsenal

Don't use `Kinetic.errors.XXX` for client API

adrienverge opened this issue · 8 comments

Kinetic errors (such as Kinetic.errors.SUCCESS, Kinetic.errors.HMAC_FAILURE, Kinetic.errors.NOT_AUTHORIZED...) are special, protocol-defined values meant to be embedded in Kinetic messages. There are not meant to be returned to client code calling the library.

Some Kinetic lib functions (e.g. send and _parse) should be modified to return in a Node.js way (either throw exception or take a callback).

Ok I put this in my TODO

Something like that ?

    _parse(data, callback) {
        const version = data.readInt8(0);
        const pbMsgLen = data.readInt32BE(1);
        const chunkLen = data.readInt32BE(5);

        if (version !== getVersion()) {
            return callback(new Error('Version Failure'));
        }

        try {
            this._cmd = protoBuild.Message.decode(data.slice(9, 9 + pbMsgLen));
            this.setProtobuf(protoBuild.Command.decode(this._cmd.commandBytes));
        } catch (e) {
            if (e.decoded) {
                this.setProtobuf(e.decoded);
            } else {
                return callback(new Error('Error in decode' + e));
            }
        }
        this.setChunk(data.slice(pbMsgLen + 9, chunkLen + pbMsgLen + 9));

        if (this.getChunkSize() !== chunkLen) {
            return callback(new Error('data error - different len'));
        }
        if (this._cmd.authType === 1 &&
                !this.hmacIntegrity(this.getSlice(this._cmd.hmacAuth.hmac)))
            return callback(new Error('Hmac fail'));
        return callback(null);
    }

with a constructor like this :

    constructor(input, callback) {
        /*
         * From Kinetic official documentation, a Kinetic Protocol Data Unit
         * (PDU) is composed of:
         * - message -- a Protocol Buffer (protobuf) message, containing
         *              operation metadata & key-value metadata,
         * - chunk   -- the value to store/read.
         */
        this._message = undefined;
        this._chunk = undefined;

        if (input !== undefined) {
            if (!Buffer.isBuffer(input))
                return callback(new Error("input is not a buffer"));
            this._parse(input, (err) => {
                if (err){
                    return callback(new Error("could not parse input buffer (" + err + ")"));
                } else {
                    return callback(null, this);
                }
            });
        }
    }

Since both _parse and constructor are synchronous, I wouldn't use callbacks here.

Instead, they could throw exceptions, explicitely say it their jsdoc, and all code calling them should check for exceptions.

Example:

_parse(data) {
    if (version !== getVersion()) {
        const err = new Error('Version Failure');
        err.badVersion = true;
        throw err;
    }
}

Client code:

try {
    const pdu = new Kinetic.PDU(new Buffer("sdkfljsdf"));
} catch (e) {
    if (e.badVersion) {
        log.error("bad version!");
    }
}

Okok

And for send ? I tried something like that but i'm not really good on error handling, so I don't really know what is the best way

        try {
            if (this.getChunk() !== undefined)
                sock.write(Buffer.concat(
                    [pduHeader, this._message.toBuffer(), this.getChunk()]));
            else
                sock.write(Buffer.concat([pduHeader,
                    this._message.toBuffer()]));
        } catch (e) {
            return callback(e);
        }
        return callback(null);

I did this before, I just forgot to update my comment X)

I tried to do a better error throwing/handling on the branch dev/CLEANUP/error_return
There are some tests on the branch

Thanks @AntoninCoulibaly. The branch you created brings lots of enhancements, can you create a pull request for it? I have a few comments to make, but mostly 👍

Ok no problem I will

Closing, since this is not relevant in this repo anymore.