Analisador de arquivo do necrotério DCSS


9

Neste desafio, você precisa analisar os arquivos do necrotério do jogo roguelike Dungeon Crawl Stone Soup e enviá-lo para STDOUT.

O que são esses arquivos do necrotério?

Quando você morre, um arquivo de texto é gerado com os dados desse personagem. Você pode ver que equipamento o personagem tinha, o que aconteceu nos últimos turnos e quantos monstros ele matou.

Você pode encontrar um exemplo de arquivo do necrotério aqui

O desafio

Seu trabalho é criar um programa que pegue um desses arquivos do STDIN, analise-o e envie os dados para o STDOUT.

Para facilitar um pouco esse desafio, você só precisa analisar o primeiro bloco de texto. (atéThe game lasted <time> (<turns> turns).

Você precisa analisar e gerar as seguintes informações:

  • O número da versão
  • A pontuação.
  • O nome do personagem, título, raça e classe.
  • O nível do personagem.
  • A causa da morte / vitória.
  • A quantidade de voltas que a corrida durou.

Exemplo:

Dungeon Crawl Stone Soup version <version number> character file.

<score> <name> the <title> (level <level>, 224/224 HPs)
         Began as a <race> <class> on Mar 16, 2015.
         Was the Champion of the Shining One.
         <cause of death/victory>

         The game lasted 16:11:01 (<turns> turns).

Casos de teste

Caso de teste 1 - Vitória

Arquivo de entrada

Exemplo de saída - Vitória:

Version: 0.16.0-8-gd9ae3a8 (webtiles)
Score: 16059087
Name: Ryuzilla the Conqueror
Character: Gargoyle Berserker
Level: 27
Cause of Death/Victory: Escaped with the Orb and 15 runes on Mar 17 2015!
Turns: 97605

Caso de teste 2 - Morte

Arquivo de entrada

Exemplo de saída - Morte:

Version: 0.16-a0-3667-g690a316 (webtiles)
Score: 462
Name: 8Escape the Ruffian
Character: Bearkin Transmuter
Level: 6
Cause of Death/Victory: Slain by an orc wielding a +0 trident (3 damage) on level 4 of the Dungeon.
Turns: 3698

Regras

  • Isso é então o código mais curto vence.
  • Em caso de empate, a resposta mais antiga vence.
  • Sem brechas padrão.
  • A entrada do arquivo deve ser obtida no STDIN
  • A saída deve ser enviada para STDOUT
  • As etiquetas antes da saída (ex. Turns:) São opcionais.

Código de exemplo ungolfed para inspiração

Código de geração de arquivos do necrotério no DCSS


A saída realmente precisa conter os rótulos de linha como Version:ou é suficiente para gerar as informações na mesma ordem, uma por linha?
Martin Ender

@ MartinBüttner Os rótulos são opcionais.
DJgamer98

A raça e a classe sempre terão uma palavra cada?
Martin Ender

@ MartinBüttner Algumas raças e classes são duas palavras, como Vine Stalker, Abyssal Knight e Deep Elf.
DJgamer98

2
Existe uma especificação desse formato de arquivo do necrotério ou apenas esses exemplos?
Paŭlo Ebermann 16/11/2015

Respostas:


3

Perl, 151 bytes

148 código + 3 interruptores ( -0, -l, -p). Tenho certeza que isso pode ser melhorado :)

Pega a entrada de STDIN e imprime o resultado ao receber EOF.

perl -lp0e 's/\.{3}|\s/ /g;y/ //s;$_=join$\,(/(\d.*?).{15}\..(\d+).(.+?).\(.+?(\d+).+?\b(?:a|an) (.+?) o.+? ([^.!]+[.!])[^.!]*?(\d+)[^(]+\)..\3/)[0..2,4,3,5..7]'

Ungolfed:

use strict;
use warnings;

# set the input record separator to undef (the -0 switch)
$/=undef;
# read the text (the -l switch)
$_=<STDIN>;

