Search code examples
goreflectiongo-reflect

How to construct slice of structs dynamically using reflection


I was trying to construct slice of Book structs with pointers but I was unable to get it work with reflection in Go.

[]*Book slice of Book struct pointers , please note that scanResults method might receive any type of slice and not only Book struct. So am looking to build a slice at runtime dynamically

Can you please let me know what I was getting wrong in the below code snippet?

package main

import (
  "reflect"
"errors"
"fmt"
)

type Book struct {
    Id    int
    Title string
    Price float32
}

func main() {
    var dest []*Book
    scanResults(&dest)
}

func scanResults(dest interface{}) error{
   resultsFromExternalSource := []interface{}{10 , "user-name" , float32(22)}

    value := reflect.ValueOf(dest)
    if value.Kind() != reflect.Ptr {
        return errors.New("must pass a pointer, not a value, to scan results into struct destination")
    }

    sliceElement := reflect.TypeOf(dest).Elem()
    if sliceElement.Kind() != reflect.Slice {
        return fmt.Errorf("expected %s but got %s", reflect.Slice, sliceElement.Kind())
    }

    structPtr := sliceElement.Elem()
    if structPtr.Kind() != reflect.Ptr {
        return fmt.Errorf("expected %s but got %s", reflect.Ptr, structPtr.Kind())
    }

    structElemType := reflect.TypeOf(structPtr).Elem()
    if structElemType.Kind() != reflect.Struct {
        return fmt.Errorf("expected %s but got %s", reflect.Struct, structElemType.Kind())
    }

 structRecordInterface := reflect.New(structElemType).Elem().Interface() // create a new struct
            structRecordType := reflect.TypeOf(structRecordInterface)
            structRecordValue := reflect.ValueOf(structRecordType)

    for i, result := range resultsFromExternalSource {



                if structRecordValue.Field(i).CanSet() {
                    structRecordValue.Field(i).Set(reflect.ValueOf(result))
                } else {
                    varName := structRecordType.Field(i).Name
                    varType := structRecordType.Field(i).Type
                    return fmt.Errorf("cannot scan results into passed struct destination as the struct field %v with %v type is not settable", varName, varType)
                }
       }
     return nil

}

https://play.golang.org/p/O9j4RobQqMy


Solution

  • You are almost there. Here's some working code with commentary:

    var errBadArg = errors.New("must pass pointer to slice of pointer to struct")
    
    func scanResults(dest interface{}) error {
        resultsFromExternalSource := [][]interface{}{
            {10, "user-name", float32(22)},
            {20, "i-love-reflect", float32(100)},
        }
    
        // Get reflect.Value for the destination confirm that
        // the destination is a pointer to a slice of pointers
        // to a struct. The tests can be omitted if it's acceptable
        // to panic on bad input argument.
    
        destv := reflect.ValueOf(dest)
    
        if destv.Kind() != reflect.Ptr {
            return errBadArg
        }
    
        // Deference the pointer to get the slice.
        destv = destv.Elem()
        if destv.Kind() != reflect.Slice {
            return errBadArg
        }
    
        elemt := destv.Type().Elem()
        if elemt.Kind() != reflect.Ptr {
            return errBadArg
        }
    
        // "deference" the element type to get the struct type.
        elemt = elemt.Elem() 
        if elemt.Kind() != reflect.Struct {
            return errBadArg
        }
    
    
        // For each row in the result set...
        for j, row := range resultsFromExternalSource {
    
            // Return error if more columns than fields in struct.
            if len(row) > elemt.NumField() {
                return errors.New("result larger than struct")
            }
    
            // Allocate a new slice element.
            elemp := reflect.New(elemt)
    
            // Dereference the pointer for field access.
            elemv := elemp.Elem()
    
            for i, col := range row {
                fieldv := elemv.Field(i)
                colv := reflect.ValueOf(col)
    
                // Check to see if assignment to field will work
                if !colv.Type().AssignableTo(fieldv.Type()) {
                    return fmt.Errorf("cannot assign %s to %s in row %d column %d", colv.Type(), fieldv.Type(), j, i)
                }
    
                // Set the field.
                fieldv.Set(colv)
            }
    
            // Append element to the slice.
            destv.Set(reflect.Append(destv, elemp))
        }
        return nil
    }
    

    Run it on the playground.