using System;
using System.Collections.Generic;
using SingularityGroup.HotReload.DTO;
using SingularityGroup.HotReload.Newtonsoft.Json;

namespace SingularityGroup.HotReload.JsonConverters {
    internal class MethodPatchResponsesConverter : JsonConverter {
        public override bool CanConvert(Type objectType) {
            return objectType == typeof(List<MethodPatchResponse>);
        }

        public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer) {
            var list = new List<MethodPatchResponse>();

            while (reader.Read()) {
                if (reader.TokenType == JsonToken.StartObject) {
                    list.Add(ReadMethodPatchResponse(reader));
                } else if (reader.TokenType == JsonToken.EndArray) {
                    break; // End of the SMethod list
                }
            }

            return list;
        }
        
        private MethodPatchResponse ReadMethodPatchResponse(JsonReader reader) {
            string id = null;
            CodePatch[] patches = null;
            string[] failures = null;
            SMethod[] removedMethod = null;
            SField[] alteredFields = null;
            SField[] addedFieldInitializerFields = null;
            SMethod[] addedFieldInitializerInitializers = null;
            SField[] removedFieldInitializers = null;
            SField[] newFieldDefinitions = null;

            while (reader.Read()) {
                if (reader.TokenType == JsonToken.EndObject) {
                    break;
                }
                if (reader.TokenType != JsonToken.PropertyName) {
                    continue;
                }
                var propertyName = (string)reader.Value;

                switch (propertyName) {
                    case nameof(MethodPatchResponse.id):
                        id = reader.ReadAsString();
                        break;

                    case nameof(MethodPatchResponse.patches):
                        patches = ReadPatches(reader);
                        break;

                    case nameof(MethodPatchResponse.failures):
                        failures = ReadStringArray(reader);
                        break;

                    case nameof(MethodPatchResponse.removedMethod):
                        removedMethod = ReadSMethodArray(reader);
                        break;
                    
                    case nameof(MethodPatchResponse.alteredFields):
                        alteredFields = ReadSFields(reader);
                        break;

                    case nameof(MethodPatchResponse.addedFieldInitializerFields):
                        addedFieldInitializerFields = ReadSFields(reader);
                        break;

                    case nameof(MethodPatchResponse.addedFieldInitializerInitializers):
                        addedFieldInitializerInitializers = ReadSMethodArray(reader);
                        break;

                    case nameof(MethodPatchResponse.removedFieldInitializers):
                        removedFieldInitializers = ReadSFields(reader);
                        break;
                        
                    case nameof(MethodPatchResponse.newFieldDefinitions):
                        newFieldDefinitions = ReadSFields(reader);
                        break;

                    default:
                        reader.Skip(); // Skip unknown properties
                        break;
                }
            }
            
            return new MethodPatchResponse(
                id ?? string.Empty,
                patches ?? Array.Empty<CodePatch>(), 
                failures ?? Array.Empty<string>(), 
                removedMethod ?? Array.Empty<SMethod>(),
                alteredFields ?? Array.Empty<SField>(),
                // Note: suggestions don't have to be persisted here 
                Array.Empty<PartiallySupportedChange>(),
                Array.Empty<HotReloadSuggestionKind>(),
                addedFieldInitializerFields ?? Array.Empty<SField>(),
                addedFieldInitializerInitializers ?? Array.Empty<SMethod>(),
                removedFieldInitializers ?? Array.Empty<SField>(),
                newFieldDefinitions ?? Array.Empty<SField>()
            );
        }

