Paarthurnax

A Golang translator

Utilisation

Init

Crée les fichiers nécessaire pour traquer les segments dans le temps

Normalize

Normalise les fichiers de traduction afin de réduire le bruit dans les reviews futurs

Translate

Traduit dans toutes les langues définit via DeepL

Init

$ paarthurnax init

2024/09/23 15:44:38 Generating state from disk...
2024/09/23 15:44:38 Processing config/locales/a_new_directory/fr.yml
2024/09/23 15:44:38 Processing config/locales/component/fr.yml
2024/09/23 15:44:38 Processing config/locales/fr.yml
2024/09/23 15:44:38 Persisting state...
  • Charge les fichiers de traduction source
  • Génere un Hash pour chaque clé basée sur son contenu 
  • Sauvegarde un fichier d'état .paarthurnax

Init

Quand?

Une seule fois a l'installation de Paarthurnax dans le projet

Prérequis?

Clean state (pas de traductions en attente)

Normalize

$ paarthurnax normalize

2024/09/23 15:59:11 Loading current state from disk...
2024/09/23 15:59:11 Normalizing translations...
2024/09/23 15:59:11 Processing config/locales/a_new_directory/fr.yml...
2024/09/23 15:59:11 Processing config/locales/component/fr.yml...
2024/09/23 15:59:11 Processing config/locales/fr.yml...
2024/09/23 15:59:11 Done!
  • Charge les fichiers de destination de traduction
  • Les réécrit directement avec le même système que lors de prochaines traductions

normalize

Quand?

A n'importe quel instant

Prérequis?

Aucun

translate

$ DEEPL_API_KEY=[...] paarthurnax translate

2024/09/23 16:00:41 Loading previous state from disk...
2024/09/23 16:00:41 Generating current state from disk...
2024/09/23 16:00:41 Reconciling states...
2024/09/23 16:00:41 Processing config/locales/a_new_directory/fr.yml
2024/09/23 16:00:41 Processing config/locales/component/fr.yml
2024/09/23 16:00:41 Processing config/locales/fr.yml
2024/09/23 16:00:41 Cleaning up removed files...
2024/09/23 16:00:41 Persisting new state...
  • Charge les fichiers de traduction source
  • Charge l'état précédent
  • Traduit touts les segments modifiés avec DeepL
  • Réécrit tous les fichiers de traduction cible modifiés

Init

Quand?

A chaque étape de traductio

Prérequis?

  • Présence du fichier .paarthurnax
  • Présence de la clé d'api DeepL FREE en variable d'env

Fonctionnement

// Variables
var i string
i := 42

// Fonctions
func (reciver File) Name(param string) (string, error) {
  [...]
}

// Types
type File struct {
  I int
}

Fonctionnement

type State struct {
	Files []translationFile.TranslationFile
}

type TranslationFile struct {
	Path           string
	SegmentsHashes map[string]string // Key: sha1
}
Chargement du state depuis les locales

Fonctionnement

Chargement du state depuis les locales
e := filepath.WalkDir(path, func(path string, d fs.DirEntry, err error) error {
		if err == nil && d.Name() == "fr.yml" {
			if verbose {
				log.Println(fmt.Sprintf("Processing %s", path))
			}

			t, err := translation.Load(path)
			if err != nil {
				return errors.New(fmt.Sprintf("Failed to load translation: %s", err))
			}

			state.Files = append(state.Files, translationFile.TranslationFile{
				Path:           path,
				SegmentsHashes: HashTranslation(t.FlattenedSegments()),
			})
		}
		return err
	})
	if e != nil {
		return nil, errors.New("Unable to process the filestructure: " + e.Error())
	}

Fonctionnement

func Load(path string) (*TranslationFile, error) {
	data, err := os.ReadFile(path)
	if err != nil {
		return nil, errors.New("Error reading file:" + err.Error())
	}

	var yamlData map[string]interface{}
	err = yaml.Unmarshal(data, &yamlData)
	if err != nil {
		return nil, errors.New("Error unmarshaling YAML:" + err.Error())
	}

	if len(yamlData) != 1 {
		return nil, errors.New("the provided YAML file is not a valid translation file")
	}

	locale := utils.MapKeys(yamlData)[0]
	file := TranslationFile{Path: path, Locale: locale, Segments: yamlData[locale].(map[string]interface{})}

	return &file, nil
}

