private static void AddMissingResource(string resourname, CelestialBody body, Dictionary <string, AtmosphericResource> bodyComposition)
        {
            // verify it is a defined resource
            var definition = PartResourceLibrary.Instance.GetDefinition(resourname);

            if (definition == null)
            {
                Debug.LogWarning("[KSPI] - AddMissingResource : Failed to find resource definition for '" + resourname + "'");
                return;
            }

            // skip it already registred or used as a Synonym
            if (bodyComposition.Values.Any(m => m.ResourceName == definition.name || m.DisplayName == definition.displayName || m.Synonyms.Contains(definition.name)))
            {
                Debug.Log("[KSPI] - AddMissingResource : Already found existing composition for '" + resourname + "'");
                return;
            }

            // retreive abundance
            var abundance = GetAbundance(definition.name, body);

            if (abundance <= 0)
            {
                Debug.LogWarning("[KSPI] - AddMissingResource : Abundance for resource '" + resourname + "' was " + abundance);
                return;
            }

            // create resource from definition and abundance
            var resource = new AtmosphericResource(definition, abundance);

            // add to composition
            Debug.Log("[KSPI] - AddMissingResource : add resource '" + resourname + "'");
            bodyComposition.Add(resource.ResourceName, resource);
        }
        private static void AddResource(int refBody, IDictionary <string, AtmosphericResource> bodyComposition, string outputResourname, string inputResource1, string inputResource2, string inputResource3, string displayname)
        {
            var abundances = new[] { GetAbundance(inputResource1, refBody), GetAbundance(inputResource2, refBody), GetAbundance(inputResource2, refBody) };

            var resource = new AtmosphericResource(outputResourname, abundances.Max(), displayname, new[] { inputResource1, inputResource2, inputResource3 });

            if (resource.ResourceAbundance <= 0)
            {
                return;
            }

            AtmosphericResource existingResource;

            if (bodyComposition.TryGetValue(outputResourname, out existingResource))
            {
                Debug.Log("[KSPI] - replaced resource " + outputResourname + " with stock defined abundance " + resource.ResourceAbundance);
                bodyComposition.Remove(existingResource.ResourceName);
            }
            bodyComposition.Add(resource.ResourceName, resource);
        }
        private static void AddResource(string outputResourname, string displayname, int refBody, IDictionary <string, AtmosphericResource> bodyComposition, string[] variants)
        {
            var abundances = new[] { GetAbundance(outputResourname, refBody) }.Concat(variants.Select(m => GetAbundance(m, refBody)));

            var resource = new AtmosphericResource(outputResourname, abundances.Max(), displayname, variants);

            if (resource.ResourceAbundance <= 0)
            {
                return;
            }

            AtmosphericResource existingResource;

            if (bodyComposition.TryGetValue(outputResourname, out existingResource))
            {
                Debug.Log("[KSPI] - Replaced resource " + outputResourname + " with stock defined abundance " + resource.ResourceAbundance);
                bodyComposition.Remove(existingResource.ResourceName);
            }
            bodyComposition.Add(resource.ResourceName, resource);
        }
        private static void AddResource(string outputResourname, string displayname, CelestialBody body, IDictionary <string, AtmosphericResource> atmosphericResourcesByName, string[] variantNames, double abundanceExponent = 1)
        {
            double finalAbundance;

            AtmosphericResource existingResource = FindAnyExistingAtmosphereVariant(atmosphericResourcesByName, variantNames);

            if (existingResource != null)
            {
                finalAbundance = existingResource.ResourceAbundance;
                Debug.Log("[KSPI] - using kspie resource definition " + outputResourname + " for " + body.name + " with abundance " + finalAbundance);
            }
            else
            {
                var abundances = new[] { GetAbundance(outputResourname, body) }.Concat(variantNames.Select(m => GetAbundance(m, body)));
                finalAbundance = abundances.Max();
                Debug.Log("[KSPI] - looked up stock resource definition " + outputResourname + " for " + body.name + " with abundance " + finalAbundance);

                if (abundanceExponent != 1)
                {
                    finalAbundance = Math.Pow(finalAbundance / 100, abundanceExponent) * 100;
                }
            }

            var resource = new AtmosphericResource(outputResourname, finalAbundance, displayname, variantNames);

            if (resource.ResourceAbundance <= 0)
            {
                return;
            }


            if (atmosphericResourcesByName.TryGetValue(outputResourname, out existingResource))
            {
                atmosphericResourcesByName.Remove(existingResource.ResourceName);
            }
            atmosphericResourcesByName.Add(resource.ResourceName, resource);
        }