2023-02-09 22:17:49 +07:00
/ *
* SPDX - License - Identifier : AGPL - 3.0 - only
2024-01-04 16:28:16 +07:00
* Copyright ( c ) 2022 - 2024 , daeuniverse Organization < dae @ v2raya . org >
2023-02-09 22:17:49 +07:00
* /
package config
import (
"errors"
"fmt"
"io"
"os"
"path/filepath"
"strings"
2023-04-23 12:27:29 +07:00
"github.com/daeuniverse/dae/common"
"github.com/daeuniverse/dae/pkg/config_parser"
2023-02-09 22:17:49 +07:00
)
var (
2023-11-15 13:32:57 +07:00
ErrCircularInclude = fmt . Errorf ( "circular include is not allowed" )
2023-02-09 22:17:49 +07:00
)
type Merger struct {
entry string
entryDir string
entryToSectionMap map [ string ] map [ string ] [ ] * config_parser . Item
}
func NewMerger ( entry string ) * Merger {
return & Merger {
entry : entry ,
entryDir : filepath . Dir ( entry ) ,
entryToSectionMap : map [ string ] map [ string ] [ ] * config_parser . Item { } ,
}
}
func ( m * Merger ) Merge ( ) ( sections [ ] * config_parser . Section , entries [ ] string , err error ) {
err = m . dfsMerge ( m . entry , "" )
if err != nil {
return nil , nil , err
}
entries , err = common . MapKeys ( m . entryToSectionMap )
if err != nil {
return nil , nil , err
}
return m . convertMapToSections ( m . entryToSectionMap [ m . entry ] ) , entries , nil
}
func ( m * Merger ) readEntry ( entry string ) ( err error ) {
// Check circular include.
_ , exist := m . entryToSectionMap [ entry ]
if exist {
2023-11-15 13:32:57 +07:00
return ErrCircularInclude
2023-02-09 22:17:49 +07:00
}
// Check filename
if ! strings . HasSuffix ( entry , ".dae" ) {
return fmt . Errorf ( "invalid config filename %v: must has suffix .dae" , entry )
}
// Check file path security.
if err = common . EnsureFileInSubDir ( entry , m . entryDir ) ; err != nil {
return fmt . Errorf ( "failed in checking path of config file %v: %w" , entry , err )
}
f , err := os . Open ( entry )
if err != nil {
return fmt . Errorf ( "failed to read config file %v: %w" , entry , err )
}
// Check file access.
fi , err := f . Stat ( )
if err != nil {
return err
}
if fi . IsDir ( ) {
return fmt . Errorf ( "cannot include a directory: %v" , entry )
}
if fi . Mode ( ) & 0037 > 0 {
return fmt . Errorf ( "permissions %04o for '%v' are too open; requires the file is NOT writable by the same group and NOT accessible by others; suggest 0640 or 0600" , fi . Mode ( ) & 0777 , entry )
}
// Read and parse.
b , err := io . ReadAll ( f )
if err != nil {
return err
}
entrySections , err := config_parser . Parse ( string ( b ) )
if err != nil {
return fmt . Errorf ( "failed to parse config file %v:\n%w" , entry , err )
}
m . entryToSectionMap [ entry ] = m . convertSectionsToMap ( entrySections )
return nil
}
func unsqueezeEntries ( patternEntries [ ] string ) ( unsqueezed [ ] string , err error ) {
unsqueezed = make ( [ ] string , 0 , len ( patternEntries ) )
for _ , pattern := range patternEntries {
files , err := filepath . Glob ( pattern )
if err != nil {
return nil , err
}
for _ , file := range files {
// We only support .dae
if ! strings . HasSuffix ( file , ".dae" ) {
continue
}
fi , err := os . Stat ( file )
if err != nil {
return nil , err
}
if fi . IsDir ( ) {
continue
}
unsqueezed = append ( unsqueezed , file )
}
}
if len ( unsqueezed ) == 0 {
unsqueezed = nil
}
return unsqueezed , nil
}
func ( m * Merger ) dfsMerge ( entry string , fatherEntry string ) ( err error ) {
// Read entry and check circular include.
if err = m . readEntry ( entry ) ; err != nil {
2023-11-15 13:32:57 +07:00
if errors . Is ( err , ErrCircularInclude ) {
2023-02-09 22:17:49 +07:00
return fmt . Errorf ( "%w: %v -> %v -> ... -> %v" , err , fatherEntry , entry , fatherEntry )
}
return err
}
sectionMap := m . entryToSectionMap [ entry ]
// Extract childEntries.
includes := sectionMap [ "include" ]
var patterEntries = make ( [ ] string , 0 , len ( includes ) )
for _ , include := range includes {
switch v := include . Value . ( type ) {
case * config_parser . Param :
2023-02-25 21:53:18 +07:00
nextEntry := v . String ( true , false )
2023-02-09 22:17:49 +07:00
patterEntries = append ( patterEntries , filepath . Join ( m . entryDir , nextEntry ) )
default :
2023-02-25 21:53:18 +07:00
return fmt . Errorf ( "unsupported include grammar in %v: %v" , entry , include . String ( false , false ) )
2023-02-09 22:17:49 +07:00
}
}
// DFS and merge children recursively.
childEntries , err := unsqueezeEntries ( patterEntries )
if err != nil {
return err
}
for _ , nextEntry := range childEntries {
if err = m . dfsMerge ( nextEntry , entry ) ; err != nil {
return err
}
}
/// Merge into father. Do not need to retrieve sectionMap again because go map is a reference.
if fatherEntry == "" {
// We are already on the top.
return nil
}
fatherSectionMap := m . entryToSectionMap [ fatherEntry ]
for sec := range sectionMap {
items := m . mergeItems ( fatherSectionMap [ sec ] , sectionMap [ sec ] )
fatherSectionMap [ sec ] = items
}
return nil
}
func ( m * Merger ) convertSectionsToMap ( sections [ ] * config_parser . Section ) ( sectionMap map [ string ] [ ] * config_parser . Item ) {
sectionMap = make ( map [ string ] [ ] * config_parser . Item )
for _ , sec := range sections {
items , ok := sectionMap [ sec . Name ]
if ok {
sectionMap [ sec . Name ] = m . mergeItems ( items , sec . Items )
} else {
sectionMap [ sec . Name ] = sec . Items
}
}
return sectionMap
}
func ( m * Merger ) convertMapToSections ( sectionMap map [ string ] [ ] * config_parser . Item ) ( sections [ ] * config_parser . Section ) {
sections = make ( [ ] * config_parser . Section , 0 , len ( sectionMap ) )
for name , items := range sectionMap {
sections = append ( sections , & config_parser . Section {
Name : name ,
Items : items ,
} )
}
return sections
}
func ( m * Merger ) mergeItems ( to , from [ ] * config_parser . Item ) ( items [ ] * config_parser . Item ) {
items = make ( [ ] * config_parser . Item , len ( to ) + len ( from ) )
copy ( items , to )
copy ( items [ len ( to ) : ] , from )
return items
}