Fonctionnement

Chargement du state depuis les locales
e := filepath.WalkDir(path, func(path string, d fs.DirEntry, err error) error {
		if err == nil && d.Name() == "fr.yml" {
			if verbose {
				log.Println(fmt.Sprintf("Processing %s", path))
			}

			t, err := translation.Load(path)
			if err != nil {
				return errors.New(fmt.Sprintf("Failed to load translation: %s", err))
			}

			state.Files = append(state.Files, translationFile.TranslationFile{
				Path:           path,
				SegmentsHashes: HashTranslation(t.FlattenedSegments()),
			})
		}
		return err
	})
	if e != nil {
		return nil, errors.New("Unable to process the filestructure: " + e.Error())
	}

Fonctionnement

Chargement du state depuis les locales
func (translation *TranslationFile) FlattenedSegments() map[string]string {
	return extractFlattenedSegments(translation.Segments)
}

func extractFlattenedSegments(m map[string]interface{}) map[string]string {
	result := make(map[string]string)

	for key, value := range m {
		switch value.(type) {
		case string:
			result[key] = value.(string)
		case map[string]interface{}:
			for subKey, subValue := range extractFlattenedSegments(value.(map[string]interface{})) {
				result[key+"."+subKey] = subValue
			}
		}
	}

	return result
}

Fonctionnement

Chargement du state depuis les locales
func HashTranslation(m map[string]string) map[string]string {
	result := make(map[string]string)

	for k, v := range m {
		h := sha1.New()
		h.Write([]byte(v))
		result[k] = hex.EncodeToString(h.Sum(nil))
	}

	return result
}

Fonctionnement

Chargement du state depuis les locales

Fonctionnement

Chargement du state depuis les locales
[[Files]]
Path = 'config/locales/a_new_directory/fr.yml'

[Files.SegmentsHashes]
a_key = '2a6dcab3038a7c1d15b427378a3483952761df8c'

[[Files]]
Path = 'config/locales/component/fr.yml'

[Files.SegmentsHashes]
'a_super.deep.key' = 'e50fc81a49f34d67292cdfd88a6e1737be1677a6'

[[Files]]
Path = 'config/locales/fr.yml'

[Files.SegmentsHashes]
'a.deep.b' = '6dcd4ce23d88e2ee9568ba546c007c63d9131c1b'
first_level_key = '153d6addf72e832f95dd72ba7f891ccb47d6d4d2'
likes = '06cee428e9b518ed6181eb637a6a6175fa8d725f'
remote_status = 'ca4830751faf5e537a84eb297e53441a599c7681'
'second_level.plural_key.one' = '137eca41db3d9cb9591ad89ceb5e78379017b67f'
'second_level.plural_key.other' = 'a1ba1d821b7e573d90064bcfe2fd080ff625a0ad'
'second_level.plural_key.zero' = 'ddcc9ab409b7ba7733d96f07e981dda1102b21a6'
'second_level.second_level_key' = '6f9588ba7a1bb1d5a7c987cc9c1211fe81f7a15f'

Fonctionnement

Normalisation des traductions
log.Println("Loading current state from disk...")
nState, err := state.LoadFromDisk("config/locales", false)
if err != nil {
    log.Fatal(err)
}

log.Println("Normalizing translations...")
for _, nFile := range nState.Files {
	log.Println(fmt.Sprintf("Processing %s...", nFile.Path))

	for _, locale := range translationGroup.DestLocales {
		path := strings.Replace(nFile.Path, "fr.yml", locale+".yml", 1)
		file, err := translation.LoadOrCreate(path)
		if err != nil {
			log.Fatal(fmt.Sprintf("%s: %s", path, err.Error()))
		}
		if err = file.Save(); err != nil {
			log.Fatal(err)
		}
	}
}

Fonctionnement

Traduction
log.Println("Loading previous state from disk...")
pState, err := state.Load()
if err != nil {
	log.Fatal(err)
}

