package wmo

import (
	"context"
	"errors"
	"go.mongodb.org/mongo-driver/bson"
	"go.mongodb.org/mongo-driver/mongo"
	"gogs.mikescher.com/BlackForestBytes/goext/exerr"
	"gogs.mikescher.com/BlackForestBytes/goext/langext"
)

func (c *Coll[TData]) FindOne(ctx context.Context, filter bson.M) (TData, error) {
	r, err := c.findOneInternal(ctx, filter, false)
	if err != nil {
		if filterId, ok := filter["_id"]; ok {
			return *new(TData), exerr.Wrap(err, "mongo-query[find-one] failed").Str("collection", c.Name()).Any("filter", filter).Any("filter_id", filterId).Build()
		} else {
			return *new(TData), exerr.Wrap(err, "mongo-query[find-one] failed").Str("collection", c.Name()).Any("filter", filter).Build()
		}
	}

	return *r, nil
}

func (c *Coll[TData]) FindOneOpt(ctx context.Context, filter bson.M) (*TData, error) {
	r, err := c.findOneInternal(ctx, filter, true)
	if err != nil {
		return nil, exerr.Wrap(err, "mongo-query[find-one-opt] failed").Str("collection", c.Name()).Any("filter", filter).Build()
	}

	return r, nil
}

func (c *Coll[TData]) FindOneByID(ctx context.Context, id EntityID) (TData, error) {
	r, err := c.findOneInternal(ctx, bson.M{"_id": id}, false)
	if err != nil {
		return *new(TData), exerr.Wrap(err, "mongo-query[find-one-by-id] failed").Id("id", id).Str("collection", c.Name()).Build()
	}

	return *r, nil
}

func (c *Coll[TData]) FindOneOptByID(ctx context.Context, id EntityID) (*TData, error) {
	r, err := c.findOneInternal(ctx, bson.M{"_id": id}, true)
	if err != nil {
		return nil, exerr.Wrap(err, "mongo-query[find-one-opt-by-id] failed").Id("id", id).Str("collection", c.Name()).Build()
	}

	return r, nil
}

func (c *Coll[TData]) findOneInternal(ctx context.Context, filter bson.M, allowNull bool) (*TData, error) {

	if len(c.extraModPipeline) == 0 {

		// simple case, use mongo FindOne

		mongoRes := c.coll.FindOne(ctx, filter)

		res, err := c.decodeSingle(ctx, mongoRes)
		if allowNull && errors.Is(err, mongo.ErrNoDocuments) {
			return nil, nil
		}
		if err != nil {
			if filterId, ok := filter["_id"]; ok {
				return nil, exerr.Wrap(err, "mongo-query[find-one|internal] failed").Str("collection", c.Name()).Any("filter", filter).Any("filter_id", filterId).NoLog().Build()
			} else {
				return nil, exerr.Wrap(err, "mongo-query[find-one|internal] failed").Str("collection", c.Name()).Any("filter", filter).NoLog().Build()
			}
		}

		return &res, nil

	} else {

		// complex case, we one ore more additional pipeline stages, convert to aggregation

		pipeline := mongo.Pipeline{}
		pipeline = append(pipeline, bson.D{{Key: "$match", Value: filter}})
		pipeline = append(pipeline, bson.D{{Key: "$limit", Value: 1}})

		for _, ppl := range c.extraModPipeline {
			pipeline = langext.ArrConcat(pipeline, ppl(ctx))
		}

		cursor, err := c.coll.Aggregate(ctx, pipeline)
		if err != nil {
			return nil, exerr.Wrap(err, "mongo-aggregation [find-one] failed").Any("pipeline", pipeline).Str("collection", c.Name()).NoLog().Build()
		}

		defer func() { _ = cursor.Close(ctx) }()

		if cursor.Next(ctx) {
			v, err := c.decodeSingle(ctx, cursor)
			if err != nil {
				return nil, exerr.Wrap(err, "mongo-aggregation [find-one] failed to decode results").Any("pipeline", pipeline).Str("collection", c.Name()).NoLog().Build()
			}
			return &v, nil
		} else if allowNull {
			return nil, nil
		} else {
			return nil, exerr.Wrap(mongo.ErrNoDocuments, "mongo-aggregation [find-one] returned no documents").Any("pipeline", pipeline).Str("collection", c.Name()).NoLog().Build()
		}
	}
}