# replace all '...' and spaces by a ' '
s/\.{3}|\s/ /g;
# squeeze all contiguous spaces into a single space
y/ //s;
# collect the captured groups into @p
my @p=
/(\d.*?).{15}\..      # version is the first string starting with a digit and ending 15 characters before the period
 (\d+).               # points is the next string with only digits
 (.+?).\(.+?          # name starts after a gap of one character
 (\d+).+?\b(?:a|an)\s # level is inside the next open paranthesis
 (.+?)\so.+?\s        # race, class occur after the 'a' or 'an' and end before ' o' i.e. (' on')
 ([^.!]+[.!])[^.!]*?  # cause of death is the a sentence ending with '.' or '!'
 (\d+)[^(]+\)..\3     # turns is the next sentence with digits within parantheses, followed by 2 characters and the player's name
/x;
$_=join"\n",@p[0..2,4,3,5..7]; # the level and race lines need to be swapped

# print the output (the -p switch)
print $_;

ideone.com


3

F #, 377 bytes

open System.Text.RegularExpressions
let s=System.String.IsNullOrWhiteSpace>>not
let m f=Regex.Match((f+"").Split[|'\r';'\n'|]|>Seq.filter s|>Seq.take 8|>Seq.reduce(fun a z->a+z.Trim()), ".*n (.*) c.*\.([0-9]+) (.*) \(l.* (.*),.*a (.*) o.*\.(?:(S.*)|W.*(E.*)).*.T.*\((.*) .*\).").Groups|>Seq.cast<Group>|>Seq.skip 1|>Seq.map(fun z ->z.Value)|>Seq.filter s|>Seq.iter(printfn"%s")

3

Javascript (ES6), 297 230 bytes

Por enquanto, essa é uma expressão regular orientada a testes.

Ele simplesmente substitui as informações indesejadas e mantém as coisas importantes.

Ele cria uma função anônima que simplesmente retorna o texto desejado.

_=>_.replace(/^.+version(.*) character file\.([\n\r]+)(\d+)([^\(]+) \([^\d]+( \d+),.+\n\s+.+as a(.+) on.+\n\s+(?:Was.+One\.\n)?((?:.|\n)+[!.])\n(?:.|\n)+\((\d+)(?:.|\n)+$/,'$1\n$3\n‌​$4\n$6\n$5\n$7\n$8').replace(/\s+(\.{3} ?)?/,' ')

Não é um animal?


Obrigado pela dica do sysreq sobre os rótulos serem opcionais. Isso me salvou 67 bytes !


Você pode testar a expressão resulgar em: https://regex101.com/r/zY0sQ0/1


Os rótulos são opcionais; você pode salvar alguns bytes omitindo-os.
gato

11
@sysreq O que ...?
Ismael Miguel


2
Eu estou dizendo que _=>_.replace(/^.+version(.*) character file\.([\n\r]+)(\d+)([^\(]+) \([^\d]+( \d+),.+\n\s+.+as a(.+) on.+\n\s+(?:Was.+One\.\n)?((?:.|\n)+[!.])\n(?:.|\n)+\((\d+)(?:.|\n)+$/,'$1\n$3\n$4\n$6\n$5\n$7\n$8').replace(/\s+(\.{3} ?)?/,' ')é uma solução aceitável em apenas 230 bytes
gato

11
@sysreq Desculpe por demorar tanto para dizer qualquer coisa. Estive vendo a postagem, mas estava em um tablet. Você não tem ideia de como é doloroso fazer algo em um tablet. Substituí meu código pela sua versão sem rótulo. Muito obrigado pela dica.
Ismael Miguel

2

Python3, 472 bytes

Eu pensei que seria capaz de ficar tão mais curto. Não surpreende que eu tenha vencido minha finalização. Execute como python3 dcss.py morgue-file.txt.

import sys
n="\n"
s=" "
f=open(sys.argv[1],'r').read().split(n)[:11]
m=range
a=len
d=","
for i in m(a(f)):
 f[i]=f[i].split(s)
 for x in m(a(f[i])):
  f[i][x]=f[i][x].strip()
h=f[0]
g=f[10]
k=f[2]
def r(j,u):
 j=list(j)
 while u in j:
  j.remove(u)
 return"".join(j)
def l(x):
 c=s
 for i in m(a(x)):
  c+=x[i]+s
 return c.strip()
print(h[6]+s+h[7]+n+k[0]+n+g[0]+s+g[1]+s+g[2]+n+r(g[3],"(")+s+r(g[4],")")+n+r(k[5],d)+n+r(l(f[4])+l(f[5])+l(f[6])+l(f[7]),".")+n+r(g[17],d))

2

Go, 589 502 489 487 bytes

package main;import(."fmt";."io/ioutil";"os";."strings");func d(z,ch string)string{return Map(func(r rune)rune{if IndexRune(ch,r)<0{return r};return -1},z)};func main(){x:=Split;f,_:=ReadFile(os.Args[1]);n:="\n";l:=" ";m:=",";h:=".";q:=x(string(f),n)[:11];k:=x(q[0],l);y:=x(q[10],l);u:=x(q[2],l);g:="";for _,e:=range Fields(d(q[4],n+h)+l+d(q[5],n+h)+l+d(q[6],n+h)+l+d(q[7],n+h)){g=g+e+l};Print(k[6]+l+k[7]+n+u[0]+n+y[0]+l+y[1]+l+y[2]+n+d(y[3]+l+y[4],"()")+n+d(u[5],m)+n+g+n+d(y[17],m))}

após a execução go fmt, go fixe go vetaqui é a versão "ungolfed":

package main

import (
    . "fmt"
    . "io/ioutil"
    "os"
    . "strings"
)

func d(z, ch string) string {
    return Map(func(r rune) rune {
        if IndexRune(ch, r) < 0 {
            return r
        }
        return -1
    }, z)
}
func main() {
    x := Split
    f, _ := ReadFile(os.Args[1])
    n := "\n"
    l := " "
    m := ","
    h := "."
    q := x(string(f), n)[:11]
    k := x(q[0], l)
    y := x(q[10], l)
    u := x(q[2], l)
    g := ""
    for _, e := range Fields(d(q[4], n+h) + l + d(q[5], n+h) + l + d(q[6], n+h) + l + d(q[7], n+h)) {
        g = g + e + l
    }
    Print(k[6] + l + k[7] + n + u[0] + n + y[0] + l + y[1] + l + y[2] + n + d(y[3]+l+y[4], "()") + n + d(u[5], m) + n + g + n + d(y[17], m))
}

Editar: o uso de importações de pontos ajuda muito.

Bastante auto-explicativo, mas posso explicar se necessário. Este é o meu primeiro programa Go "real" e ainda sou iniciante no codegolf, portanto, dicas são bem-vindas!

Editar: você disse "pegue um arquivo do STDIN" e pode executar este script (se você estiver instalado) executando go install <foldername>e, em seguida, <binaryname> morgue-file.txtougo run main.go morgue.txt

Ao utilizar nosso site, você reconhece que leu e compreendeu nossa Política de Cookies e nossa Política de Privacidade.
Licensed under cc by-sa 3.0 with attribution required.