log.Println("Generating current state from disk...")
nState, err := state.LoadFromDisk("config/locales", false)
if err != nil {
	log.Fatal(err)
}

Fonctionnement

Traduction
log.Println("Reconciling states...")
for _, nFile := range nState.Files {
	log.Println(fmt.Sprintf("Processing %s", nFile.Path))
	changes := nFile.Changes(pState.GetFile(nFile.Path))
	if len(changes) != 0 {
		log.Println(fmt.Sprintf("Applying changes and translating %s...", nFile.Path))
		group, err := translationGroup.New(nFile.Path)
		if err != nil {
			log.Fatal(err)
		}
		if err = group.Apply(changes); err != nil {
			log.Fatal(err)
		}
	}
}

Fonctionnement

Traduction
const (
	Added   = 0
	Removed = 1
	Updated = 2
)

type Change struct {
	Kind int8
	Path string
}

func (file *TranslationFile) Changes(other *TranslationFile) []Change {
	var changes []Change

	for path, hash := range file.SegmentsHashes {
		if other == nil { // When the other translation state does not exist (new source file)
			changes = append(changes, Change{Kind: Added, Path: path})
		} else {
			oHash, ok := other.SegmentsHashes[path]
			if !ok {
				changes = append(changes, Change{Kind: Added, Path: path})
			} else if hash != oHash {
				changes = append(changes, Change{Kind: Updated, Path: path})
			}
		}
	}

	if other != nil {
		for path, _ := range other.SegmentsHashes {
			if _, ok := file.SegmentsHashes[path]; !ok {
				changes = append(changes, Change{Kind: Removed, Path: path})
			}
		}
	}

	return changes
}

Fonctionnement

Traduction
log.Println("Reconciling states...")
for _, nFile := range nState.Files {
	log.Println(fmt.Sprintf("Processing %s", nFile.Path))
	changes := nFile.Changes(pState.GetFile(nFile.Path))
	if len(changes) != 0 {
		log.Println(fmt.Sprintf("Applying changes and translating %s...", nFile.Path))
		group, err := translationGroup.New(nFile.Path)
		if err != nil {
			log.Fatal(err)
		}
		if err = group.Apply(changes); err != nil {
			log.Fatal(err)
		}
	}
}

var DestLocales = [...]string{"en", "es", "it", "de", "hu", "pt", "pl", "ro", "uk"}

type TranslationGroup struct {
	Path   string
	source *translation.TranslationFile
	files  []*translation.TranslationFile
}

Fonctionnement

Traduction
func (group *TranslationGroup) Apply(changes []translationFile.Change) error {
	translator, err := deepl.New(os.Getenv("DEEPL_API_KEY"), "api-free.deepl.com")
	if err != nil {
		return errors.New("Unable to initialize the translation engine: " + err.Error())
	}
	for _, change := range changes {
		keyParts := strings.Split(change.Path, ".")
		if utils.Includes(keyParts[len(keyParts)-1], []string{"zero", "one", "other"}) {
			if err := handlePluralSegment(keyParts[len(keyParts)-1], group, &change, translator); err != nil {
				return err
			}
		} else {
			if err := handleStandaloneSegment(group, &change, translator); err != nil {
				return err
			}
		}
	}

	for _, file := range group.files {
		if err := file.Save(); err != nil {
			return errors.New("Failed to save updated translation file " + file.Path + ": " + err.Error())
		}
	}

	return nil
}

Fonctionnement

