﻿using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using VW.Serializer.Attributes;
using VW.Serializer;
using VW;
using VW.Labels;

namespace cs_test
{
    public static class AttributesSample
    {
        public static void Attributes()
        {
            var d1 = new DocumentFeature
            {
                Id = "d1",
                Time = new DateTime(2015, 1, 1),
                Value = new LDAFeatureVector { Values = new[] { 1.0, 2.0, 3.0 } }
            };

            var context = new UserContext
            {
                User = new UserFeature
                {
                    Age = Age.Adult,
                    Gender = Gender.Female,
                    Location = "WA",
                    FeatureBag = new Dictionary<string, float>
                    {
                        { "Foo", 1.1f },
                        { "Bar", 2.0f }
                    }
                },
                UserLDATopicPreference = new LDAFeatureVector { Values = new[] { 0.1, 0.2, 0.3 } },
                ActionDependentFeatures = new List<DocumentFeature>
                {
                    d1,
                    new DocumentFeature
                    {
                        Id = "d2",
                        Time = new DateTime(2015,1,1),
                        Value = new LDAFeatureVector { Values = new [] { 1.0, 2.0, 3.0 } }
                    },
                    d1
                }
            };

            //var visitor = new VowpalWabbitStringVisitor();
            //var serializer = VowpalWabbitSerializerFactory.CreateSerializer<UserContext>(visitor);
            //var serializerDependent = VowpalWabbitSerializerFactory.CreateSerializer<DocumentFeature>(visitor);
            //Console.WriteLine(serializer.Serialize(context));

            //foreach (var actionDependentFeature in context.ActionDependentFeatures)
            //{
            //    Console.WriteLine(serializerDependent.Serialize(actionDependentFeature));
            //}

            //using (var pool = new ObjectPool<VowpalWabbit<UserContext, DocumentFeature>>(() => new VowpalWabbit<UserContext, DocumentFeature>("")))
            //{
            //    using (var vw = pool.Get())
            //    {
            //        // do work with VW
            //        // vw.Value.CreateEmptyExample();
            //    }

            //    // don't modify this model from another thread!
            //    var newVwModel = new VowpalWabbitModel("model init");
            //    pool.UpdateFactory(() => new VowpalWabbit<UserContext, DocumentFeature>(newVwModel));

            //    // this will get a new VW instance with a newer version!
            //    using (var vw = pool.Get())
            //    {
            //        // do work with VW
            //        // vw.Value.CreateEmptyExample();
            //    }
            //}

            Console.ReadKey();
        }

        public static void RunFeaturesTest()
        {
            var context = new FeatureTestContext
            {
                S = new[] { "p^the_man", "w^thew^man\u0394", "w^man" },
                T = new[] { "p^un_homme", "w^un", "w^homme" }
            };

            var vw = new VowpalWabbit<FeatureTestContext>("-q st --noconstant --quiet");

            vw.Native.Learn("1 |s p^the_man w^the w^man |t p^un_homme w^un w^homme");

            var prediction = vw.Learn(context, new SimpleLabel { Label = 1f }, VowpalWabbitPredictionType.Scalar);

            Console.Error.WriteLine("p2 = {0}", prediction);
        }
    }

    public class FeatureTestContext
    {
        [Feature(FeatureGroup = 's')]
        public IEnumerable<string> S { get; set; }

        [Feature(FeatureGroup = 't')]
        public IEnumerable<string> T { get; set; }
    }

    public class UserContext
    {
        [Feature(Namespace = "otheruser", FeatureGroup = 'o')]
        public UserFeature User { get; set; }

        [Feature(Namespace = "userlda", FeatureGroup = 'u')]
        public LDAFeatureVector UserLDATopicPreference { get; set; }

        public IReadOnlyList<DocumentFeature> ActionDependentFeatures { get; set; }
    }

    [Cacheable(EqualityComparer = typeof(DocumentFeatureEqualityComparer))]
    public class DocumentFeature
    {
        public string Id { get; set; }

        public string SomeOtherId { get; set; }

        public DateTime Time { get; set; }

        // If we include this, it would result in mixing dense and non-dense features.
        // [Feature]
        public string ContentProvider { get; set; }

        [Feature(Namespace = "doclda", FeatureGroup = 'd')]
        public LDAFeatureVector Value { get; set; }
    }

    public class DocumentFeatureEqualityComparer : IEqualityComparer<DocumentFeature>
    {
        public bool Equals(DocumentFeature x, DocumentFeature y)
        {
            return x.Id == y.Id &&
                   x.Time == y.Time;
            // maybe compare the full vector - not so sure on this part though
            //   x.Value.Zip(y.Value, (a, b) => a == b).All(c => c);
        }

        public int GetHashCode(DocumentFeature obj)
        {
            return obj.Id.GetHashCode() + obj.Time.GetHashCode();
        }
    }

    public enum Gender
    {
        Female,
        Male,
        Unknown
    }

    public class LDAFeatureVector
    {
        public string Compressed { get; set; }

        private double[] values;

        [Feature]
        public double[] Values
        {
            get
            {
                if (this.Compressed == null)
                {
                    // e.g. call into decompression
                }
                return this.values;
            }
            set { this.values = value; }
        }
    }

    public class UserFeature
    {
        [Feature]
        public Age Age { get; set; }

        [Feature]
        public int? Income { get; set; }

        [Feature]
        public Gender Gender { get; set; }

        [Feature]
        public string Location { get; set; }

        [Feature]
        public Dictionary<string, float> FeatureBag { get; set; }

        [Feature]
        public DayOfWeek DayOfWeek { get; set; }

        /// <summary>
        /// Will generate 24 parameters
        /// </summary>
        [Feature(Enumerize = true)]
        public int HourOfDay { get; set; }
    }

    public enum Age
    {
        Child,
        Adult,
        Elderly
    }
}
