Анализатор файлов морга DCSS

9

В этом задании вам нужно проанализировать файлы морга из игры-мошенника Dungeon Crawl Stone Soup и вывести ее в STDOUT.

Что это за файлы морга?

Когда вы умираете, генерируется текстовый файл с данными этого персонажа внутри. Вы можете увидеть, какое снаряжение было у персонажа, что произошло за последние несколько ходов и сколько монстров он убил.

Вы можете найти пример файла морга здесь

Соревнование

Ваша задача - создать программу, которая берет один из этих файлов из STDIN, анализирует его и выводит данные в STDOUT.

Чтобы облегчить эту задачу, вам нужно только разобрать первый блок текста. (доThe game lasted <time> (<turns> turns).

Вам необходимо проанализировать и вывести следующую информацию:

  • Номер версии.
  • Счет.
  • Имя персонажа, звание, раса и класс.
  • Уровень персонажа.
  • Причина смерти / победы.
  • Количество ходов продолжалось.

Пример:

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).

Тестовые случаи

Тестовый пример 1 - Победа

Входной файл

Пример вывода - Победа:

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

Тестовый пример 2 - Смерть

Входной файл

Пример вывода - Смерть:

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

правила

  • Это поэтому выигрывает самый короткий код.
  • В случае ничьей побеждает самый старый ответ.
  • Нет стандартных лазеек.
  • Ввод файла должен быть взят из STDIN
  • Вывод должен быть отправлен в STDOUT
  • Метки перед выводом (напр. Turns:) Являются необязательными.

Ungolfed пример кода для вдохновения

Код генерации файла морга в DCSS

DJgamer98
источник
Действительно ли вывод должен содержать метки строк, такие как Version:или достаточно для вывода фрагментов информации в том же порядке, по одному на строку?
Мартин Эндер
@ MartinBüttner Метки являются необязательными.
DJgamer98
Будет ли гонка и класс всегда одним словом?
Мартин Эндер,
@ MartinBüttner Некоторые расы и классы - это два слова, такие как Vine Stalker, Abyssal Knight и Deep Elf.
DJgamer98
2
Есть ли спецификация этого формата файла морга или только эти примеры?
Paŭlo Ebermann

Ответы:

3

Perl, 151 байт

148 кодов + 3 переключателя ( -0, -l, -p). Я уверен, что это можно улучшить :)

Принимает ввод из STDIN и печатает результат при получении 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

svsd
источник
3

F #, 377 байт

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 байтов

На данный момент это регулярное выражение, управляемое тестом.

Он просто заменяет нежелательную информацию и сохраняет важные вещи.

Он создает анонимную функцию, которая просто возвращает нужный текст.

_=>_.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} ?)?/,' ')

Разве это не зверь?


Спасибо за подсказку sysreq о том, что ярлыки необязательны. Это спасло меня 67 байтов !


Вы можете проверить выражение resulgar на: https://regex101.com/r/zY0sQ0/1

Исмаэль Мигель
источник
Метки являются необязательными; Вы можете сохранить немало байтов, пропустив их.
кот
1
@sysreq Что ...?
Исмаэль Мигель
2
Я говорю, что _=>_.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} ?)?/,' ')приемлемое решение всего за 230 байтов
кошка
1
@sysreq Извините, что так долго ничего не говорил. Я видел почту, но я был на планшете. Вы не представляете, как больно делать что-либо в планшете. Я заменил свой код вашей версией без меток. Большое спасибо за совет.
Исмаэль Мигель
2

Python3, 472 байта

Я думал, что смогу получить это намного короче. Не удивлен, что я побил свое собственное представление, все же. Запустите это как 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 байт

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))}

после запуска go fmt, go fixи go vetвот версия "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))
}

Редактировать: использование многоточечного импорта очень помогает.

Довольно очевидно, но я могу объяснить, если это будет необходимо. Это моя первая «настоящая» программа Go, и я все еще новичок в Codegolf, поэтому советы приветствуются!

Изменить: вы сказали «взять файл из STDIN», и вы можете запустить этот скрипт (если вы установили), запустив go install <foldername>и затем <binaryname> morgue-file.txtилиgo run main.go morgue.txt

кошка
источник