Traduction
func handleStandaloneSegment(group *TranslationGroup, change *translationFile.Change, translator *deepl.Deepl) error {
	for _, file := range group.files {
		if change.Kind == translationFile.Added || change.Kind == translationFile.Updated {
			value, err := group.source.GetSegmentValueAt(change.Path)
			if err != nil {
				return errors.New("Unable to get the value of the source segment " + change.Path + ": " + err.Error())
			}
			valueForTranslation, ctx, err := prepareForTranslation(value)
			if err != nil {
				return errors.New("Unable to prepare for translation " + change.Path + ": " + err.Error())
			}
			translation, err := translator.Translate(valueForTranslation, "fr", file.Locale)
			if err != nil {
				return errors.New("Unable to translate the value of the source segment " + change.Path + ": " + err.Error())
			}
			translation, err = revertTranslationPreparation(translation, ctx)
			if err != nil {
				return errors.New("Unable to revert translation preparation for " + change.Path + ": " + err.Error())
			}

			if err = checkVariableEquity(value, translation); err != nil {
				return errors.New("The translation does not contain the required variables: " + err.Error())
			}

			if err = file.SetSegmentValueAt(change.Path, translation); err != nil {
				return errors.New("Unable to set the value of the source segment " + change.Path + " in locale " + file.Locale + ": " + err.Error())
			}
		} else if change.Kind == translationFile.Removed {
			if err := file.RemoveSegmentAt(change.Path); err != nil {
				return errors.New("Unable to remove segment " + change.Path + " in locale " + file.Locale + ": " + err.Error())
			}
		}
	}
	return nil
}

Fonctionnement

Traduction
func prepareForTranslation(text string) (string, map[string]string, error) {
	ctx := make(map[string]string)
	r, err := regexp.Compile("%{(?P<variable>[a-zA-Z_ -]+)}")
	if err != nil {
		return "", nil, errors.New("Unable to prepare variable extraction: " + err.Error())
	}
	for _, variable := range r.FindAllStringSubmatch(text, -1) {
		ctx[variable[1]] = "<span translate=\"no\">" + variable[1] + "</span>"
		text = strings.Replace(text, "%{"+variable[1]+"}", ctx[variable[1]], -1)
	}
	return text, ctx, nil
}

Fonctionnement

Traduction
func handleStandaloneSegment(group *TranslationGroup, change *translationFile.Change, translator *deepl.Deepl) error {
	for _, file := range group.files {
		if change.Kind == translationFile.Added || change.Kind == translationFile.Updated {
			value, err := group.source.GetSegmentValueAt(change.Path)
			if err != nil {
				return errors.New("Unable to get the value of the source segment " + change.Path + ": " + err.Error())
			}
			valueForTranslation, ctx, err := prepareForTranslation(value)
			if err != nil {
				return errors.New("Unable to prepare for translation " + change.Path + ": " + err.Error())
			}
			translation, err := translator.Translate(valueForTranslation, "fr", file.Locale)
			if err != nil {
				return errors.New("Unable to translate the value of the source segment " + change.Path + ": " + err.Error())
			}
			translation, err = revertTranslationPreparation(translation, ctx)
			if err != nil {
				return errors.New("Unable to revert translation preparation for " + change.Path + ": " + err.Error())
			}

			if err = checkVariableEquity(value, translation); err != nil {
				return errors.New("The translation does not contain the required variables: " + err.Error())
			}

			if err = file.SetSegmentValueAt(change.Path, translation); err != nil {
				return errors.New("Unable to set the value of the source segment " + change.Path + " in locale " + file.Locale + ": " + err.Error())
			}
		} else if change.Kind == translationFile.Removed {
			if err := file.RemoveSegmentAt(change.Path); err != nil {
				return errors.New("Unable to remove segment " + change.Path + " in locale " + file.Locale + ": " + err.Error())
			}
		}
	}
	return nil
}

Fonctionnement

Traduction

func revertTranslationPreparation(text string, ctx map[string]string) (string, error) {
	for value, placeholder := range ctx {
		text = strings.Replace(text, placeholder, "%{"+value+"}", -1)
	}
	return text, nil
}

Fonctionnement

