123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138 |
- package main
- import (
- "math/rand"
- "sort"
- "unicode"
- )
- func encodeWord(word []rune, wordLength int) {
- if wordLength == 4 {
- word[1], word[2] = word[2], word[1]
- return
- }
- original := string(word)
- toShuffle := word[1 : wordLength-1]
- toShuffleLength := wordLength - 2
- rand.Shuffle(toShuffleLength, func(i, j int) {
- toShuffle[i], toShuffle[j] = toShuffle[j], toShuffle[i]
- })
- if string(word) == original {
- rand.Shuffle(toShuffleLength, func(i, j int) {
- toShuffle[i], toShuffle[j] = toShuffle[j], toShuffle[i]
- })
- }
- }
- //EncodeText returns encoded provided text and sorted array of encoded words
- func EncodeText(text string) EncodedText {
- runes := []rune(text)
- var currentWord []rune
- var newString []rune
- var encodedWords []string
- for _, item := range runes {
- if unicode.IsPunct(item) || unicode.IsSpace(item) {
- currentWordLength := len(currentWord)
- if currentWordLength >= 4 {
- beforeEncoding := string(currentWord)
- encodeWord(currentWord, currentWordLength)
- if string(currentWord) != beforeEncoding {
- encodedWords = append(encodedWords, beforeEncoding)
- }
- }
- for _, letter := range currentWord {
- newString = append(newString, letter)
- }
- currentWord = []rune{}
- newString = append(newString, item)
- continue
- }
- currentWord = append(currentWord, item)
- }
- sort.Strings(encodedWords)
- return EncodedText{string(newString), encodedWords}
- }
- func removeString(array []string, index int) []string {
- return append(array[:index], array[index+1:]...)
- }
- func removeRune(array []rune, index int) []rune {
- return append(array[:index], array[index+1:]...)
- }
- func decodeWord(word []rune, wordLength int, encodedWords []string) (string, []string) {
- for index, encodedWord := range encodedWords {
- if len(encodedWord) != wordLength {
- continue
- }
- encoded := []rune(encodedWord)
- if word[0] != encoded[0] && word[wordLength-1] != encoded[wordLength-1] {
- continue
- }
- found := false
- partOfEncoded := encoded[1 : wordLength-1]
- for i := 1; i < wordLength-1; i++ {
- for j, letter := range partOfEncoded {
- if letter == word[i] {
- partOfEncoded = removeRune(partOfEncoded, j)
- found = true
- break
- }
- }
- if !found {
- break
- }
- }
- if found {
- return encodedWord, removeString(encodedWords, index)
- }
- }
- return string(word), encodedWords
- }
- //DecodeText returns decoded provided text using provided array of encoded words
- func DecodeText(encoded EncodedText) string {
- runes := []rune(encoded.Text)
- encodedWords := make([]string, len(encoded.EncodedWords))
- copy(encodedWords, encoded.EncodedWords)
- var currentWord []rune
- var newString []rune
- for _, item := range runes {
- if unicode.IsPunct(item) || unicode.IsSpace(item) {
- currentWordLength := len(currentWord)
- if currentWordLength >= 4 {
- var decoded string
- decoded, encodedWords = decodeWord(currentWord, currentWordLength, encodedWords)
- currentWord = []rune(decoded)
- }
- for _, letter := range currentWord {
- newString = append(newString, letter)
- }
- currentWord = []rune{}
- newString = append(newString, item)
- continue
- }
- currentWord = append(currentWord, item)
- }
- return string(newString)
- }
|