        private CodePatch[] ReadPatches(JsonReader reader) {
            var patches = new List<CodePatch>();

            while (reader.Read()) {
                if (reader.TokenType == JsonToken.EndArray) {
                    break;
                }
                if (reader.TokenType != JsonToken.StartObject) {
                    continue;
                }
                string patchId = null;
                string assemblyName = null;
                byte[] patchAssembly = null;
                byte[] patchPdb = null;
                SMethod[] modifiedMethods = null;
                SMethod[] patchMethods = null;
                SMethod[] newMethods = null;
                SUnityJob[] unityJobs = null;
                SField[] newFields = null;
                SField[] deletedFields = null;
                SField[] renamedFieldsFrom = null;
                SField[] renamedFieldsTo = null;
                SField[] propertyAttributesFieldOriginal = null;
                SField[] propertyAttributesFieldUpdated = null;

                while (reader.Read()) {
                    if (reader.TokenType == JsonToken.EndObject) {
                        break;
                    }
                    if (reader.TokenType != JsonToken.PropertyName) {
                        continue;
                    }
                    var propertyName = (string)reader.Value;

                    switch (propertyName) {
                        case nameof(CodePatch.patchId):
                            patchId = reader.ReadAsString();
                            break;

                        case nameof(CodePatch.assemblyName):
                            assemblyName = reader.ReadAsString();
                            break;
                        
                        case nameof(CodePatch.patchAssembly):
                            patchAssembly = Convert.FromBase64String(reader.ReadAsString());
                            break;
                        
                        case nameof(CodePatch.patchPdb):
                            patchPdb = Convert.FromBase64String(reader.ReadAsString());
                            break;
                        
                        case nameof(CodePatch.modifiedMethods):
                            modifiedMethods = ReadSMethodArray(reader);
                            break;
                        
                        case nameof(CodePatch.patchMethods):
                            patchMethods = ReadSMethodArray(reader);
                            break;
                        
                        case nameof(CodePatch.newMethods):
                            newMethods = ReadSMethodArray(reader);
                            break;
                        
                        case nameof(CodePatch.unityJobs):
                            unityJobs = ReadSUnityJobArray(reader);
                            break;
                        
                        case nameof(CodePatch.newFields):
                            newFields = ReadSFields(reader);
                            break;
                        case nameof(CodePatch.deletedFields):
                            deletedFields = ReadSFields(reader);
                            break;
                        
                        case nameof(CodePatch.renamedFieldsFrom):
                            renamedFieldsFrom = ReadSFields(reader);
                            break;
                        
                        case nameof(CodePatch.renamedFieldsTo):
                            renamedFieldsTo = ReadSFields(reader);
                            break;
                        
                        case nameof(CodePatch.propertyAttributesFieldOriginal):
                            propertyAttributesFieldOriginal = ReadSFields(reader);
                            break;
                        
                        case nameof(CodePatch.propertyAttributesFieldUpdated):
                            propertyAttributesFieldUpdated = ReadSFields(reader);
                            break;

                        default:
                            reader.Skip(); // Skip unknown properties
                            break;
                    }
                }

                patches.Add(new CodePatch(
                    patchId: patchId ?? string.Empty,
                    assemblyName: assemblyName ?? string.Empty,
                    patchAssembly: patchAssembly ?? Array.Empty<byte>(),
                    patchPdb: patchPdb ?? Array.Empty<byte>(),
                    modifiedMethods: modifiedMethods ?? Array.Empty<SMethod>(),
                    patchMethods: patchMethods ?? Array.Empty<SMethod>(),
                    newMethods: newMethods ?? Array.Empty<SMethod>(),
                    unityJobs: unityJobs ?? Array.Empty<SUnityJob>(),
                    newFields: newFields ?? Array.Empty<SField>(),
                    deletedFields: deletedFields ?? Array.Empty<SField>(),
                    renamedFieldsFrom: renamedFieldsFrom ?? Array.Empty<SField>(),
                    renamedFieldsTo: renamedFieldsTo ?? Array.Empty<SField>(),
                    propertyAttributesFieldOriginal: propertyAttributesFieldOriginal ?? Array.Empty<SField>(),
                    propertyAttributesFieldUpdated: propertyAttributesFieldUpdated ?? Array.Empty<SField>()
                ));
            }

            return patches.ToArray();
        }