Traduction
func handleStandaloneSegment(group *TranslationGroup, change *translationFile.Change, translator *deepl.Deepl) error {
	for _, file := range group.files {
		if change.Kind == translationFile.Added || change.Kind == translationFile.Updated {
			value, err := group.source.GetSegmentValueAt(change.Path)
			if err != nil {
				return errors.New("Unable to get the value of the source segment " + change.Path + ": " + err.Error())
			}
			valueForTranslation, ctx, err := prepareForTranslation(value)
			if err != nil {
				return errors.New("Unable to prepare for translation " + change.Path + ": " + err.Error())
			}
			translation, err := translator.Translate(valueForTranslation, "fr", file.Locale)
			if err != nil {
				return errors.New("Unable to translate the value of the source segment " + change.Path + ": " + err.Error())
			}
			translation, err = revertTranslationPreparation(translation, ctx)
			if err != nil {
				return errors.New("Unable to revert translation preparation for " + change.Path + ": " + err.Error())
			}

			if err = checkVariableEquity(value, translation); err != nil {
				return errors.New("The translation does not contain the required variables: " + err.Error())
			}

			if err = file.SetSegmentValueAt(change.Path, translation); err != nil {
				return errors.New("Unable to set the value of the source segment " + change.Path + " in locale " + file.Locale + ": " + err.Error())
			}
		} else if change.Kind == translationFile.Removed {
			if err := file.RemoveSegmentAt(change.Path); err != nil {
				return errors.New("Unable to remove segment " + change.Path + " in locale " + file.Locale + ": " + err.Error())
			}
		}
	}
	return nil
}

Fonctionnement

Traduction
func checkVariableEquity(a string, b string) error {
	r, err := regexp.Compile("(?P<variable>%{[a-zA-Z_ -]+})")
	if err != nil {
		return errors.New("Unable to prepare variable check: " + err.Error())
	}
	aResults := r.FindAllStringSubmatch(a, -1)

	var bResults []string
	for _, match := range r.FindAllStringSubmatch(b, -1) {
		bResults = append(bResults, match[1])
	}

	for _, variable := range aResults {
		if !utils.Includes(variable[1], bResults) {
			return errors.New(fmt.Sprintf("'%s' is not present in '%s'", variable[1], b))
		}
	}
	return nil
}

Fonctionnement

Traduction
func handleStandaloneSegment(group *TranslationGroup, change *translationFile.Change, translator *deepl.Deepl) error {
	for _, file := range group.files {
		if change.Kind == translationFile.Added || change.Kind == translationFile.Updated {
			value, err := group.source.GetSegmentValueAt(change.Path)
			if err != nil {
				return errors.New("Unable to get the value of the source segment " + change.Path + ": " + err.Error())
			}
			valueForTranslation, ctx, err := prepareForTranslation(value)
			if err != nil {
				return errors.New("Unable to prepare for translation " + change.Path + ": " + err.Error())
			}
			translation, err := translator.Translate(valueForTranslation, "fr", file.Locale)
			if err != nil {
				return errors.New("Unable to translate the value of the source segment " + change.Path + ": " + err.Error())
			}
			translation, err = revertTranslationPreparation(translation, ctx)
			if err != nil {
				return errors.New("Unable to revert translation preparation for " + change.Path + ": " + err.Error())
			}

			if err = checkVariableEquity(value, translation); err != nil {
				return errors.New("The translation does not contain the required variables: " + err.Error())
			}

			if err = file.SetSegmentValueAt(change.Path, translation); err != nil {
				return errors.New("Unable to set the value of the source segment " + change.Path + " in locale " + file.Locale + ": " + err.Error())
			}
		} else if change.Kind == translationFile.Removed {
			if err := file.RemoveSegmentAt(change.Path); err != nil {
				return errors.New("Unable to remove segment " + change.Path + " in locale " + file.Locale + ": " + err.Error())
			}
		}
	}
	return nil
}

Fonctionnement

