sql >> Databáze >  >> NoSQL >> MongoDB

Spring -Mongodb ukládá/načítá výčty jako int ne řetězec

Po dlouhém vrtání v kódu převodníku jaro-mongodb, OK, skončil jsem a teď to funguje :) tady to je (pokud existuje jednodušší řešení, rád uvidím i to, co jsem udělal) :

nejprve definujte :

public interface IntEnumConvertable {
      public int getValue();    
}

a jednoduchý výčet, který jej implementuje:

public enum tester implements IntEnumConvertable{   
    vali(0),secondvali(1),thirdvali(5);

    private final int val;
    private tester(int num)
    {
        val = num;          
    }
    public int getValue(){
        return val;
    }
}

Dobře, nyní budete potřebovat 2 převodníky, jeden je jednoduchý, druhý je složitější. jednoduchý (toto jednoduché dítě také zpracovává jednoduchý převod a vrací řetězec, když přetypování není možné, to je skvělé, pokud chcete mít enum uložené jako řetězce a pro enum, což jsou čísla, která mají být uložena jako celá čísla):

public class IntegerEnumConverters {
    @WritingConverter
    public static class EnumToIntegerConverter implements Converter<Enum<?>, Object> {
        @Override
        public Object convert(Enum<?> source) {
            if(source instanceof IntEnumConvertable)
            {
                return ((IntEnumConvertable)(source)).getValue();
            }
            else
            {
                return source.name();
            }               
        }
    }   
 }

ta složitější je ve skutečnosti továrna na převodníky:

public class IntegerToEnumConverterFactory implements ConverterFactory<Integer, Enum> {
        @Override
        public <T extends Enum> Converter<Integer, T> getConverter(Class<T> targetType) {
            Class<?> enumType = targetType;
            while (enumType != null && !enumType.isEnum()) {
                enumType = enumType.getSuperclass();
            }
            if (enumType == null) {
                throw new IllegalArgumentException(
                        "The target type " + targetType.getName() + " does not refer to an enum");
            }
            return new IntegerToEnum(enumType);
        }
        @ReadingConverter
        public static class IntegerToEnum<T extends Enum>  implements Converter<Integer, Enum> {
            private final Class<T> enumType;

            public IntegerToEnum(Class<T> enumType) {
                this.enumType = enumType;
            }

            @Override
            public Enum convert(Integer source) {
                  for(T t : enumType.getEnumConstants()) {
                      if(t instanceof IntEnumConvertable)
                      {
                          if(((IntEnumConvertable)t).getValue() == source.intValue()) {
                                return t;
                            }                         
                      }                     
                    }
                    return null;   
            }
        }
}

a teď k té části hacku, osobně jsem nenašel žádný "programově" způsob, jak zaregistrovat továrnu na převodníky v rámci mongoConverter, takže jsem se ponořil do kódu a s malým přehazováním, tady to je (dejte tyto 2 baby funkce do vaší @Configuration třída)

      @Bean
        public CustomConversions customConversions() {
            List<Converter<?, ?>> converters = new ArrayList<Converter<?, ?>>();
            converters.add(new IntegerEnumConverters.EnumToIntegerConverter());     
// this is a dummy registration , actually it's a work-around because
// spring-mongodb doesnt has the option to reg converter factory.
// so we reg the converter that our factory uses. 
converters.add(new IntegerToEnumConverterFactory.IntegerToEnum(null));      
            return new CustomConversions(converters);
        }

    @Bean
    public MappingMongoConverter mappingMongoConverter() throws Exception {
        MongoMappingContext mappingContext = new MongoMappingContext();
        mappingContext.setApplicationContext(appContext);
        DbRefResolver dbRefResolver = new DefaultDbRefResolver(mongoDbFactory());
        MappingMongoConverter mongoConverter = new MappingMongoConverter(dbRefResolver, mappingContext);        
        mongoConverter.setCustomConversions(customConversions());       
        ConversionService convService = mongoConverter.getConversionService();
        ((GenericConversionService)convService).addConverterFactory(new IntegerToEnumConverterFactory());                  
        mongoConverter.afterPropertiesSet();
        return mongoConverter;
    } 


  1. Proč potřebujeme „arbitera“ v replikaci MongoDB?

  2. Použití MongoDB $pull k odstranění dokumentů v poli

  3. MongoDB $weeklyUpdate #66 (22. dubna 2022):Hackathons, mongosh a Github

  4. Jak vytvořím dynamický dotaz pomocí Apache Camel a MongoDB?