        private string[] ReadStringArray(JsonReader reader) {
            var list = new List<string>();

            while (reader.Read()) {
                if (reader.TokenType == JsonToken.String) {
                    list.Add((string)reader.Value);
                } else if (reader.TokenType == JsonToken.EndArray) {
                    break; // End of the string list
                }
            }

            return list.ToArray();
        }

        private SMethod[] ReadSMethodArray(JsonReader reader) {
            var list = new List<SMethod>();

            while (reader.Read()) {
                if (reader.TokenType == JsonToken.StartObject) {
                    list.Add(ReadSMethod(reader));
                } else if (reader.TokenType == JsonToken.EndArray) {
                    break; // End of the SMethod list
                }
            }

            return list.ToArray();
        }

        private SType[] ReadSTypeArray(JsonReader reader) {
            var list = new List<SType>();

            while (reader.Read()) {
                if (reader.TokenType == JsonToken.StartObject) {
                    list.Add(ReadSType(reader));
                } else if (reader.TokenType == JsonToken.EndArray) {
                    break; // End of the SType list
                }
            }

            return list.ToArray();
        }
        
        private SUnityJob[] ReadSUnityJobArray(JsonReader reader) {
            var array = new List<SUnityJob>();

            while (reader.Read()) {
                if (reader.TokenType == JsonToken.StartObject) {
                    array.Add(ReadSUnityJob(reader));
                } else if (reader.TokenType == JsonToken.EndArray) {
                    break; // End of the SUnityJob array
                }
            }

            return array.ToArray();
        }
        
        private SField[] ReadSFields(JsonReader reader) {
            var array = new List<SField>();
            while (reader.Read()) {
                if (reader.TokenType == JsonToken.StartObject) {
                    array.Add(ReadSField(reader));
                } else if (reader.TokenType == JsonToken.EndArray) {
                    break; // End of the SUnityJob array
                }
            }
            return array.ToArray();
        }

        private SMethod ReadSMethod(JsonReader reader) {
            string assemblyName = null;
            string displayName = null;
            int metadataToken = default(int);
            string simpleName = null;

            while (reader.Read()) {
                if (reader.TokenType == JsonToken.EndObject) {
                    break;
                }
                if (reader.TokenType != JsonToken.PropertyName) {
                    continue;
                }
                var propertyName = (string)reader.Value;

                switch (propertyName) {
                    case nameof(SMethod.assemblyName):
                        assemblyName = reader.ReadAsString();
                        break;

                    case nameof(SMethod.displayName):
                        displayName = reader.ReadAsString();
                        break;
                    
                    case nameof(SMethod.metadataToken):
                        metadataToken = reader.ReadAsInt32() ?? default(int);
                        break;
                    
                    case nameof(SMethod.simpleName):
                        simpleName = reader.ReadAsString();
                        break;

                    default:
                        reader.Skip(); // Skip unknown properties
                        break;
                }
            }

            return new SMethod(
                assemblyName ?? string.Empty,
                displayName ?? string.Empty,
                metadataToken, 
                simpleName ?? string.Empty
            );
        }

        private SType ReadSType(JsonReader reader) {
            string assemblyName = null;
            string typeName = null;
            int? metadataToken = null;

            while (reader.Read()) {
                if (reader.TokenType == JsonToken.Null) {
                    return null;
                }
                if (reader.TokenType == JsonToken.EndObject) {
                    break;
                }
                if (reader.TokenType != JsonToken.PropertyName) {
                    continue;
                }
                var propertyName = (string)reader.Value;

                switch (propertyName) {
                    case nameof(SType.assemblyName):
                        assemblyName = reader.ReadAsString();
                        break;

                    case nameof(SType.typeName):
                        typeName = reader.ReadAsString();
                        break;

                    case nameof(SType.metadataToken):
                        metadataToken = reader.ReadAsInt32();
                        break;

                    default:
                        reader.Skip(); // Skip unknown properties
                        break;
                }
            }

            return new SType(
                assemblyName ?? string.Empty,
                typeName ?? string.Empty,
                metadataToken ?? 0
            );
        }