Traduction
func handlePluralSegment(part string, group *TranslationGroup, change *translationFile.Change, translator *deepl.Deepl) error {
	for _, file := range group.files {
		affectedKeys := determineAffectedKeysIn(part, file.Locale)
		for _, definition := range affectedKeys {
			pathParts := strings.Split(change.Path, ".")
			localKey := strings.Join(append(pathParts[:len(pathParts)-1], definition.key), ".")

			if change.Kind == translationFile.Added || change.Kind == translationFile.Updated {
				value, err := group.source.GetSegmentValueAt(change.Path)
				if err != nil {
					return errors.New("Unable to get the value of the source segment " + change.Path + ": " + err.Error())
				}

				localValue := strings.ReplaceAll(value, "%{count}", strconv.Itoa(int(definition.tip)))
				valueForTranslation, ctx, err := prepareForTranslation(localValue)
				if err != nil {
					return errors.New("Unable to prepare for translation " + change.Path + ": " + err.Error())
				}
				translation, err := translator.Translate(valueForTranslation, "fr", file.Locale)
				if err != nil {
					return errors.New("Unable to translate the value of the source segment " + change.Path + ": " + err.Error())
				}
				translation, err = revertTranslationPreparation(translation, ctx)
				if err != nil {
					return errors.New("Unable to revert translation preparation for " + change.Path + ": " + err.Error())
				}
				translation = strings.ReplaceAll(translation, strconv.Itoa(int(definition.tip)), "%{count}")

				if err = checkVariableEquity(value, translation); err != nil {
					return errors.New("The translation does not contain the required variables: " + err.Error())
				}

				if err = file.SetSegmentValueAt(localKey, translation); err != nil {
					return errors.New("Unable to set the value of the source segment " + change.Path + " in locale " + file.Locale + ": " + err.Error())
				}
			} else if change.Kind == translationFile.Removed {
				if err := file.RemoveSegmentAt(localKey); err != nil {
					return errors.New("Unable to remove segment " + localKey + " in locale " + file.Locale + ": " + err.Error())
				}
			}
		}
	}
	return nil
}

Fonctionnement

Traduction
func determineAffectedKeysIn(part string, locale string) []pluralDefinition {
	keysDirections := map[string]map[string][]pluralDefinition{
		"zero": {
			"en": {
				{
					key: "zero",
					tip: 0,
				},
			},
        },
        "one": {
            "en": {
                key: "one",
                tip: 1
            }
        }
    }
    return keysDirections[part][locale]
}

Fonctionnement

Traduction
func handlePluralSegment(part string, group *TranslationGroup, change *translationFile.Change, translator *deepl.Deepl) error {
	for _, file := range group.files {
		affectedKeys := determineAffectedKeysIn(part, file.Locale)
		for _, definition := range affectedKeys {
			pathParts := strings.Split(change.Path, ".")
			localKey := strings.Join(append(pathParts[:len(pathParts)-1], definition.key), ".")

			if change.Kind == translationFile.Added || change.Kind == translationFile.Updated {
				value, err := group.source.GetSegmentValueAt(change.Path)
				if err != nil {
					return errors.New("Unable to get the value of the source segment " + change.Path + ": " + err.Error())
				}

				localValue := strings.ReplaceAll(value, "%{count}", strconv.Itoa(int(definition.tip)))
				valueForTranslation, ctx, err := prepareForTranslation(localValue)
				if err != nil {
					return errors.New("Unable to prepare for translation " + change.Path + ": " + err.Error())
				}
				translation, err := translator.Translate(valueForTranslation, "fr", file.Locale)
				if err != nil {
					return errors.New("Unable to translate the value of the source segment " + change.Path + ": " + err.Error())
				}
				translation, err = revertTranslationPreparation(translation, ctx)
				if err != nil {
					return errors.New("Unable to revert translation preparation for " + change.Path + ": " + err.Error())
				}
				translation = strings.ReplaceAll(translation, strconv.Itoa(int(definition.tip)), "%{count}")

				if err = checkVariableEquity(value, translation); err != nil {
					return errors.New("The translation does not contain the required variables: " + err.Error())
				}

				if err = file.SetSegmentValueAt(localKey, translation); err != nil {
					return errors.New("Unable to set the value of the source segment " + change.Path + " in locale " + file.Locale + ": " + err.Error())
				}
			} else if change.Kind == translationFile.Removed {
				if err := file.RemoveSegmentAt(localKey); err != nil {
					return errors.New("Unable to remove segment " + localKey + " in locale " + file.Locale + ": " + err.Error())
				}
			}
		}
	}
	return nil
}

Quoi retenir?

$ paarthurnax -h

A simple and quick translation tool for in place
translation of Rails project

Usage:
  Paarthurnax [flags]
  Paarthurnax [command]

Available Commands:
  completion  Generate the autocompletion script for the specified shell
  help        Help about any command
  init        Initialize the repository
  normalize   Normalize the repository
  translate   Translate the repository

Flags:
  -h, --help   help for Paarthurnax

Use "Paarthurnax [command] --help" for more information about a command.

Code source

Paarthurnax

By foret_a

Paarthurnax

  • 31