        private SUnityJob ReadSUnityJob(JsonReader reader) {
            int metadataToken = default(int);
            UnityJobKind jobKind = default(UnityJobKind);

            while (reader.Read()) {
                if (reader.TokenType == JsonToken.EndObject) {
                    break;
                }
                if (reader.TokenType != JsonToken.PropertyName) {
                    continue;
                }
                var propertyName = (string)reader.Value;

                switch (propertyName) {
                    case nameof(SUnityJob.metadataToken):
                        metadataToken = reader.ReadAsInt32() ?? 0;
                        break;

                    case nameof(SUnityJob.jobKind):
                        var jobKindStr = reader.ReadAsString();
                        Enum.TryParse(jobKindStr, out jobKind);
                        break;

                    default:
                        reader.Skip(); // Skip unknown properties
                        break;
                }
            }

            return new SUnityJob(metadataToken, jobKind);
        }
        
        private SField ReadSField(JsonReader reader) {
            SType declaringType = null;
            string fieldName = null;
            string assemblyName = null;
            int? metadataToken = null;
            bool? serializable = null;
            bool? isStatic = null;

            while (reader.Read()) {
                if (reader.TokenType == JsonToken.EndObject) {
                    break;
                }
                if (reader.TokenType != JsonToken.PropertyName) {
                    continue;
                }
                var propertyName = (string)reader.Value;

                switch (propertyName) {
                    case nameof(SField.declaringType):
                        declaringType = ReadSType(reader);
                        break;
                    
                    case nameof(SField.fieldName):
                        fieldName = reader.ReadAsString();
                        break;

                    case nameof(SField.assemblyName):
                        assemblyName = reader.ReadAsString();
                        break;
                    
                    case nameof(SField.metadataToken):
                        metadataToken = reader.ReadAsInt32();
                        break;

                    case nameof(SField.serializable):
                        serializable = reader.ReadAsBoolean();
                        break;
                    
                    case nameof(SField.isStatic):
                        isStatic = reader.ReadAsBoolean();
                        break;
                    
                    default:
                        reader.Skip(); // Skip unknown properties
                        break;
                }
            }

            return new SField(declaringType: declaringType, fieldName: fieldName, assemblyName: assemblyName, metadataToken ?? 0, isStatic ?? false, serializable ?? false);
        }

        public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer) {
            var responses = (List<MethodPatchResponse>)value;
            if (responses == null) {
                writer.WriteNull();
                return;
            }
            
            writer.WriteStartArray();
            foreach (var response in responses) {
                writer.WriteStartObject();
                
                writer.WritePropertyName(nameof(response.id));
                writer.WriteValue(response.id);

                if (response.patches != null) {
                    writer.WritePropertyName(nameof(response.patches));
                    writer.WriteStartArray();
                    foreach (var responsePatch in response.patches) {
                        writer.WriteStartObject();
                        
                        writer.WritePropertyName(nameof(responsePatch.patchId));
                        writer.WriteValue(responsePatch.patchId);
                        writer.WritePropertyName(nameof(responsePatch.assemblyName));
                        writer.WriteValue(responsePatch.assemblyName);
                        writer.WritePropertyName(nameof(responsePatch.patchAssembly));
                        writer.WriteValue(Convert.ToBase64String(responsePatch.patchAssembly));
                        writer.WritePropertyName(nameof(responsePatch.patchPdb));
                        writer.WriteValue(Convert.ToBase64String(responsePatch.patchPdb));

                        if (responsePatch.modifiedMethods != null) {
                            writer.WritePropertyName(nameof(responsePatch.modifiedMethods));
                            writer.WriteStartArray();
                            foreach (var modifiedMethod in responsePatch.modifiedMethods) {
                                WriteSMethod(writer, modifiedMethod);
                            }
                            writer.WriteEndArray();
                        }

                        if (responsePatch.patchMethods != null) {
                            writer.WritePropertyName(nameof(responsePatch.patchMethods));
                            writer.WriteStartArray();
                            foreach (var patchMethod in responsePatch.patchMethods) {
                                WriteSMethod(writer, patchMethod);
                            }
                            writer.WriteEndArray();
                        }

                        if (responsePatch.newMethods != null) {
                            writer.WritePropertyName(nameof(responsePatch.newMethods));
                            writer.WriteStartArray();
                            foreach (var newMethod in responsePatch.newMethods) {
                                WriteSMethod(writer, newMethod);
                            }
                            writer.WriteEndArray();
                        }

                        if (responsePatch.unityJobs != null) {
                            writer.WritePropertyName(nameof(responsePatch.unityJobs));
                            writer.WriteStartArray();
                            foreach (var unityJob in responsePatch.unityJobs) {
                                writer.WriteStartObject();

                                writer.WritePropertyName(nameof(unityJob.metadataToken));
                                writer.WriteValue(unityJob.metadataToken);
                                writer.WritePropertyName(nameof(unityJob.jobKind));
                                writer.WriteValue(unityJob.jobKind.ToString());

                                writer.WriteEndObject();
                            }
                            writer.WriteEndArray();
                        }
                        
                        if (responsePatch.newFields != null) {
                            writer.WritePropertyName(nameof(responsePatch.newFields));
                            writer.WriteStartArray();
                            foreach (var newField in responsePatch.newFields) {
                                WriteSField(writer, newField);
                            }
                            writer.WriteEndArray();
                        }
                        
                        if (responsePatch.deletedFields != null) {
                            writer.WritePropertyName(nameof(responsePatch.deletedFields));
                            writer.WriteStartArray();
                            foreach (var deletedField in responsePatch.deletedFields) {
                                WriteSField(writer, deletedField);
                            }
                            writer.WriteEndArray();
                        }
                        
                        if (responsePatch.renamedFieldsFrom != null) {
                            writer.WritePropertyName(nameof(responsePatch.renamedFieldsFrom));
                            writer.WriteStartArray();
                            foreach (var removedFieldFrom in responsePatch.renamedFieldsFrom) {
                                WriteSField(writer, removedFieldFrom);
                            }
                            writer.WriteEndArray();
                        }
                        
                        if (responsePatch.renamedFieldsTo != null) {
                            writer.WritePropertyName(nameof(responsePatch.renamedFieldsTo));
                            writer.WriteStartArray();
                            foreach (var removedFieldTo in responsePatch.renamedFieldsTo) {
                                WriteSField(writer, removedFieldTo);
                            }
                            writer.WriteEndArray();
                        }
                        
                        if (responsePatch.propertyAttributesFieldOriginal != null) {
                            writer.WritePropertyName(nameof(responsePatch.propertyAttributesFieldOriginal));
                            writer.WriteStartArray();
                            foreach (var removedFieldFrom in responsePatch.propertyAttributesFieldOriginal) {
                                WriteSField(writer, removedFieldFrom);
                            }
                            writer.WriteEndArray();
                        }
                        
                        if (responsePatch.propertyAttributesFieldUpdated != null) {
                            writer.WritePropertyName(nameof(responsePatch.propertyAttributesFieldUpdated));
                            writer.WriteStartArray();
                            foreach (var removedFieldTo in responsePatch.propertyAttributesFieldUpdated) {
                                WriteSField(writer, removedFieldTo);
                            }
                            writer.WriteEndArray();
                        }

                        writer.WriteEndObject();
                    }
                    writer.WriteEndArray();
                }

                if (response.failures != null) {
                    writer.WritePropertyName(nameof(response.failures));
                    writer.WriteStartArray();
                    foreach (var failure in response.failures) {
                        writer.WriteValue(failure);
                    }
                    writer.WriteEndArray();
                }

                if (response.removedMethod != null) {
                    writer.WritePropertyName(nameof(response.removedMethod));
                    writer.WriteStartArray();
                    foreach (var removedMethod in response.removedMethod) {
                        WriteSMethod(writer, removedMethod);
                    }
                    writer.WriteEndArray();
                }
                
                if (response.alteredFields != null) {
                    writer.WritePropertyName(nameof(response.alteredFields));
                    writer.WriteStartArray();
                    foreach (var alteredField in response.alteredFields) {
                        WriteSField(writer, alteredField);
                    }
                    writer.WriteEndArray();
                }
                
                if (response.addedFieldInitializerFields != null) {
                    writer.WritePropertyName(nameof(response.addedFieldInitializerFields));
                    writer.WriteStartArray();
                    foreach (var addedFieldInitializerField in response.addedFieldInitializerFields) {
                        WriteSField(writer, addedFieldInitializerField);
                    }
                    writer.WriteEndArray();
                }

                if (response.addedFieldInitializerInitializers != null) {
                    writer.WritePropertyName(nameof(response.addedFieldInitializerInitializers));
                    writer.WriteStartArray();
                    foreach (var addedFieldInitializerInitializer in response.addedFieldInitializerInitializers) {
                        WriteSMethod(writer, addedFieldInitializerInitializer);
                    }
                    writer.WriteEndArray();
                }
                
                if (response.removedFieldInitializers != null) {
                    writer.WritePropertyName(nameof(response.removedFieldInitializers));
                    writer.WriteStartArray();
                    foreach (var removedFieldInitializer in response.removedFieldInitializers) {
                        WriteSField(writer, removedFieldInitializer);
                    }
                    writer.WriteEndArray();
                }
                
                if (response.newFieldDefinitions != null) {
                    writer.WritePropertyName(nameof(response.newFieldDefinitions));
                    writer.WriteStartArray();
                    foreach (var newFieldDefinition in response.newFieldDefinitions) {
                        WriteSField(writer, newFieldDefinition);
                    }
                    writer.WriteEndArray();
                }
                
                writer.WriteEndObject();
            }
            writer.WriteEndArray();
        }
        
        void WriteSMethod(JsonWriter writer, SMethod method) {
            writer.WriteStartObject();
            
            writer.WritePropertyName(nameof(method.assemblyName));
            writer.WriteValue(method.assemblyName);
            writer.WritePropertyName(nameof(method.displayName));
            writer.WriteValue(method.displayName);
            writer.WritePropertyName(nameof(method.metadataToken));
            writer.WriteValue(method.metadataToken);
            writer.WritePropertyName(nameof(method.simpleName));
            writer.WriteValue(method.simpleName);
            
            writer.WriteEndObject();
        }
        
        void WriteSField(JsonWriter writer, SField field) {
            writer.WriteStartObject();
            
            writer.WritePropertyName(nameof(field.declaringType));
            writer.WriteSType(field.declaringType);
            writer.WritePropertyName(nameof(field.fieldName));
            writer.WriteValue(field.fieldName);
            writer.WritePropertyName(nameof(field.assemblyName));
            writer.WriteValue(field.assemblyName);
            writer.WritePropertyName(nameof(field.metadataToken));
            writer.WriteValue(field.metadataToken);
            writer.WritePropertyName(nameof(field.serializable));
            writer.WriteValue(field.serializable);
            
            writer.WriteEndObject();
        }

    }
    internal static class MethodPatchResponsesConverterExtensions {
        public static void WriteSType(this JsonWriter writer, SType type) {
            if (type == null) {
                writer.WriteNull();
                return;
            }
            writer.WriteStartObject();
            
            writer.WritePropertyName(nameof(type.assemblyName));
            writer.WriteValue(type.assemblyName);
            writer.WritePropertyName(nameof(type.typeName));
            writer.WriteValue(type.typeName);
            writer.WritePropertyName(nameof(type.metadataToken));
            writer.WriteValue(type.metadataToken);
            
            writer.WriteEndObject();
        }
    }

}
