final revision
This commit is contained in:
parent
ecc73413c9
commit
bc73926122
BIN
Informe.pdf
(Stored with Git LFS)
BIN
Informe.pdf
(Stored with Git LFS)
Binary file not shown.
619
Informe.tex
619
Informe.tex
@ -1,13 +1,37 @@
|
|||||||
\documentclass[
|
\documentclass[spanish,12pt]{book}
|
||||||
spanish,
|
\usepackage[tc]{titlepic}
|
||||||
12pt]{article}
|
\usepackage[table, xcdraw]{xcolor}
|
||||||
|
\usepackage[executivepaper,margin=1in]{geometry}
|
||||||
\newcommand{\titlename}{Diseño e Implementación de un Lenguaje de Programación de Tipo Declarativo Inspirado en Prolog para Control de Agentes en Videojuegos}
|
\definecolor{title}{RGB}{180,0,0}
|
||||||
\newcommand{\Titlename}{\expandafter\MakeUppercase\expandafter{\titlename}}
|
\definecolor{other}{RGB}{171,0,255}
|
||||||
|
\definecolor{name}{RGB}{255,0,0}
|
||||||
|
\definecolor{phd}{RGB}{0,0,240}
|
||||||
|
|
||||||
\usepackage[T1]{fontenc}
|
\usepackage[T1]{fontenc}
|
||||||
\usepackage[utf8]{inputenc}
|
\usepackage[utf8]{inputenc}
|
||||||
\usepackage{babel}
|
|
||||||
|
\usepackage{graphicx}
|
||||||
|
\usepackage{mathptmx}
|
||||||
|
|
||||||
|
\usepackage{setspace}
|
||||||
|
|
||||||
|
\usepackage[spanish]{babel}
|
||||||
|
\usepackage{blindtext}
|
||||||
|
\usepackage{hyperref}
|
||||||
|
\hypersetup{
|
||||||
|
colorlinks=true,
|
||||||
|
linkcolor=black,
|
||||||
|
citecolor=black,
|
||||||
|
filecolor=black,
|
||||||
|
urlcolor=black,
|
||||||
|
%allcolors=black,
|
||||||
|
pdftitle={Diseño e Implementación de un Lenguaje de Programación Declarativo para el Control de Agentes en Videojuegos}
|
||||||
|
}
|
||||||
|
\usepackage{hypcap}
|
||||||
|
|
||||||
|
\usepackage{float}
|
||||||
|
\usepackage{csquotes}
|
||||||
|
|
||||||
\usepackage[
|
\usepackage[
|
||||||
backend=biber,
|
backend=biber,
|
||||||
style=numeric,
|
style=numeric,
|
||||||
@ -15,73 +39,15 @@
|
|||||||
defernumbers=true
|
defernumbers=true
|
||||||
]{biblatex}
|
]{biblatex}
|
||||||
\addbibresource{references.bib}
|
\addbibresource{references.bib}
|
||||||
\usepackage{csquotes}
|
|
||||||
\usepackage{graphicx}
|
|
||||||
\usepackage{mathptmx}
|
|
||||||
\usepackage{array}
|
|
||||||
\usepackage{float}
|
|
||||||
\usepackage{soulutf8}
|
|
||||||
\sethlcolor{blue}
|
|
||||||
\usepackage{listings}
|
|
||||||
\usepackage[table, xcdraw]{xcolor}
|
|
||||||
|
|
||||||
\usepackage{setspace}
|
|
||||||
|
|
||||||
\usepackage{titlesec}
|
|
||||||
|
|
||||||
\usepackage{titling}
|
|
||||||
|
|
||||||
\usepackage{tocloft}
|
|
||||||
\renewcommand{\cftsecleader}{\cftdotfill{\cftdotsep}}
|
|
||||||
|
|
||||||
\titleclass{\subsubsubsection}{straight}[\subsection]
|
|
||||||
|
|
||||||
\newcounter{subsubsubsection}[subsubsection]
|
|
||||||
\renewcommand\thesubsubsubsection{\thesubsubsection.\arabic{subsubsubsection}}
|
|
||||||
|
|
||||||
\titleformat{\subsubsubsection}
|
|
||||||
{\normalfont\normalsize\bfseries}{\thesubsubsubsection.}{1em}{}
|
|
||||||
\titlespacing*{\subsubsubsection}
|
|
||||||
{0pt}{3.25ex plus 1ex minus .2ex}{1.5ex plus .2ex}
|
|
||||||
|
|
||||||
\makeatletter
|
|
||||||
\def\toclevel@subsubsubsection{4}
|
|
||||||
\def\l@subsubsubsection{\@dottedtocline{4}{7em}{4em}}
|
|
||||||
\makeatother
|
|
||||||
|
|
||||||
\setcounter{secnumdepth}{4}
|
|
||||||
\setcounter{tocdepth}{4}
|
|
||||||
|
|
||||||
\onehalfspacing
|
|
||||||
\setstretch{1.5}
|
|
||||||
|
|
||||||
\renewcommand{\labelitemi}{$\bullet$}
|
\renewcommand{\labelitemi}{$\bullet$}
|
||||||
|
|
||||||
\usepackage{hyperref}
|
|
||||||
\hypersetup{
|
|
||||||
colorlinks=true,
|
|
||||||
linkcolor=blue,
|
|
||||||
citecolor=blue,
|
|
||||||
filecolor=blue,
|
|
||||||
urlcolor=blue,
|
|
||||||
%allcolors=blue,
|
|
||||||
pdftitle={\titlename}
|
|
||||||
}
|
|
||||||
\usepackage{hypcap}
|
|
||||||
|
|
||||||
\usepackage[left=3.0cm,
|
|
||||||
right=3.0cm,
|
|
||||||
top=2.5cm,
|
|
||||||
bottom=2.5cm,
|
|
||||||
letterpaper]{geometry}
|
|
||||||
|
|
||||||
\usepackage{ragged2e}
|
|
||||||
\justifying
|
|
||||||
\setlength\RaggedRightParindent{36pt}
|
|
||||||
\RaggedRight
|
|
||||||
|
|
||||||
\newcommand{\sectionbreak}{\phantomsection}
|
\newcommand{\sectionbreak}{\phantomsection}
|
||||||
|
|
||||||
|
\raggedbottom
|
||||||
|
|
||||||
|
\onehalfspacing
|
||||||
|
|
||||||
\begin{document}
|
\begin{document}
|
||||||
|
|
||||||
\hypersetup{pageanchor=false}
|
\hypersetup{pageanchor=false}
|
||||||
@ -89,75 +55,494 @@
|
|||||||
\thispagestyle{empty}
|
\thispagestyle{empty}
|
||||||
\clearpage
|
\clearpage
|
||||||
|
|
||||||
\pretitle{\begin{center}}
|
\begingroup
|
||||||
\posttitle{\\[40pt]\par\end{center}}
|
\let\cleardoublepage=\clearpage
|
||||||
|
|
||||||
\preauthor{\begin{center} \large \uppercase{Proyecto de Título Presentado por} \MakeUppercase}
|
\title{\bfseries {\sc\textcolor{title}{Diseño e Implementación de un Lenguaje de Programación Declarativo para el Control de Agentes en Videojuegos}}}
|
||||||
\postauthor{\uppercase{ de la Carrera Ingeniería Civil en Informática}\\ \large \uppercase{Dirigido por Clemente Rubio-Manzano}\\[50pt] \end{center}}
|
\author{\textcolor{other}{Universidad del Bío-Bío, Chile} \\[5pt]
|
||||||
|
\emph{\textcolor{other}{Facultad Ciencias Empresariales}}\\
|
||||||
|
\textcolor{other}{Departamento de Sistemas de Información} \\
|
||||||
|
\textsc{\Large{\textcolor{phd}{Ingeniería Civil en Informática}}} \\[3pt]
|
||||||
|
\textcolor{other}{} \vspace{0.4cm} \\[1in]
|
||||||
|
\textcolor{name}{By Christopher Cromer y Martín Araneda Acuña}\\[3pt] {\Large \sc \textcolor{other}{Dirigido por Clemente Rubio-Manzano}}
|
||||||
|
\vspace{2cm}
|
||||||
|
\titlepic{\includegraphics[width=0.19\textwidth]{figures/ubb.png}\\[5pt]
|
||||||
|
\textcolor{other}{}\\[5pt]
|
||||||
|
\textcolor{other}{}\\[5pt]
|
||||||
|
\textcolor{other}{}\\
|
||||||
|
\vfill
|
||||||
|
\textcolor{other}{Enero 2023}}}
|
||||||
|
|
||||||
\date{2022}
|
\date{}
|
||||||
|
|
||||||
\title{
|
\begin{spacing}{1.0}
|
||||||
\includegraphics[width=0.2\textwidth, height=0.2\textheight, keepaspectratio]{figures/ubb.png}\\
|
|
||||||
\normalsize
|
|
||||||
Universiad del Bío Bío, Chile\\
|
|
||||||
Facultad de Ciencias Empresariales\\
|
|
||||||
Departamento de Sistemas de Información
|
|
||||||
\\[32pt]
|
|
||||||
\LARGE
|
|
||||||
\Titlename
|
|
||||||
}
|
|
||||||
|
|
||||||
\author{Christopher Cromer y Martín Araneda Acuña}
|
|
||||||
\maketitle
|
\maketitle
|
||||||
|
\end{spacing}
|
||||||
|
|
||||||
\newpage
|
\chapter*{Agradecimientos}
|
||||||
\phantomsection
|
|
||||||
\pagenumbering{gobble}
|
|
||||||
\thispagestyle{empty}
|
|
||||||
\clearpage
|
|
||||||
\tableofcontents
|
|
||||||
|
|
||||||
\newpage
|
\noindent\textit{From Chris,}
|
||||||
\phantomsection
|
|
||||||
\hypersetup{pageanchor=false}
|
|
||||||
\pagenumbering{gobble}
|
|
||||||
\thispagestyle{empty}
|
|
||||||
\clearpage
|
|
||||||
\listoffigures
|
|
||||||
|
|
||||||
\newpage
|
\textit{I would like to thank my wife Elia. Her constant help and support is what has allowed me to go back to college and to achieve my goals and become the professional I am today. I owe her everything and am very lucky to have had someone on my side through this whole process during all these years of working, studying, and investigating.}
|
||||||
\phantomsection
|
|
||||||
\hypersetup{pageanchor=false}
|
\textit{I would also like to thank our thesis supervisor, Clemente Rubio-Manzano who inspired me to dig deeper into video game design and artificial intelligence. His support during this thesis was indispensable and helped us to bring this project to life.}
|
||||||
\pagenumbering{gobble}
|
\\[32pt]
|
||||||
\thispagestyle{empty}
|
|
||||||
\clearpage
|
\noindent\textit{De Martin,}
|
||||||
\listoftables
|
|
||||||
|
\textit{Agradezco a mi Padre, por su constante apoyo, esfuerzo y sabiduría. A mi madre, por su cariño, paciencia y siempre la disposición a escucharme.}
|
||||||
|
|
||||||
|
\textit{Por otro lado, quiero agradecer a nuestro profesor guía Clemente Rubio-Manzano, por darnos todo el apoyo necesario y tener siempre la disposición de ayudarnos para poder hacer de esta idea realidad.}
|
||||||
|
|
||||||
\clearpage
|
\clearpage
|
||||||
\pagenumbering{arabic}
|
\pagenumbering{arabic}
|
||||||
\hypersetup{pageanchor=true}
|
\hypersetup{pageanchor=true}
|
||||||
|
|
||||||
\include{sections/agradecimientos.tex}
|
\chapter*{Resumen}
|
||||||
|
|
||||||
\include{sections/resumen.tex}
|
Los videojuegos se han convertido en el medio de entretenimiento más importante del mundo. Su diseño e implementación depende, en gran medida, de áreas como las Ciencias de la Computación o la Inteligencia Artificial ya que permiten dotarlos del realismo necesario para alcanzar niveles óptimos de inmersión tanto desde el punto de vista gráfico como de la jugabilidad. El uso de estas tecnologías ha permitido darle un lado más autónomo y humando, así como también la posibilidad de experimentar en bastantes aspectos, dado que los videojuegos contienen muchos subgéneros y por tanto, las elecciones son incontables.
|
||||||
|
|
||||||
\include{sections/introduccion}
|
La programación de la inteligencia artificial en los videojuegos puede resultar compleja debido a los conocimientos que se debe tener en la parte gráfica que van unidos estrechamente a la parte de comportamiento. Existe un gran interés en la comunidad científica en crear lenguajes de programación que permita separar la parte gráfica de la del comportamiento, esto es, programar ambas módulos de forma independiente.
|
||||||
|
|
||||||
\include{sections/marco_teorico}
|
En base a esto, el objetivo de este proyecto título ha sido investigar sobre el diseño e implementación de lenguajes declarativos aplicados al desarrollo de videojuegos. En particular, se ha creado un lenguaje de programación de tipo declarativo, donde sólo se indica qué se debe hacer, sin ir a niveles bajos de detalle. Todo esto es con la finalidad de que sea más afín a la lógica del ser humano y que también contribuya a que la implementación del agente sea más instintivo.
|
||||||
|
|
||||||
\include{sections/estado_del_arte}
|
Para la creación del lenguaje, este ha sido escrito en C++ y también se utilizó LLVM, una tecnología que nos permitió generar y compilar nuestro lenguaje a uno máquina. Por otro lado, la parte lógica del lenguaje fue construida utilizando una base de conocimientos hecha en SQLite, que almacena todas los elementos contenidos en las palabras claves implementadas, tales como acciones, hechos y reglas.
|
||||||
|
|
||||||
\include{sections/diseño_lenguaje.tex}
|
Los resultados de prueba obtenidos de la base de datos que almacena cada partida hecha por un jugador, sea humano o no, más el software de medición de rendimiento en R nos permitió elegir los mejores modelos estadísticos para que en un futuro contrastar estos valores con el agente y comprender que tan cercano es su comportamiento con el de un ser humano.
|
||||||
|
|
||||||
\include{sections/implementacion_lenguaje.tex}
|
Finalmente, existen algunos aspectos importantes que quedan por terminar para que nuestro lenguaje de programación este completo, siendo estos el compilador, integración en el videojuego \textit{Alai}, pruebas de rendimiento para evaluar su desempeño y por último un frontend que permita visualizar los resultados de las partidas.
|
||||||
|
|
||||||
\include{sections/incorporacion_lenguaje_motor.tex}
|
\noindent\textbf{Palabras Claves:} Videojuegos, Inteligencia Artificial, Programación Declarativa, Base de Conocimiento, Compilación a Lenguaje Máquina
|
||||||
|
|
||||||
\include{sections/evaluacion_desempeño.tex}
|
\tableofcontents
|
||||||
|
|
||||||
\include{sections/conclusion.tex}
|
\listoffigures
|
||||||
|
|
||||||
\include{sections/referencias}
|
\listoftables
|
||||||
|
|
||||||
|
\chapter{Introducción}
|
||||||
|
\section{Videojuegos y su Programación}
|
||||||
|
|
||||||
|
Los videojuegos han superado al cine en cuota de mercado y se han posicionado como uno de los medios de entretenimiento más populares y complejos del mundo. Además, se han incorporado para apoyar al desarrollo en otras disciplinas como la educación o la investigación científica. Por ejemplo, en \cite{gemine2012imitative} se menciona que \textit{``los videojuegos modernos se han convertido en una alternativa de calidad y de bajo coste para evaluar algoritmos de aprendizaje automático''}. \par
|
||||||
|
|
||||||
|
Una de las fases más importantes en el desarrollo de videojuego es el modelado y programación del comportamiento de los personajes. Estos personajes se mueven de forma automática y juegan un rol fundamental en la calidad final del videojuego; esto es, permiten juegos más desafiantes y divertidos \cite{feng2016towards}. \par
|
||||||
|
El rol de los agentes tiene al menos dos dimensiones: cantidad y calidad. La cantidad se refiere a contar con un numero adecuado de actores que den la sensación al jugador de estar participando en el mundo real; la segunda es la habilidad de crear la ilusión de la credibilidad de comportamiento, directamente relacionada a la inteligencia artificial de los actores \cite{borovikov2019towards}.
|
||||||
|
|
||||||
|
En la actualidad, los agentes se programan empleando técnicas deterministas como las máquinas de estado o los lenguajes de scripting. En la literatura actual se mencionan algunos desafíos y problemas abiertos relacionados a la programación de los agentes:
|
||||||
|
\begin{enumerate}
|
||||||
|
\item \textit{''Desarrollar agentes autónomos que realicen tareas complejas es caro y consume mucho tiempo. La parte más costosa del desarrollo de estos agentes requiere extraer conocimiento de expertos humanos y adaptarlo al entorno''} \cite{van1999learning}.
|
||||||
|
\item \textit{''La I.A. de los videojuegos modernos todavía se basa en enfoques basados en reglas, y hasta ahora no se ha logrado desarrollar oponentes que resulten convincentes y sean similares a un humano.''} \cite{thurau2007bayesian}.
|
||||||
|
\item \textit{“Además, mientras los jugadores se familiarizan con la mecánica del juego y mejoran sus habilidades e idean nuevas estrategias, los agentes no cambian y eventualmente se vuelven obsoletos"} \cite{gemine2012imitative}.
|
||||||
|
\end{enumerate}
|
||||||
|
|
||||||
|
El lenguaje de scripting es diversamente usando en muchas áreas, sin embargo, en el campo de la inteligencia artificial, se utiliza mucho por la rapidez de desarrollo dado que no es necesario compilarlo y eso reduce el tiempo de iteración cuando se programa.
|
||||||
|
A este respecto, la programación del comportamiento de agentes en videojuegos se ha realizado desde diferentes puntos de vista. Los trabajos se pueden agrupar en dos categorías: (i) basados en el lenguaje de programación lógico Prolog; (ii) basados en el lenguaje de programación lógico GOLOG.
|
||||||
|
|
||||||
|
En \cite{FerreinJacobsLakemeyer2011}, se
|
||||||
|
describe y explica un lenguaje de programación lógico llamado GOLOG que permite implementar agentes de videojuego del estilo FPS (Disparos en Primera Persona) y enfrentarlos a NPCs (Non-Player Characters). El videojuego en cuestión, llamado ''Unreal Tournament 2004'', es de tipo multijugador, donde los jugadores humanos compiten para alcanzar objetivos. Los oponentes pueden ser tanto humanos o controlados por el computador. Originalmente, la toma de decisiones de los bots fue hecha usando un lenguaje orientado a objetos llamado ''UScript'', que es propio del juego y es de tipo script, con lógica basada en un \textit{''state machine''}, constando de nueve estados controlados por el motor del juego. Sin embargo, el manejo de los bots se realizó en \textit{''ReadyLog''}, un lenguaje basado en GOLOG, que permite el razonamiento basado en acciones. Con esta información se agregó una interfaz al motor del juego que le permite transmitir información importante relacionada al mapa, tal como items (munición, vida y armas), ubicación de los jugadores y estilo del ambiente en general. Por tanto, la información de estos elementos sera transmitida al bot si este es capaz de percibirlos, lo que da posibilidad a cambiar el comportamiento del bot dentro del framework.
|
||||||
|
|
||||||
|
Por otro lado, en el articulo \cite{Prolog-Scripted2016} se explica la creación e implementación de un script basado en el lenguaje Prolog dentro de un juego estilo FPS (Disparos en Primera Persona), con la finalidad de crear diferentes tácticas de comportamiento en un equipo de bots compuesto por agentes. La idea fue inspirada principalmente en la observación de tácticas de equipo presentes en torneos reales de Counter-Strike. Por otro lado, la construcción de este framework está basado en un proyecto anterior hecho por los autores para crear scripts para bots. El pilar de esta investigación se construye en la premisa de diseñar el framework para utilizar un creador de mapas y así personalizar el comportamiento del bot para mapas nuevos, pues con esto se le entregaría al agente conocimiento necesario del ambiente que le rodea para adquirir una ventaja táctica frente a oponentes humanos. Nos centramos en estudiar dos aspecto: el desarrollo del razonamiento lógica del agente y las características de su I.A. Cada bot contiene dos pilas, una de acciones y otra de tareas. Estas siempre ejecutan el bloque que esta en el tope de la pila. Para que un bot sea capaz de cumplir una tarea de manera exitosa, hará ciertas acciones asociadas a esa tarea con tal de cumplirla. Algunas condiciones causarán que se agreguen o quiten acciones dentro de esta pila. Por tanto, cuando se complete una tarea o una acción, se borrará el bloque y luego el bot intentará ejecutar la acción o tarea siguiente en la pila.Uno de los mecanismos de razonamiento utilizado es uno llamado ''Razonamiento de reflejos'', el cual sucede en cada actualización del estado actual del juego, prestando gran apoyo cuando ocurren cambios repentinos en un nivel, como en el ambiente o cuando el bot este siendo atacado. En consecuencia, este mecanismo posee una característica poderosa para agregar nuevas tareas o limpiar la pila de tareas, adaptándose a cada situación.
|
||||||
|
|
||||||
|
Los trabajos estudiados tienen algunas características relevantes:
|
||||||
|
\begin{itemize}
|
||||||
|
\item En \cite{FerreinJacobsLakemeyer2011} el tipo de I.A. esta basada en objetivos y emplea técnicas de planificación. Con respecto a la planificación de camino y la detección de colisiones debemos mencionar que la interfaz entre el juego, junto con el motor y los bots no permite modificar los algoritmos de la planificación de los caminos ni de la detección de colisiones para permitir el cumplimiento de los objetivos. El agente solo recibe información si está en su campo de visión, por tanto, no es omnisciente y no tiene información suficiente que le permita obtener ventaja contra otros agentes o jugadores.
|
||||||
|
\item En \cite{Prolog-Scripted2016} el agente usado en este proyecto es de tipo racional basado en objetivos, con base en un árbol de decisión con instrucciones condicionales. En los trabajos previos hechos por el autor, se implementó una base de datos que contiene puntos de navegación, con el fin de calcular y planificar el camino que debe seguir al moverse el bot. También se usan los puntos de navegación para tomar decisiones racionales de naturaleza táctica. El bot adquiere total conocimiento del mapa, lo que le permite tener una ventaja táctica y así aumentar sus posibilidades para ganar la partida.
|
||||||
|
\end{itemize}
|
||||||
|
|
||||||
|
\section{Nuestra Investigación}
|
||||||
|
|
||||||
|
Dado lo anterior, tiene un gran beneficio implementar una I.A. empleando un lenguaje declarativo ya que no se indica el cómo se debe realizar el comportamiento desde un punto de gráfico, solo indicamos el qué debe realizar el agente, es decir, cómo se tiene que comportar independientemente de los aspectos gráficos. Esto último es importante ya que conseguimos que la tarea de programación se aproxime a la lógica humana y el programador pueda implementar el comportamiento de los agentes de una forma más intuitiva y empleando reglas simples. En este sentido, uno de los lenguajes declarativos más importantes ha sido Prolog demostrando sus buenas propiedades en este ámbito. Por lo tanto, vamos a crear un lenguaje declarativo inspirado en Prolog que se pueda utilizar para programar el comportamiento de los personajes de un videojuego que tienen, al menos, que realizar las siguientes acciones: i) superar obstáculos, con la opción de saltar con velocidad; ii) recoger items que se agregan a la puntuación final; iii) evitar enemigos.
|
||||||
|
|
||||||
|
En este proyecto hemos desarrollado un lenguaje de programación de tipo declarativo inspirado en Prolog que permite modelar el comportamiento de los agentes de un videojuego empleando reglas lógicas.
|
||||||
|
|
||||||
|
Para llevar a cabo el objetivo anterior se han tenido que realizar los las siguientes tareas: i) Se revisó la bibliografía sobre Prolog, el motor Godot y programación de videojuegos; Se analizó la información recopilada de la bibliografía investigada; ii) Se creó el lenguaje de programación de tipo declarativo inspirado en Prolog, de nombre \textit{''Obelisk''}; iii) Se implementó una inteligencia artificial basada en el lenguaje anteriormente creado en un videojuego; iv) Se evalúo del desempeño del lenguaje inventado, con verificación del cumplimiento exitoso de la superación de obstáculos por parte del agente.
|
||||||
|
|
||||||
|
El resto de la memoria se organiza como sigue: en el Capítulo \ref{cap2} (Marco Teórico) explicaremos las herramientas que se van utilizar, análisis de rendimiento de software similares y descripción de conceptos abarcados en el proyecto; en el Capitulo 3 (Diseño e Implementación del lenguaje) presentamos el diseño del lenguaje de programación definiendo la estructura del lenguaje de programación \textit{Obelisk}, incluyendo su sintaxis y semasiología respectiva. También se detalla el desarrollo del aspecto funcional del lenguaje, tales como el funcionamiento del compilador, base de conocimiento y librerías. Asimismo se describe el proceso para utilizar el lenguaje \textit{Obelisk} con software externos; en el Capítulo 4 (Evaluación y Desempeño del Agente) realizamos una explicación de los elementos desarrollados para medir el rendimiento del agente en el videojuego \textit{Alai}; Por ultimo, les explicamos las Conclusiones del proyecto y el trabajo futuro.
|
||||||
|
|
||||||
|
\chapter{Marco Teórico}
|
||||||
|
\label{cap2}
|
||||||
|
|
||||||
|
Este capítulo tiene como propósito describir los conceptos más importantes en el desarrollo del lenguaje de programación para control de la Inteligencia Artificial, así como también su implementación en un juego estilo plataformas.
|
||||||
|
|
||||||
|
\section{Compilación del Código Fuente}
|
||||||
|
La compilación del código fuente es el proceso en el cual se transforma un lenguaje de alto nivel a uno que se puede comprender una máquina. Este procedimiento esta compuesto de varios pasos, los que serán descritos a continuación.
|
||||||
|
|
||||||
|
\begin{itemize}
|
||||||
|
\item \textbf{Lexer:} El trabajo de un lexer es leer texto y hacer un análisis léxico para encontrar los símbolos y los tokens reconocidos. Un símbolo en el código, por ejemplo \textit{\#}, representa un comentario. Por otro lado, un token es un conjunto de símbolos que contiene un significado, es decir, puede ser el nombre de un variable o función, un número, una palabra clave como \textit{if} o \textit{while}, etc.
|
||||||
|
\item \textbf{Parser:} El parser es responsable de interpretar y llevar a cabo operaciones basadas en los tokens y símbolos recibidos desde el lexer. Usando todos los tokens y símbolos encontrados, el parser construye un \textit{Abstract Syntax Tree}.
|
||||||
|
\item \textbf{\textit{Abstract Syntax Tree(AST):}} El AST se utiliza para generar código intermedio(IR). Por otro lado, es útil para controlar el flujo de la lógica y generar el código de objeto en archivos de objeto que corresponde.
|
||||||
|
\begin{figure}[H]
|
||||||
|
\centering
|
||||||
|
\includegraphics[width=0.8\textwidth, height=0.8\textheight, keepaspectratio]{figures/ast.png}
|
||||||
|
\caption{Abstract Syntaxis Tree}
|
||||||
|
\label{fig:ast}
|
||||||
|
\end{figure}
|
||||||
|
\item \textbf{Linker:} El propósito del linker es tomar los archivos de objeto, unirlos y convertirlos en un ejecutable y/o librería que la máquina puede utilizar. Los binarios del ejecutable y librería contienen código de máquina basado en la arquitectura del sistema, por ejemplo x86\_64 en caso de GNU/Linux de 64bit, que es usualmente encontrado en computadores de escritorio o notebooks usados hoy en día.
|
||||||
|
\end{itemize}
|
||||||
|
|
||||||
|
\section{Motor de Videojuegos}
|
||||||
|
|
||||||
|
Un motor de videojuegos es un conjunto de rutinas de programación que tiene como objetivo proveer facilidad en la utilización de elementos gráficos, sonidos, físicas, redes y varios otros aspectos que ayudan en el desarrollo de un videojuego. El motor que se utilizará en este proyecto tiene por nombre ''Godot''. \cite{Desarrollo-de-Videojuegos,Game-Mechanics,Godot-Projects}
|
||||||
|
|
||||||
|
El motor Godot tiene un gran valor en el desarrollo del proyecto, principalmente por su capacidad para implementar arte de píxel, en comparación con motores similares como Unity o Unreal, porque en Godot las distancias entre los elementos es medido en píxeles mientras que otros motores utilizan metros, lo que facilita en gran medida la eficacia al trabajar en juegos 2D, que usualmente para este género de videojuegos se usan píxeles y no metros.
|
||||||
|
|
||||||
|
Otro factor importante es que Godot posee un elemento llamado ''GDNative''. GDNative es una API que permite usar lenguajes de programación como C, C++ o Rust, siendo estos capaces de compilar código a lenguaje máquina. Lo anterior posee especial importancia para el proyecto, puesto que permite implementar otro lenguaje de programación que hará interacción con GDNative. \cite{Godot-24-Hours}
|
||||||
|
|
||||||
|
También la API permite que la I.A tome decisiones con mayor rapidez dado que el código objeto de máquina se ejecuta con mayor velocidad que un lenguaje interpretado o que se compila a bytecode, como se puede notar en los siguientes gráficos que comparan GDScript(lenguaje interpretado similar a Python), C\# y C++.
|
||||||
|
|
||||||
|
\begin{figure}[H]
|
||||||
|
\centering
|
||||||
|
\includegraphics[width=0.5\textwidth, height=0.5\textheight, keepaspectratio]{figures/godot-gdnative.png}
|
||||||
|
\caption{Voxel Data Generation} \cite{GDNative-Benchmarks}
|
||||||
|
\label{fig:godot_gdnative_comparison}
|
||||||
|
\end{figure}
|
||||||
|
|
||||||
|
En el caso de generación de datos voxel, existe un aumento de 8,846.43\% en su rendimiento comparando GDNative con GDScript y un aumento de 651.07\% comparando GDNative con C\#.
|
||||||
|
|
||||||
|
\begin{figure}[H]
|
||||||
|
\centering
|
||||||
|
\includegraphics[width=0.5\textwidth, height=0.5\textheight, keepaspectratio]{figures/godot-gdnative2.png}
|
||||||
|
\caption{Mesh Generation} \cite{GDNative-Benchmarks}
|
||||||
|
\label{fig:godot_gdnative_comparison2}
|
||||||
|
\end{figure}
|
||||||
|
|
||||||
|
En la prueba de generación de mallas, GDNative tiene un aumento de 651.07\% en su rendimiento comparado con C\#.
|
||||||
|
|
||||||
|
\section{Game I.A.}
|
||||||
|
|
||||||
|
La Inteligencia Artificial (I.A) es una disciplina científica relativamente nueva que nació de los avances en las ciencias de la computación y/o la informática. De forma intuitiva, la I.A puede entenderse como la capacidad que tiene una máquina para tomar decisiones de forma autónoma a partir de los estímulos que recibe del exterior. Una de las primeras veces que se hablo de la I.A como disciplina fue en el año 1956 cuando John McCarthy, importante matemático e informático, la definió como \textit{''La ciencia e ingeniería de hacer máquinas inteligentes, con mayor énfasis en programas computacionales inteligentes''} \cite{mccarthy2007artificial}.
|
||||||
|
|
||||||
|
Actualmente, la I.A puede considerarse también una tecnología moderna que ha ido evolucionando y se ha ido incorporando a nuestra vida cotidiana. Un ejemplo bastante importante es el reconocimiento facial introducido en teléfonos inteligentes, lo que permite agregar ciertos efectos a las imágenes, ajustar automáticamente el enfoque o balancear la exposición en condiciones de poca luz.
|
||||||
|
|
||||||
|
El lenguaje responsable que se encargará de la toma de decisiones por parte de la I.A será de carácter propio e inspirado en Prolog, un lenguaje declarativo y lógico. \cite{Logic-Programming}
|
||||||
|
|
||||||
|
\begin{figure}[H]
|
||||||
|
\centering
|
||||||
|
\includegraphics[width=0.9\textwidth, height=0.9\textheight, keepaspectratio]{figures/Programacion logica vs funcional.png}
|
||||||
|
\caption{Programación Lógica vs. Funcional}
|
||||||
|
\label{fig:prolog_logic_functional}
|
||||||
|
\end{figure}
|
||||||
|
|
||||||
|
Al hablar de programación funcional, la idea principal es que todos los elementos sean funciones y estos sean capaces de poder ejecutarse de manera secuencial, por lo cual se utiliza la lógica paso por paso para resolver el problema. Por otro lado, la programación lógica utiliza una base de conocimiento para hacer preguntas y recibir respuestas que se utilizarán para resolver el problema \cite{Prolog-Tutorial}.
|
||||||
|
|
||||||
|
El lenguaje tipo Prolog debe tener 3 elementos importantes para funcionar:
|
||||||
|
\begin{enumerate}
|
||||||
|
\item \textbf{Hechos:} Son datos verdaderos, como por ejemplo ''el español es una idioma''.
|
||||||
|
\item \textbf{Reglas:} Son cláusulas condicionales que conectan los hechos. Un ejemplo es: ''si vives en Chile hablas el español''.
|
||||||
|
\item \textbf{Preguntas:} Son necesarias para tener una respuesta por parte de la base de conocimiento. Un ejemplo sería ''¿El español es una idioma?''.
|
||||||
|
\end{enumerate}
|
||||||
|
|
||||||
|
\section{Lenguaje de Programación Compilado}
|
||||||
|
|
||||||
|
El proyecto LLVM es un conjunto de tecnologías de compilador y toolchain, el cual permite crear un lenguaje propio de programación \cite{LLVM-Cookbook}.
|
||||||
|
|
||||||
|
\begin{figure}[H]
|
||||||
|
\centering
|
||||||
|
\includegraphics[width=0.3\textwidth, height=0.3\textheight, keepaspectratio]{figures/llvm.png}
|
||||||
|
\caption{LLVM}
|
||||||
|
\label{fig:llvm}
|
||||||
|
\end{figure}
|
||||||
|
|
||||||
|
LLVM consiste de varios sub-proyectos, pero el que será utilizado principalmente es ''LLVM Core''. Este sub-proyecto contiene un optimizador y generador de código, siendo este último llamado LLVM Intermediate Representation(LLVM IR). La funcionalidad es similar a una Virtual Machine de bytecode que es portátil y se puede correr en cualquier sistema que posee el LLVM.
|
||||||
|
|
||||||
|
Otro aspecto importante de LLVM es que se puede utilizar el LLVM IR, que fue generado anteriormente y así compilarlo a lenguaje máquina para la arquitectura computacional que se desee. Luego, el código objeto generado se puede utilizar con un linker para crear librerías y binarios, lo que tendrá importancia al querer integrar el código que compila nuestra compilador en el motor Godot.
|
||||||
|
|
||||||
|
\section{Máquinas de Estados}
|
||||||
|
|
||||||
|
Una máquina de estados se utiliza para controlar el estado de un objeto que tiene una ramificación compleja y estados mutables. \cite{Game-Programming-Patterns} La máquina de estados finita es parte de una rama de las Ciencia de la Computación llamado \textit{''teoría de autómata''}, la cual incluye la famosa máquina de Turing. La máquina de estados es usualmente usada en programación de I.A., videojuegos y en la creación de compiladores de código, sin embargo fuera de estas 3 áreas, posee muy poca adopción y uso.
|
||||||
|
|
||||||
|
\begin{figure}[H]
|
||||||
|
\centering
|
||||||
|
\includegraphics[width=0.5\textwidth, height=0.5\textheight, keepaspectratio]{figures/state_machine.png}
|
||||||
|
\caption{Máquina de Estado Finito}
|
||||||
|
\label{fig:state_machine}
|
||||||
|
\end{figure}
|
||||||
|
|
||||||
|
La máquina de estado finita posee un conjunto concreto de estados en la que es capaz de estar, como ''saltar'' o ''caminar'', la cual también posee una restricción importante que consiste en que esta máquina solo puede presentar un solo estado en un instante de tiempo, es decir, el jugador no es capaz de saltar y caminar en un mismo momento.
|
||||||
|
|
||||||
|
El funcionamiento de la máquina de estados finita consiste en una secuencia de entradas y eventos que son enviadas a esta, los cuales se usan para cambiar entre estados. Cada máquina tiene un conjunto de transiciones y cada una de ellas está asociada con una entrada o un evento, lo que finalmente apunta a otro estado. Por tanto, cuando llega la entrada o evento y este coincide con una transición dentro del estado actual, la máquina cambiará al estado al que apunta la transición, por lo que si un jugador se encuentra ubicado en el estado ''caminar'', se puede presionar el botón de saltar para cambiar al estado de ''saltar''.
|
||||||
|
|
||||||
|
\chapter{Diseño e Implementación del Lenguaje}
|
||||||
|
|
||||||
|
Nuestro lenguaje de programación, de nombre \textit{''Obelisk''}, tiene como pilar fundamental en su diseño el paradigma declarativo, es decir, posee un fuerte enfoque en definir el resultado al que se desea llegar, en contraste a describir el flujo de control para obtener la solución.
|
||||||
|
En consecuencia, el idioma posee por naturaleza un nivel de abstracción alto.
|
||||||
|
|
||||||
|
Por otro lado, las palabras claves utilizadas están inspiradas en el lenguaje Prolog, el cual posee hechos y reglas. Esto es para luego agregar una palabra propia de las acciones a tomar, dependiendo de los hechos y las reglas.
|
||||||
|
|
||||||
|
Debido a esto, se abren posibilidades para ser perfeccionado con facilidad ya que al escribir código no es necesario determinar el procedimiento al cual se desea llegar, dándole la poderosa cualidad de ser bastante flexible.
|
||||||
|
|
||||||
|
\textit{''Obelisk''} contiene palabras claves que cuando sean utilizadas permitan que ciertas operaciones lógicas sean declaradas y ejecutadas, y por lo tanto reconocidas para su posterior uso, las cuales serán descritas a continuación.
|
||||||
|
|
||||||
|
\section{Sintaxis}
|
||||||
|
|
||||||
|
La sintaxis del lenguaje fue diseñado teniendo en mente el asemejarse al lenguaje humano, específicamente el idioma inglés. La razón es debido a que el inglés es utilizado en todas partes del mundo y es muy fácil de aprender y usar.
|
||||||
|
|
||||||
|
\begin{figure}[H]
|
||||||
|
\centering
|
||||||
|
\includegraphics[width=0.8\textwidth, height=0.8\textheight, keepaspectratio]{figures/ejemplofunc1.png}
|
||||||
|
\caption{Estructura Básica de una Palabra Clave Lógica}
|
||||||
|
\label{fig:ejemplofunc1}
|
||||||
|
\end{figure}
|
||||||
|
|
||||||
|
La estructura de la sintaxis del lenguaje de programación \textit{''Obelisk''} se desglosa en tres partes. El nombre, los elementos y el punto y coma (;) componen en su totalidad lo que nuestro idioma comprende como una palabra clave del lenguaje. Cabe señalar que las palabras claves del lenguaje están restringidas solo en el idioma inglés.
|
||||||
|
|
||||||
|
En este caso, el idioma posee tres palabras claves implementadas, las cuales son las acciones, los hechos y las reglas. Cada una de ellas tiene un orden ligeramente similar y difieren sólo en el contenido de los elementos, el cual se describirá a continuación.
|
||||||
|
|
||||||
|
\subsection{Hechos}
|
||||||
|
\begin{figure}[H]
|
||||||
|
\centering
|
||||||
|
\includegraphics[width=0.8\textwidth, height=0.8\textheight, keepaspectratio]{figures/ejemplohecho1.png}
|
||||||
|
\caption{Estructura Básica de un Hecho}
|
||||||
|
\label{fig:ejemplohecho1}
|
||||||
|
\end{figure}
|
||||||
|
|
||||||
|
Los hechos tienen su contenido dividido en dos porciones. Sin embargo, la separación semántica es hecha por un verbo.
|
||||||
|
|
||||||
|
Por lo tanto, tenemos:
|
||||||
|
|
||||||
|
\begin{itemize}
|
||||||
|
\item 1°ra Entidad(es) (primera porción): Deben estar en comillas dobles, pueden ser más de uno y deben estar separados por la conjunción \textit{and}. Ej. \textit{''chris and martin...etc''}.
|
||||||
|
\item Verbo: Representa la división semántica y la relación entre las dos porciones. Ej. \textit{''can''}
|
||||||
|
\item 2°da Entidad(es) (segunda porción): Deben estar en comillas dobles, pueden ser más de uno y deben estar separados por la conjunción \textit{and}. Ej. \textit{''program and speak english...etc''}
|
||||||
|
\end{itemize}
|
||||||
|
|
||||||
|
\subsection{Acciones}
|
||||||
|
|
||||||
|
\begin{figure}[H]
|
||||||
|
\centering
|
||||||
|
\includegraphics[width=0.8\textwidth, height=0.8\textheight, keepaspectratio]{figures/ejemploaccion1.png}
|
||||||
|
\caption{Estructura Básica de una Acción}
|
||||||
|
\label{fig:ejemploaccion1}
|
||||||
|
\end{figure}
|
||||||
|
|
||||||
|
La sintaxis de las acciones destaca principalmente, como las demás palabras clave, en los parámetros.
|
||||||
|
El contenido esta compuesto por dos porciones, divididos por el adverbio.
|
||||||
|
Usando como referencia el ejemplo anterior, la estructura es:
|
||||||
|
|
||||||
|
\begin{itemize}
|
||||||
|
\item 1°ra Entidad (primera porción): Debe estar en comillas dobles. Ej. \textit{''martin''}.
|
||||||
|
\item Verbo: Representa la relación entre las dos entidades de la primera porción. Ej. \textit{''is''}.
|
||||||
|
\item 2°da Entidad (primera porción): Debe estar en comillas dobles. Ej. \textit{''dangerous''}.
|
||||||
|
\item Separación semántica: El adverbio describe la implicación que tiene la primera porción sobre la segunda.
|
||||||
|
. Ej. \textit{''then''}.
|
||||||
|
\item Sugerencia verdadera(segunda porción): Presenta la opción a sugerir en caso en que el hecho sea verdad. Ej. \textit{''avoid''}.
|
||||||
|
\item Sugerencia falsa (segunda porción): Manifiesta la elección a suscitar si es que el hecho sea falso. Ej. \textit{''ignore''}.
|
||||||
|
\end{itemize}
|
||||||
|
|
||||||
|
\subsection{Reglas}
|
||||||
|
|
||||||
|
\begin{figure}[H]
|
||||||
|
\centering
|
||||||
|
\includegraphics[width=0.8\textwidth, height=0.8\textheight, keepaspectratio]{figures/ejemploregla1.png}
|
||||||
|
\caption{Estructura Básica de una Regla}
|
||||||
|
\label{fig:ejemploregla1}
|
||||||
|
\end{figure}
|
||||||
|
|
||||||
|
La sintaxis de las reglas, como en el resto de las palabras claves, posee una división semántica que divide el contenido de los elementos en dos fragmentos. Cabe mencionar que las reglas deben contener si o si dos entidades por cada porción, separados por un verbo. Por tanto, su estructura es:
|
||||||
|
|
||||||
|
\begin{itemize}
|
||||||
|
\item 1°era Entidad (primera porción): Debe estar entre comillas dobles. Ej. \textit{''player''}.
|
||||||
|
\item Verbo (primera porción): Es la relación entre ambas entidades de la primera porción.
|
||||||
|
\item 2°da Entidad (primera porción): Debe estar entre comillas dobles. Ej. \textit{''die''}.
|
||||||
|
\item Conjunción condicional: Indica la división semántica entre las dos porciones. En español expresa ''cuando, a ser, etc.''. Ej. \textit{''if''}.
|
||||||
|
\item 1°ra Entidad (segunda porción): Debe estar entre comillas dobles. Ej. \textit{''enemy1''}.
|
||||||
|
\item Verbo (primera porción): Es la relación entre ambas entidades de la segunda porción.
|
||||||
|
\item 2°da Entidad (segunda porción): Debe estar entre comillas dobles. Ej. \textit{''dangerous''}.
|
||||||
|
\end{itemize}
|
||||||
|
|
||||||
|
\section{Semántica}
|
||||||
|
La semasiología de \textit{Obelisk} es en base a las palabras claves contenidas en el lenguaje y en consecuencia, describen la lógica y funcionamiento que tendrán.
|
||||||
|
|
||||||
|
Cabe mencionar que cada vez que se agreguen nuevas palabras claves, quedará un registro de estos en la base de conocimientos, el cual almacena toda la información contenida en el lenguaje. Este último será explicado con detalle más adelante.
|
||||||
|
|
||||||
|
Finalmente, nuestro lenguaje posee tres palabras claves y su funcionamiento será descrito a continuación:
|
||||||
|
|
||||||
|
\begin{itemize}
|
||||||
|
\item \textbf{Hechos:} Los hechos, como su nombre indica, describen una afirmación en referencia a la entidad declarada en los elementos de la palabra clave. Es importante mencionar que si un hecho no esta presente en la base de conocimiento, se deja a entender que por defecto su valor booleano es \textit{false}.
|
||||||
|
\item \textbf{Reglas:} Las reglas se utilizan para crear condiciones y el resultado de estas dependen fundamentalmente de los hechos. Las reglas se utilizan para crear condiciones y el resultado de estas dependen fundamentalmente de los hechos. Esto quiere decir que una regla tendrá un valor booleano \textit{true} si existe el hecho que respalde tal regla. En caso contrario la regla quedará con valor \textit{false}.
|
||||||
|
\item \textbf{Acciones:} Finalmente, las acciones definen que acto(s) se realizará(n) dependiendo si el hecho al cual alude tenga valor booleano \textit{true}.
|
||||||
|
\end{itemize}
|
||||||
|
|
||||||
|
\section{Implementación del Lenguaje}
|
||||||
|
|
||||||
|
\subsection{Arquitectura}
|
||||||
|
|
||||||
|
La arquitectura del lenguaje esta formada por tres módulos: compilador, base de conocimiento y librería.
|
||||||
|
|
||||||
|
\begin{itemize}
|
||||||
|
\item \textbf{Compilador:} El compilador de Obelisk tiene como propósito leer el código fuente usando un \textit{Lexer}. Luego se utiliza el \textit{Parser} para analizar y aplicar operaciones en base a los tokens que encontró el \textit{Lexer}.
|
||||||
|
Tras esto, se crea una Base de Conocimiento, el cual posee la característica de poder consultar su información utilizando un software externo. Finalmente, el código relevante se transforma en código intermedio(IR) que luego será transformado en binario.
|
||||||
|
|
||||||
|
\begin{figure}[H]
|
||||||
|
\centering
|
||||||
|
\includegraphics[width=0.8\textwidth, height=0.6\textheight, keepaspectratio]{figures/diagramacompi.png}
|
||||||
|
\caption{Diagrama Estructural del Compilador}
|
||||||
|
\label{fig:diagramacompi}
|
||||||
|
\end{figure}
|
||||||
|
|
||||||
|
\item \textbf{Base de Conocimiento:} La Base de Conocimiento es donde se almacena toda la lógica proveniente de los hechos, reglas y acciones. Su implementación fue hecha utilizando SQLite, el cual permite hacer consultas a la base de conocimiento utilizando el lenguaje SQL.
|
||||||
|
|
||||||
|
\begin{figure}[H]
|
||||||
|
\centering
|
||||||
|
\includegraphics[width=0.6\textwidth, height=0.6\textheight, keepaspectratio]{figures/kb_diagram.png}
|
||||||
|
\caption{Estructura del Base de Conocimiento}
|
||||||
|
\label{fig:kbstructure}
|
||||||
|
\end{figure}
|
||||||
|
|
||||||
|
La estructura de la base de conocimientos consiste en seis tablas principales las cuales se describen en el siguiente cuadro:
|
||||||
|
|
||||||
|
\begin{table}[H]
|
||||||
|
\centering
|
||||||
|
\begin{tabular}{|l|p{0.66\linewidth}|}
|
||||||
|
\hline
|
||||||
|
\multicolumn{1}{|c|}{\textbf{Nombre de Tabla}} & \multicolumn{1}{c|}{\textbf{Propósito}} \\ \hline
|
||||||
|
entity & La tabla entity es usada para almacenar los nombres de las entidades presentes en los hechos, reglas y acciones. \\ \hline
|
||||||
|
verb & Se utiliza la tabla verb para almacenar los nombres de los verbos usados en los hechos y reglas. \\ \hline
|
||||||
|
action & La tabla action almacena los nombres de las posibles acciones que se pueden tomar. \\ \hline
|
||||||
|
fact & La tabla fact tiene los hechos verdaderos. Si existe una fila en esta tabla, la relación entre las dos entidades es verdadera. \\ \hline
|
||||||
|
rule & La tabla rule contiene reglas. Si una regla resulta ser verdad, se inserta el hecho en la tabla fact. \\ \hline
|
||||||
|
suggest\_action & La tabla suggest\_action contiene las dos posibles acciones que se pueden tomar dependiendo si el hecho es verdadero o falso. \\ \hline
|
||||||
|
\end{tabular}
|
||||||
|
\caption{Estructura del Base de Conocimiento}
|
||||||
|
\label{tab:kb-structure}
|
||||||
|
\end{table}
|
||||||
|
|
||||||
|
\item \textbf{Librería:} La librería de Obelisk permite interactuar y consultar a la Base de Conocimiento de Obelisk, con el uso de un software externo. Hay dos tipos de datos que puede devolver la consulta. Un string que representa la acción a tomar o un numero entre 0 y 1 que representa su valor de verdad.
|
||||||
|
\end{itemize}
|
||||||
|
|
||||||
|
\subsection{Incorporación del Lenguaje en el Motor de Videojuegos}
|
||||||
|
|
||||||
|
Para incorporar el lenguaje en un software externo, como por ejemplo nuestro videojuego \textit{Alai}, se utiliza la librería de Obelisk, la cual permite hacer consultas a la base de conocimientos y tomar acciones respectivamente.
|
||||||
|
|
||||||
|
Tomando en cuenta la utilización de un software \textit{Third-Party}, cuando uno de estos utilice el idioma \textit{Obelisk}, se hacen consultas a la base de conocimientos, la cual maneja la lógica proveniente de las palabras claves implementadas. Es importante mencionar que la base de conocimientos es compilada antes de ejecutar cualquier consulta a esta.
|
||||||
|
|
||||||
|
Después de esto, son dos los posibles resultados que pueden tener las consultas, los cuales son las acciones que debe tomar el agente o un valor numérico que represente su resultado y que se puede interpretarse en el software externo.
|
||||||
|
|
||||||
|
Basado en las decisiones elegidas en relación a la consulta de la base de conocimiento, es la responsabilidad del software poner la acción en marcha. Esto se puede llevar a cabo utilizando varias técnicas de inteligencia artificial tales como A*, máquina de estado, planificación de acciones orientados a metas, etc.
|
||||||
|
|
||||||
|
\begin{figure}[H]
|
||||||
|
\centering
|
||||||
|
\includegraphics[width=0.7\textwidth, height=0.7\textheight, keepaspectratio]{figures/alai-1.png}
|
||||||
|
\caption{Videojuego Alai}
|
||||||
|
\label{fig:alai-1}
|
||||||
|
\end{figure}
|
||||||
|
|
||||||
|
\chapter{Evaluación del Desempeño del Agente}
|
||||||
|
La evaluación del agente tiene como propósito estimar el correcto comportamiento de la inteligencia artificial dentro del videojuego, como por ejemplo la cantidad de monedas recolectadas, veces que murió, etc.
|
||||||
|
\section{Sistema}
|
||||||
|
|
||||||
|
\subsection{Monitor}
|
||||||
|
Desarrollamos un monitor hecho en el lenguaje GDScript, que es parte del motor de videojuegos Godot y que graba todo la sesión del agente y/o jugadores. La grabación del partido es basado en el ciclo de update del motor Godot, es decir que si la pantalla del jugador tiene 60Hz como su taza de refresco, se grabará 60 cuadros de información cada segundo. Al terminar un partido toda esta información se envía a un servidor donde se procesa y guarda para futuro análisis. Cabe mencionar que se envía todo los datos en formato JSON a un servidor de REST.
|
||||||
|
|
||||||
|
\subsection{Servidor}
|
||||||
|
El servidor fue programado en el lenguaje Go. La razón es por su alto nivel de rendimiento y bajo nivel de uso de recursos, tales como CPU y RAM. Esto nos permite recibir muchos más datos simultáneamente de varias partidas al mismo tiempo sin complicaciones. También el servidor provee una API de tipo REST con varios endpoints que permiten almacenar las partidas del juego y consultar la información guardada en ello para futuro estudio.
|
||||||
|
|
||||||
|
\subsection{Estructura de Base de Datos}
|
||||||
|
Todo la información de las partidas jugadas se almacenan en un base de datos MySQL.
|
||||||
|
|
||||||
|
\begin{figure}[H]
|
||||||
|
\centering
|
||||||
|
\includegraphics[width=1.4\textwidth, height=1.4\textheight, keepaspectratio, angle=270]{figures/backend_db_structure.png}
|
||||||
|
\caption{Estructura del Base de Datos del Backend}
|
||||||
|
\label{fig:dbstructure-backend}
|
||||||
|
\end{figure}
|
||||||
|
|
||||||
|
\begin{table}[H]
|
||||||
|
\centering
|
||||||
|
\begin{tabular}{|l|p{0.66\linewidth}|}
|
||||||
|
\hline
|
||||||
|
\multicolumn{1}{|c|}{\textbf{Nombre de Tabla}} & \multicolumn{1}{c|}{\textbf{Propósito}} \\ \hline
|
||||||
|
frame & La tabla frame se utiliza para almacenar la cantidad de puntos, monedas, tiempo que pasó y posiciones de los objetos en el mundo cada cuadro que dibuja el juego. \\ \hline
|
||||||
|
game & La tabla game contiene información sobre el partido y el equipo que corre tal partido, incluyendo sistema operativo dimensiones de la pantalla y datos similares. \\ \hline
|
||||||
|
godot\_version & La tabla godot\_version se usa para almacenar las versiones de Godot que han corrido un partido del juego. \\ \hline
|
||||||
|
level & La tabla level es una tabla de parametros que contiene los nombres de los niveles que se puede jugar. \\ \hline
|
||||||
|
object & La tabla object se utiliza para almacenar el estado, posición y velocidad de un objeto en un cuadro especifico del partido. \\ \hline
|
||||||
|
object\_name & La tabla object\_name es una tabla de parámetros que contiene los nombres de todos los objetos que existen en un partido del juego. \\ \hline
|
||||||
|
object\_state & La tabla object\_state tiene todos los nombres de los estados de un objeto. \\ \hline
|
||||||
|
os & La tabla os es una tabla de parámetros que contiene los posibles sistemas operativos que pueden jugar el juego. \\ \hline
|
||||||
|
player & La tabla player es una tabla opcional donde se puede almacenar los datos de la persona que está jugando un partido para poder hacer un análisis sin ser anónima. \\ \hline
|
||||||
|
user & La tabla user contiene los usuarios y contraseñas de los personas que tienen permiso hacer consultas al API para obtener los datos y usarlos. \\ \hline
|
||||||
|
\end{tabular}
|
||||||
|
\caption{Estructura del Base de Datos del Backend}
|
||||||
|
\label{tab:db-structure-backend}
|
||||||
|
\end{table}
|
||||||
|
|
||||||
|
\section{Análisis}
|
||||||
|
Son muchos los tipos de análisis que son posibles con los datos que podemos recolectar de nuestro agente. Por tanto, para facilitar el estudio, utilizamos el lenguaje de programación R para calcular todas las estadísticas de las partidas. El idioma R genera gráficos basados en los datos y formulas para facilitar la comprensión del comportamiento del agente/jugador.
|
||||||
|
|
||||||
|
\subsection{Distribución Normal}
|
||||||
|
En pocas palabras, el proceso de uso de la distribución de probabilidad normal es básicamente analizar el comportamiento de varias partidas dentro del juego para luego calcular las probabilidades de que un evento ocurra.
|
||||||
|
|
||||||
|
\begin{figure}[H]
|
||||||
|
\centering
|
||||||
|
\includegraphics[width=0.6\textwidth, height=0.6\textheight, keepaspectratio]{figures/normal-coin.png}
|
||||||
|
\caption{Distribución Normal de Densidad de Probabilidad vs. Moneda}
|
||||||
|
\label{fig:normal-coin}
|
||||||
|
\end{figure}
|
||||||
|
|
||||||
|
En la figura anterior podemos ver en el eje X la cantidad de monedas obtenidas por el agente y/o jugador. En el eje Y tenemos la densidad de probabilidad. La densidad de probabilidad se puede interpretar como la probabilidad relativa en que un valor de un variable al azar seria igual a la muestra correspondiente.
|
||||||
|
|
||||||
|
La linea verde es el valor medio representado por la formula $ \bar{X} = \frac{\Sigma_{i=1}^{n} X_{i}}{n} $ donde $ \bar{X} $ es el valor medio, $ n $ es la cantidad de partidas y $ \Sigma_{i=1}^{n} X_{i} $ es la sumatoria de los monedas en cada partida. Las lineas rojas se llaman desviación estándar. Para obtener la desviación estándar el primer paso es obtener la varianza que es representado por la formula $ \sigma^2 = \frac{\Sigma_{i=1}^{n} (X_{i})^2}{n} $ donde $ \sigma^2 $ es la varianza. Finalmente obtenemos la desviación estándar con la formula $ \sigma = \sqrt{\sigma^2} $ donde $ \sigma $ es la desviación estándar.
|
||||||
|
|
||||||
|
De 9 partidas del juego, tuvimos un valor medio de 4 monedas representadas en la linea verde. Al calcular la desviación estándar obtuvimos el valor 3, por lo tanto la primera linea roja se coloca en 1 moneda porque 4 menos 3 es 1. Y en el caso de la segunda linea roja 4 mas 3 es 7.
|
||||||
|
|
||||||
|
Si la densidad de probabilidad en un punto X es muy grande, eso significa que un valor de un variable es probablemente cerca el punto X. En el caso de la figura \ref{fig:normal-coin}, se puede interpretar eso en que es muy probable que el valor de un variable sería entre 1 a 7 acercando a 4.
|
||||||
|
|
||||||
|
\begin{figure}[H]
|
||||||
|
\centering
|
||||||
|
\includegraphics[width=0.6\textwidth, height=0.6\textheight, keepaspectratio]{figures/normal-time.png}
|
||||||
|
\caption{Distribución Normal de Densidad de Probabilidad vs. Tiempo}
|
||||||
|
\label{fig:normal-time}
|
||||||
|
\end{figure}
|
||||||
|
|
||||||
|
En la figura anterior analizamos 28 partidas con un valor medio de 22 segundos y un desviación estándar de 14 segundos. Podemos concluir que, en el valor de la variable, es bastante probable que una partida dure entre 8 segundos y 36 segundos, con un valor muy cercano a 22 segundos.
|
||||||
|
|
||||||
|
\subsection{Serie de Tiempo}
|
||||||
|
Utilizamos la serie de tiempo para analizar el comportamiento del agente durante una partida. Eso nos permite entender mejor sus decisiones y cuanto demoró en lograr los objetivos.
|
||||||
|
|
||||||
|
\begin{figure}[H]
|
||||||
|
\centering
|
||||||
|
\includegraphics[width=0.6\textwidth, height=0.6\textheight, keepaspectratio]{figures/time-series.png}
|
||||||
|
\caption{Serie de Tiempo de Monedas en un Partido}
|
||||||
|
\label{fig:serie-tiempo}
|
||||||
|
\end{figure}
|
||||||
|
|
||||||
|
En la figura \ref{fig:serie-tiempo} analizamos una partida de un jugador humano que demoró 58 segundos y recolectó 7 monedas. En el eje X tenemos el tiempo medido en segundos, y en el eje Y tenemos la cantidad de monedas que tiene el agente y/o jugador en cualquier momento.
|
||||||
|
|
||||||
|
La mejor forma de utilizar un gráfico de serie tiempo en el análisis del agente y/o jugador es compararlo con otras partidas para ver que tan cerca el gráfico es con los demás. Idealmente el agente bien desarrollado debe demorar menos tiempo en completar sus objetivos y estar similar a un partido de jugador humano por cantidad de monedas obtenidas y en el tiempo que demoró en llegar a la meta final.
|
||||||
|
|
||||||
|
\chapter*{Conclusiones y Trabajo Futuro}
|
||||||
|
En base al desarrollo del lenguaje, hemos probado con éxito el funcionamiento básico de la palabra clave lógica ''hechos'', llevando los elementos contenidos en el último a la base de conocimientos, siendo así guardados apropiadamente para futuras consultas de ser necesario.
|
||||||
|
|
||||||
|
Con respecto al videojuego \textit{Alai}, este funciona en su totalidad y con todas las mecánicas respectivas que usaremos para nuestro agente, como movimiento en dos dimensiones, saltar, doble salto, recolección de monedas, meta para finalizar el nivel y recolección de datos por cada partida.
|
||||||
|
|
||||||
|
Por otro lado, en el área de pruebas de desempeño, hemos realizado satisfactoriamente pruebas con datos de un jugador real utilizando nuestro software estadístico escrito en R, dando así paso a aplicar modelos descriptivos, como distribución normal y serie de tiempo, a nuestro agente y compararlo con un ser humano.
|
||||||
|
|
||||||
|
Además, el servidor backend que almacenará los datos recopilados del juego \textit{Alai} está completo, para en un futuro ser consultados por el frontend y visualizarlos en un formato presentable.
|
||||||
|
|
||||||
|
Sin embargo, tenemos varios elementos que proponemos desarrollar en un futuro, con la intención de complementar nuestro lenguaje de programación y su funcionalidad. Estos ya poseen una base y solo requieren su finalización.
|
||||||
|
|
||||||
|
\begin{itemize}
|
||||||
|
\item Perfeccionar nuestro lenguaje de programación para que sea \textit{Touring-complete}.
|
||||||
|
\item Análisis del comportamiento del agente dentro de cualquier software, utilizando los datos obtenidos del videojuego.
|
||||||
|
\item Pasar la lógica de consultas proveniente de la base de conocimientos y así utilizar la GPU, con librerías tales como CUDA de Nvidia.
|
||||||
|
\item Implementar lógica difusa en los hechos, reglas y acciones.
|
||||||
|
\end{itemize}
|
||||||
|
|
||||||
|
\phantomsection
|
||||||
|
\printbibliography[heading=bibintoc,title={Referencias}]
|
||||||
|
|
||||||
|
%\phantomsection
|
||||||
|
%\printbibliography[heading=subbibintoc,keyword={prolog},title={Prolog}]
|
||||||
|
%\phantomsection
|
||||||
|
%\printbibliography[heading=subbibintoc,keyword={ai},title={Inteligencia Artificial}]
|
||||||
|
%\phantomsection
|
||||||
|
%\printbibliography[heading=subbibintoc,keyword={llvm},title={LLVM}]
|
||||||
|
%\phantomsection
|
||||||
|
%\printbibliography[heading=subbibintoc,keyword={godot},title={Godot}]
|
||||||
|
%\phantomsection
|
||||||
|
%\printbibliography[heading=subbibintoc,keyword={videojuegos},title={Videojuegos}]
|
||||||
|
|
||||||
|
\endgroup
|
||||||
|
|
||||||
\end{document}
|
\end{document}
|
BIN
figures/alai-1.png
(Stored with Git LFS)
Normal file
BIN
figures/alai-1.png
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
figures/ast.png
(Stored with Git LFS)
Normal file
BIN
figures/ast.png
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
figures/godot.png
(Stored with Git LFS)
BIN
figures/godot.png
(Stored with Git LFS)
Binary file not shown.
@ -25,6 +25,15 @@
|
|||||||
keywords="videojuegos"
|
keywords="videojuegos"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@inproceedings{gemine2012imitative,
|
||||||
|
title={Imitative learning for real-time strategy games},
|
||||||
|
author={Gemine, Quentin and Safadi, Firas and Fonteneau, Rapha{\"e}l and Ernst, Damien},
|
||||||
|
booktitle={2012 IEEE Conference on Computational Intelligence and Games (CIG)},
|
||||||
|
pages={424--429},
|
||||||
|
year={2012},
|
||||||
|
organization={IEEE}
|
||||||
|
}
|
||||||
|
|
||||||
@BOOK{LLVM-Cookbook,
|
@BOOK{LLVM-Cookbook,
|
||||||
author={Pandey, Mayur and Sarda, Suyog},
|
author={Pandey, Mayur and Sarda, Suyog},
|
||||||
title={LLVM Cookbook},
|
title={LLVM Cookbook},
|
||||||
@ -105,3 +114,48 @@
|
|||||||
year= {2011},
|
year= {2011},
|
||||||
keywords="ai,prolog,videojuegos"
|
keywords="ai,prolog,videojuegos"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@article{feng2016towards,
|
||||||
|
title={Towards autonomous behavior learning of non-player characters in games},
|
||||||
|
author={Feng, Shu and Tan, Ah-Hwee},
|
||||||
|
journal={Expert Systems with Applications},
|
||||||
|
volume={56},
|
||||||
|
pages={89--99},
|
||||||
|
year={2016},
|
||||||
|
publisher={Elsevier}
|
||||||
|
}
|
||||||
|
|
||||||
|
@article{borovikov2019towards,
|
||||||
|
title={Towards interactive training of non-player characters in video games},
|
||||||
|
author={Borovikov, Igor and Harder, Jesse and Sadovsky, Michael and Beirami, Ahmad},
|
||||||
|
journal={arXiv preprint arXiv:1906.00535},
|
||||||
|
year={2019}
|
||||||
|
}
|
||||||
|
|
||||||
|
@inproceedings{van1999learning,
|
||||||
|
title={Learning hierarchical performance knowledge by observation},
|
||||||
|
author={Van Lent, Michael and Laird, John},
|
||||||
|
booktitle={ICML},
|
||||||
|
pages={229--238},
|
||||||
|
year={1999}
|
||||||
|
}
|
||||||
|
|
||||||
|
@article{thurau2007bayesian,
|
||||||
|
title={Bayesian imitation learning in game characters},
|
||||||
|
author={Thurau, Christian and Paczian, Tobias and Sagerer, Gerhard and Bauckhage, Christian},
|
||||||
|
journal={International journal of intelligent systems technologies and applications},
|
||||||
|
volume={2},
|
||||||
|
number={2-3},
|
||||||
|
pages={284--295},
|
||||||
|
year={2007},
|
||||||
|
publisher={Inderscience Publishers}
|
||||||
|
}
|
||||||
|
|
||||||
|
@inproceedings{thurau2004learning,
|
||||||
|
title={Learning human-like movement behavior for computer games},
|
||||||
|
author={Thurau, Christian and Bauckhage, Christian and Sagerer, Gerhard},
|
||||||
|
booktitle={Proc. Int. Conf. on the Simulation of Adaptive Behavior},
|
||||||
|
pages={315--323},
|
||||||
|
year={2004}
|
||||||
|
}
|
||||||
|
|
||||||
|
@ -1,15 +0,0 @@
|
|||||||
\section*{Agradecimientos}
|
|
||||||
\addcontentsline{toc}{section}{\protect{}Agradecimientos}
|
|
||||||
|
|
||||||
\noindent\textit{From Chris,}
|
|
||||||
|
|
||||||
\textit{I would like to thank my wife Alejandra. Her constant help and support is what has allowed me to go back to college and to achieve my goals and become the professional I am today. I owe her everything and am very lucky to have had someone on my side through this whole process during all these years of working, studying, and investigating.}
|
|
||||||
|
|
||||||
\textit{I would also like to thank our thesis supervisor, Clemente Rubio-Manzano who inspired me to dig deeper into video game design and artificial intelligence. His support during this thesis was indispensable and helped us to bring this project to life.}
|
|
||||||
\\[24pt]
|
|
||||||
|
|
||||||
\noindent\textit{De Martin,}
|
|
||||||
|
|
||||||
\textit{Agradezco a mi Padre, por su constante apoyo, esfuerzo y sabiduría. A mi madre, por su cariño, paciencia y siempre la disposición a escucharme.}
|
|
||||||
|
|
||||||
\textit{Por otro lado, quiero agradecer a nuestro profesor guía Clemente Rubio-Manzano, por darnos todo el apoyo necesario y tener siempre la disposición de ayudarnos para poder hacer de esta idea realidad.}
|
|
@ -1,2 +0,0 @@
|
|||||||
\subsection{Seccion 3}
|
|
||||||
bla bla
|
|
@ -1,2 +0,0 @@
|
|||||||
\subsection{Seccion 1}
|
|
||||||
bla bla
|
|
@ -1,2 +0,0 @@
|
|||||||
\subsection{Seccion 2}
|
|
||||||
bla bla
|
|
@ -1,18 +0,0 @@
|
|||||||
El presente informe se encuentra dividido en cuatro capítulos. A continuación se describe brevemente el contenido de cada uno de ellos.
|
|
||||||
|
|
||||||
\begin{enumerate}
|
|
||||||
\item \textbf{Marco Teórico:} Explicación de las herramientas a utilizar, análisis de rendimiento de software similares y descripción de conceptos abarcados en el proyecto.
|
|
||||||
|
|
||||||
\item \textbf{Estado del Arte:} Se describen las principales ideas de trabajos de investigación similares, contrastándolo con este trabajo. Se han dividido estos en dos categorías: basado en el lenguaje de programación Prolog y GOLOG, con descripciones acerca de metodología e implementación de agentes en videojuegos.
|
|
||||||
|
|
||||||
\item \textbf{Diseño del lenguaje:} Se define la estructura del lenguaje de programación \textit{Obelisk}, incluyendo su sintaxis y semasiología respectiva.
|
|
||||||
|
|
||||||
\item \textbf{Implementación del Lenguaje:} Se detalla el desarrollo del aspecto funcional del lenguaje, tales como el funcionamiento del compilador, base de conocimiento y librerías.
|
|
||||||
|
|
||||||
\item \textbf{Incorporación del Lenguaje en el Motor de Videojuegos:} Descripción del proceso para utilizar el lenguaje \textit{Obelisk} con software externos.
|
|
||||||
|
|
||||||
\item \textbf{Evaluación del Desempeño del Agente:} Explicación de los elementos desarrollados para medir el rendimiento del agente en el videojuego \textit{Alai}
|
|
||||||
|
|
||||||
\item \textbf{Conclusión:} Epílogo del proyecto de título. También se habla del trabajo futuro que nuestra tesis tendrá.
|
|
||||||
\end{enumerate}
|
|
||||||
Además, al final del informe se adjuntan las referencias con los artículos utilizados en el proceso de investigación.
|
|
@ -1,14 +0,0 @@
|
|||||||
\section{Conclusión}
|
|
||||||
El trabajo realizado y la experiencia que tuvimos con nuestra tesis nos ha enriquecido como futuros profesionales en más de un modo, especialmente en el desarrollo de inteligencia artificial, el cual es una subrama de la programación que ha estado en constante evolución desde ya mas de 50 años.
|
|
||||||
|
|
||||||
Cabe destacar de igual forma el arduo camino de aprendizaje que recorrimos para desarrollar este proyecto, conociendo y experimentando con herramientas nuevas, desarrollo de soluciones ante problemas que no conocíamos, entre otros. Todo esto nos dejo valores y maestría necesaria para en un futuro, continuar con esta idea.
|
|
||||||
|
|
||||||
\subsection{Trabajo Futuro}
|
|
||||||
Dentro de este aspecto, tenemos varios elementos que proponemos desarrollar en un futuro, con la intención de complementar nuestro lenguaje de programación o su funcionalidad. Estos ya poseen una base y solo requieren su finalización.
|
|
||||||
|
|
||||||
\begin{itemize}
|
|
||||||
\item Perfeccionar nuestro lenguaje de programación para que sea \textit{Touring-complete}.
|
|
||||||
\item Análisis del comportamiento del agente dentro de cualquier software, utilizando los datos obtenidos del videojuego.
|
|
||||||
\item Pasar la lógica de consultas proveniente de la base de conocimientos y así utilizar la GPU, con librerías tales como CUDA de Nvidia.
|
|
||||||
\item Implementar lógica difusa en los hechos, reglas y acciones.
|
|
||||||
\end{itemize}
|
|
@ -1,108 +0,0 @@
|
|||||||
\section{Diseño del Lenguaje}
|
|
||||||
Nuestro lenguaje de programación, de nombre \textit{''Obelisk''}, tiene como pilar fundamental en su diseño el paradigma declarativo, es decir, posee un fuerte enfoque en definir el resultado al que se desea llegar, en contraste a describir el flujo de control para obtener la solución.
|
|
||||||
En consecuencia, el idioma posee por naturaleza un nivel de abstracción alto.
|
|
||||||
|
|
||||||
Por otro lado, las palabras claves utilizadas están inspiradas en el lenguaje Prolog, el cual posee hechos y reglas. Esto es para luego agregar una palabra propia de las acciones a tomar, dependiendo de los hechos y las reglas.
|
|
||||||
|
|
||||||
Debido a esto, se abren posibilidades para ser perfeccionado con facilidad ya que al escribir código no es necesario determinar el procedimiento al cual se desea llegar, dándole la poderosa cualidad de ser bastante flexible.
|
|
||||||
|
|
||||||
\textit{''Obelisk''} contiene palabras claves que cuando sean utilizadas permitan que ciertas operaciones lógicas sean declaradas y ejecutadas, y por lo tanto reconocidas para su posterior uso, las cuales serán descritas a continuación.
|
|
||||||
\subsection{Sintaxis}
|
|
||||||
|
|
||||||
La sintaxis del lenguaje fue diseñado teniendo en mente el asemejarse al lenguaje humano, específicamente el idioma inglés. La razón es debido a que el inglés es utilizado en todas partes del mundo y es muy fácil de aprender y usar.
|
|
||||||
|
|
||||||
\begin{figure}[H]
|
|
||||||
\centering
|
|
||||||
\includegraphics[width=0.8\textwidth, height=0.8\textheight, keepaspectratio]{figures/ejemplofunc1.png}
|
|
||||||
\caption{Estructura Básica de una Palabra Clave Lógica}
|
|
||||||
\label{fig:ejemplofunc1}
|
|
||||||
\end{figure}
|
|
||||||
|
|
||||||
La estructura de la sintaxis del lenguaje de programación \textit{''Obelisk''} se desglosa en tres partes. El nombre, los elementos y el punto y coma (;) componen en su totalidad lo que nuestro idioma comprende como una palabra clave del lenguaje. Cabe señalar que las palabras claves del lenguaje están restringidas solo en el idioma inglés.
|
|
||||||
|
|
||||||
En este caso, el idioma posee tres palabras claves implementadas, las cuales son las acciones, los hechos y las reglas. Cada una de ellas tiene un orden ligeramente similar y difieren sólo en el contenido de los elementos, el cual se describirá a continuación.
|
|
||||||
|
|
||||||
\subsubsection{Hechos}
|
|
||||||
\begin{figure}[H]
|
|
||||||
\centering
|
|
||||||
\includegraphics[width=0.8\textwidth, height=0.8\textheight, keepaspectratio]{figures/ejemplohecho1.png}
|
|
||||||
\caption{Estructura Básica de un Hecho}
|
|
||||||
\label{fig:ejemplohecho1}
|
|
||||||
\end{figure}
|
|
||||||
|
|
||||||
Los hechos tienen su contenido dividido en dos porciones. Sin embargo, la separación semántica es hecha por un verbo.
|
|
||||||
|
|
||||||
Por lo tanto, tenemos:
|
|
||||||
|
|
||||||
\begin{itemize}
|
|
||||||
\item 1°ra Entidad(es) (primera porción): Deben estar en comillas dobles, pueden ser más de uno y deben estar separados por la conjunción \textit{and}. Ej. \textit{''chris and martin...etc''}.
|
|
||||||
\item Verbo: Representa la división semántica y la relación entre las dos porciones. Ej. \textit{''can''}
|
|
||||||
\item 2°da Entidad(es) (segunda porción): Deben estar en comillas dobles, pueden ser más de uno y deben estar separados por la conjunción \textit{and}. Ej. \textit{''program and speak english...etc''}
|
|
||||||
\end{itemize}
|
|
||||||
|
|
||||||
\subsubsection{Acciones}
|
|
||||||
|
|
||||||
\begin{figure}[H]
|
|
||||||
\centering
|
|
||||||
\includegraphics[width=0.8\textwidth, height=0.8\textheight, keepaspectratio]{figures/ejemploaccion1.png}
|
|
||||||
\caption{Estructura Básica de una Acción}
|
|
||||||
\label{fig:ejemploaccion1}
|
|
||||||
\end{figure}
|
|
||||||
|
|
||||||
La sintaxis de las acciones destaca principalmente, como las demás palabras clave, en los parámetros.
|
|
||||||
El contenido esta compuesto por dos porciones, divididos por el adverbio.
|
|
||||||
Usando como referencia el ejemplo anterior, la estructura es:
|
|
||||||
|
|
||||||
\begin{itemize}
|
|
||||||
\item 1°ra Entidad (primera porción): Debe estar en comillas dobles. Ej. \textit{''martin''}.
|
|
||||||
\item Verbo: Representa la relación entre las dos entidades de la primera porción. Ej. \textit{''is''}.
|
|
||||||
\item 2°da Entidad (primera porción): Debe estar en comillas dobles. Ej. \textit{''dangerous''}.
|
|
||||||
\item Separación semántica: El adverbio describe la implicación que tiene la primera porción sobre la segunda.
|
|
||||||
. Ej. \textit{''then''}.
|
|
||||||
\item Sugerencia verdadera(segunda porción): Presenta la opción a sugerir en caso en que el hecho sea verdad. Ej. \textit{''avoid''}.
|
|
||||||
\item Sugerencia falsa (segunda porción): Manifiesta la elección a suscitar si es que el hecho sea falso. Ej. \textit{''ignore''}.
|
|
||||||
\end{itemize}
|
|
||||||
|
|
||||||
\subsubsection{Reglas}
|
|
||||||
|
|
||||||
\begin{figure}[H]
|
|
||||||
\centering
|
|
||||||
\includegraphics[width=0.8\textwidth, height=0.8\textheight, keepaspectratio]{figures/ejemploregla1.png}
|
|
||||||
\caption{Estructura Básica de una Regla}
|
|
||||||
\label{fig:ejemploregla1}
|
|
||||||
\end{figure}
|
|
||||||
|
|
||||||
La sintaxis de las reglas, como en el resto de las palabras claves, posee una división semántica que divide el contenido de los elementos en dos fragmentos. Cabe mencionar que las reglas deben contener si o si dos entidades por cada porción, separados por un verbo
|
|
||||||
|
|
||||||
Por tanto, su estructura es:
|
|
||||||
|
|
||||||
\begin{itemize}
|
|
||||||
\item 1°era Entidad (primera porción): Debe estar entre comillas dobles. Ej. \textit{''player''}.
|
|
||||||
\item Verbo (primera porción): Es la relación entre ambas entidades de la primera porción.
|
|
||||||
\item 2°da Entidad (primera porción): Debe estar entre comillas dobles. Ej. \textit{''die''}.
|
|
||||||
\item Conjunción condicional: Indica la división semántica entre las dos porciones. En español expresa ''cuando, a ser, etc.''. Ej. \textit{''if''}.
|
|
||||||
\item 1°ra Entidad (segunda porción): Debe estar entre comillas dobles. Ej. \textit{''enemy1''}.
|
|
||||||
\item Verbo (primera porción): Es la relación entre ambas entidades de la segunda porción.
|
|
||||||
\item 2°da Entidad (segunda porción): Debe estar entre comillas dobles. Ej. \textit{''dangerous''}.
|
|
||||||
\end{itemize}
|
|
||||||
|
|
||||||
\subsection{Semántica}
|
|
||||||
La semasiología de \textit{Obelisk} es en base a las palabras claves contenidas en el lenguaje y en consecuencia, describen la lógica y funcionamiento que tendrán.
|
|
||||||
|
|
||||||
Cabe mencionar que cada vez que se agreguen nuevas palabras claves, quedará un registro de estos en la base de conocimientos, el cual almacena toda la información contenida en el lenguaje. Este ultimo será explicado con detalle más adelante.
|
|
||||||
|
|
||||||
Finalmente, nuestro lenguaje posee tres palabras claves y su funcionamiento será descrito a continuación:
|
|
||||||
|
|
||||||
\subsubsection{Hechos}
|
|
||||||
Los hechos, como su nombre indica, describen una afirmación en referencia a la entidad declarada en los elementos de la palabra clave.
|
|
||||||
|
|
||||||
Es importante mencionar que si un hecho no esta presente en la base de conocimiento, se deja a entender que por defecto su valor booleano es \textit{false}.
|
|
||||||
|
|
||||||
|
|
||||||
\subsubsection{Reglas}
|
|
||||||
Las reglas se utilizan para crear condiciones y el resultado de estas dependen fundamentalmente de los hechos.
|
|
||||||
|
|
||||||
Esto quiere decir que una regla tendrá un valor booleano \textit{true} si existe el hecho que respalde tal regla. En caso contrario la regla quedará con valor \textit{false}.
|
|
||||||
|
|
||||||
\subsubsection{Acciones}
|
|
||||||
Finalmente, las acciones definen que acto(s) se realizará(n) dependiendo si el hecho al cual alude tenga valor booleano \textit{true}.
|
|
@ -1,9 +0,0 @@
|
|||||||
\section{Estado del Arte}
|
|
||||||
|
|
||||||
La programación del comportamiento de agentes en videojuegos se ha realizado desde diferentes puntos de vista. Los trabajos de investigación que hemos estudiado se pueden agrupar en dos categorías: (i) basados en el lenguaje de programación lógico Prolog; (ii) basados en el lenguaje de programación lógico GOLOG. A continuación se describirán cada una de las propuestas de forma más detallada.
|
|
||||||
|
|
||||||
\input{sections/estado_del_arte/prolog}
|
|
||||||
|
|
||||||
\input{sections/estado_del_arte/golog}
|
|
||||||
|
|
||||||
\input{sections/estado_del_arte/comparacion}
|
|
@ -1,23 +0,0 @@
|
|||||||
\subsection{Comparación de Trabajos}
|
|
||||||
En el Cuadro 1 se realiza un resumen general de las principales características de proyectos de investigación similares al nuestro, contrastando las diferencias de cada uno de ellos. Cabe destacar que una de las mayores diferencias presentes en relación a los otros trabajos es el tipo de lenguaje y si es un juego propio del autor. Lo primero es porque el uso de lenguaje compilado trae consigo una enorme ventaja en aspectos como por ejemplo la velocidad de toma de decisiones complejas en distintos ambientes en que el agente pueda encontrarse.
|
|
||||||
|
|
||||||
\begin{table}[H]
|
|
||||||
\centering
|
|
||||||
\begin{tabular}{|r|c|c|c|}
|
|
||||||
\hline
|
|
||||||
\multicolumn{1}{|l|}{} & \textbf{Obelisk} & \textbf{Prolog} & \textbf{ReadyLog} \\ \hline
|
|
||||||
\textbf{Basado en} & Prolog & Prolog & Golog \\ \hline
|
|
||||||
\textbf{Tipo de Lenguaje} & Compilado & Scripted & Scripted \\ \hline
|
|
||||||
\textbf{Juego Propio} & Sí & No & No \\ \hline
|
|
||||||
\textbf{Tipo de Juego} & Platformer & FPS & FPS \\ \hline
|
|
||||||
\textbf{Tipo de I.A.} & Racional & Racional & Racional \\ \hline
|
|
||||||
\textbf{Uso de \textit{BOT}} & No & Sí & Sí \\ \hline
|
|
||||||
\textbf{Planificación de Camino} & Posible & Sí & Imposible \\ \hline
|
|
||||||
\textbf{Detección de Colisión} & Posible & Desconocido & Imposible \\ \hline
|
|
||||||
\textbf{Agente Omnisciente} & No & Sí & No \\ \hline
|
|
||||||
\end{tabular}
|
|
||||||
\caption{Comparación de Trabajos}
|
|
||||||
\label{tab:comparar-trabajos}
|
|
||||||
\end{table}
|
|
||||||
|
|
||||||
El desarrollo de un juego propio permite una mejor integración entre la inteligencia artificial y el juego en cuestión, porque nos da mejor control en todos los aspectos que permiten el control del agente.
|
|
@ -1,15 +0,0 @@
|
|||||||
\subsection{Basado en el Lenguaje de Programación GOLOG}
|
|
||||||
|
|
||||||
En el articulo \textit{Controlling Unreal Tournament 2004 Bots with the logic-based action language Golog} \cite{FerreinJacobsLakemeyer2011}, se
|
|
||||||
describe y explica un lenguaje de programación lógico llamado GOLOG que permite implementar agentes de videojuego del estilo FPS (Disparos en Primera Persona) y enfrentarlos a NPCs (Non-Player Characters). El videojuego en cuestión, llamado ''Unreal Tournament 2004'', es de tipo multijugador, donde los jugadores humanos compiten para alcanzar objetivos. Los oponentes pueden ser tanto humanos o controlados por el computador. Nos centramos en estudiar dos aspectos de la propuesta: el control de los oponentes y la características de la I.A.
|
|
||||||
|
|
||||||
\subsubsection{Control de Oponentes}
|
|
||||||
|
|
||||||
Originalmente, la toma de decisiones de los bots fue hecha usando un lenguaje orientado a objetos llamado ''UScript'', que es propio del juego y es de tipo script, con lógica basada en un \textit{''state machine''}, constando de nueve estados controlados por el motor del juego. Sin embargo, el manejo de los bots se realizó en \textit{''ReadyLog''}, un lenguaje basado en GOLOG, que permite el razonamiento basado en acciones.
|
|
||||||
|
|
||||||
Con esta información se agregó una interfaz al motor del juego que le permite transmitir información importante relacionada al mapa, tal como items (munición, vida y armas), ubicación de los jugadores y estilo del ambiente en general. Por tanto, la información de estos elementos sera transmitida al bot si este es capaz de percibirlos, lo que da posibilidad a cambiar el comportamiento del bot dentro del framework.
|
|
||||||
|
|
||||||
\subsubsection{Características de la I.A.}
|
|
||||||
|
|
||||||
El tipo de I.A. esta basada en objetivos y emplea técnicas de planificación. Con respecto a la
|
|
||||||
planificación de camino y la detección de colisiones debemos mencionar que la interfaz entre el juego, junto con el motor y los bots no permite modificar los algoritmos de la planificación de los caminos ni de la detección de colisiones para permitir el cumplimiento de los objetivos. El agente solo recibe información si está en su campo de visión, por tanto, no es omnisciente y no tiene información suficiente que le permita obtener ventaja contra otros agentes o jugadores.
|
|
@ -1,15 +0,0 @@
|
|||||||
\subsection{Basado en el Lenguaje de Programación Lógico Prolog}
|
|
||||||
|
|
||||||
El articulo \textit{Prolog-Scripted Tactics Negotiation and Coordinated Team
|
|
||||||
Actions for Counter-Strike Game Bots} \cite{Prolog-Scripted2016}, explica la creación e implementación de un script basado en el lenguaje Prolog dentro de un juego estilo FPS (Disparos en Primera Persona), con la finalidad de crear diferentes tácticas de comportamiento en un equipo de bots compuesto por agentes. La idea fue inspirada principalmente en la observación de tácticas de equipo presentes en torneos reales de Counter-Strike.
|
|
||||||
|
|
||||||
Por otro lado, la construcción de este framework está basado en un proyecto anterior hecho por los autores para crear scripts para bots. El pilar de esta investigación se construye en la premisa de diseñar el framework para utilizar un creador de mapas y así personalizar el comportamiento del bot para mapas nuevos, pues con esto se le entregaría al agente conocimiento necesario del ambiente que le rodea para adquirir una ventaja táctica frente a oponentes humanos. Nos centramos en estudiar dos aspecto: el desarrollo del razonamiento lógica del agente y las características de su I.A.
|
|
||||||
|
|
||||||
\subsubsection{Desarrollo del Razonamiento Lógico del Agente}
|
|
||||||
Cada bot contiene dos pilas, una de acciones y otra de tareas. Estas siempre ejecutan el bloque que esta en el tope de la pila. Para que un bot sea capaz de cumplir una tarea de manera exitosa, hará ciertas acciones asociadas a esa tarea con tal de cumplirla. Algunas condiciones causarán que se agreguen o quiten acciones dentro de esta pila. Por tanto, cuando se complete una tarea o una acción, se borrará el bloque y luego el bot intentará ejecutar la acción o tarea siguiente en la pila.
|
|
||||||
|
|
||||||
Uno de los mecanismos de razonamiento utilizado es uno llamado ''Razonamiento de reflejos'', el cual sucede en cada actualización del estado actual del juego, prestando gran apoyo cuando ocurren cambios repentinos en un nivel, como en el ambiente o cuando el bot este siendo atacado. En consecuencia, este mecanismo posee una característica poderosa para agregar nuevas tareas o limpiar la pila de tareas, adaptándose a cada situación.
|
|
||||||
|
|
||||||
\subsubsection{Características de la I.A.}
|
|
||||||
|
|
||||||
El agente usado en este proyecto es de tipo racional basado en objetivos, con base en un árbol de decisión con instrucciones condicionales. En los trabajos previos hechos por el autor, se implementó una base de datos que contiene puntos de navegación, con el fin de calcular y planificar el camino que debe seguir al moverse el bot. También se usan los puntos de navegación para tomar decisiones racionales de naturaleza táctica. El bot adquiere total conocimiento del mapa, lo que le permite tener una ventaja táctica y así aumentar sus posibilidades para ganar la partida.
|
|
@ -1,83 +0,0 @@
|
|||||||
\section{Evaluación del Desempeño del Agente}
|
|
||||||
La evaluación del agente tiene como propósito estimar el correcto comportamiento de la inteligencia artificial dentro del videojuego, como por ejemplo la cantidad de monedas recolectadas, veces que murió, etc.
|
|
||||||
\subsection{Sistema}
|
|
||||||
|
|
||||||
\subsubsection{Monitor}
|
|
||||||
Desarrollamos un monitor hecho en el lenguaje GDScript, que es parte del motor de videojuegos Godot y que graba todo la sesión del agente y/o jugadores. La grabación del partido es basado en el ciclo de update del motor Godot, es decir que si la pantalla del jugador tiene 60Hz como su taza de refresco, se grabará 60 cuadros de información cada segundo. Al terminar un partido toda esta información se envía a un servidor donde se procesa y guarda para futuro análisis. Cabe mencionar que se envía todo los datos en formato JSON a un servidor de REST.
|
|
||||||
|
|
||||||
\subsubsection{Servidor}
|
|
||||||
El servidor fue programado en el lenguaje Go. La razón es por su alto nivel de rendimiento y bajo nivel de uso de recursos, tales como CPU y RAM. Esto nos permite recibir muchos más datos simultáneamente de varios partidos al mismo tiempo sin complicaciones. También el servidor provee una API de tipo REST con varios endpoints que permiten almacenar las partidas del juego y consultar la información guardada en ello para futuro estudio.
|
|
||||||
|
|
||||||
\subsubsubsection{Estructura de Base de Datos}
|
|
||||||
Todo la información de las partidas jugadas se almacenan en un base de datos MySQL.
|
|
||||||
|
|
||||||
\begin{figure}[H]
|
|
||||||
\centering
|
|
||||||
\includegraphics[width=1.4\textwidth, height=1.4\textheight, keepaspectratio, angle=270]{figures/backend_db_structure.png}
|
|
||||||
\caption{Estructura del Base de Datos del Backend}
|
|
||||||
\label{fig:dbstructure-backend}
|
|
||||||
\end{figure}
|
|
||||||
|
|
||||||
\begin{table}[H]
|
|
||||||
\centering
|
|
||||||
\begin{tabular}{|l|p{0.66\linewidth}|}
|
|
||||||
\hline
|
|
||||||
\multicolumn{1}{|c|}{\textbf{Nombre de Tabla}} & \multicolumn{1}{c|}{\textbf{Propósito}} \\ \hline
|
|
||||||
frame & La tabla frame se utiliza para almacenar la cantidad de puntos, monedas, tiempo que pasó y posiciones de los objetos en el mundo cada cuadro que dibuja el juego. \\ \hline
|
|
||||||
game & La tabla game contiene información sobre el partido y el equipo que corre tal partido, incluyendo sistema operativo dimensiones de la pantalla y datos similares. \\ \hline
|
|
||||||
godot\_version & La tabla godot\_version se usa para almacenar las versiones de Godot que han corrido un partido del juego. \\ \hline
|
|
||||||
level & La tabla level es una tabla de parametros que contiene los nombres de los niveles que se puede jugar. \\ \hline
|
|
||||||
object & La tabla object se utiliza para almacenar el estado, posición y velocidad de un objeto en un cuadro especifico del partido. \\ \hline
|
|
||||||
object\_name & La tabla object\_name es una tabla de parámetros que contiene los nombres de todos los objetos que existen en un partido del juego. \\ \hline
|
|
||||||
object\_state & La tabla object\_state tiene todos los nombres de los estados de un objeto. \\ \hline
|
|
||||||
os & La tabla os es una tabla de parámetros que contiene los posibles sistemas operativos que pueden jugar el juego. \\ \hline
|
|
||||||
player & La tabla player es una tabla opcional donde se puede almacenar los datos de la persona que está jugando un partido para poder hacer un análisis sin ser anónima. \\ \hline
|
|
||||||
user & La tabla user contiene los usuarios y contraseñas de los personas que tienen permiso hacer consultas al API para obtener los datos y usarlos. \\ \hline
|
|
||||||
\end{tabular}
|
|
||||||
\caption{Estructura del Base de Datos del Backend}
|
|
||||||
\label{tab:db-structure-backend}
|
|
||||||
\end{table}
|
|
||||||
|
|
||||||
\subsection{Análisis}
|
|
||||||
Muchos tipos de análisis son posibles con los datos que recolectamos. Para facilitar el análisis utilizamos el lenguaje de programación R para calcular todas las estadísticas de los partidos. El lenguaje R se genera gráficos basados en los datos y formulas que utilizamos para ayudar entender el comportamiento del agente/jugador.
|
|
||||||
|
|
||||||
\subsubsection{Distribución Normal}
|
|
||||||
La distribución normal nos permite analizar el comportamiento de varios partidos del juego y ver la probabilidades que un evento ocurriría.
|
|
||||||
|
|
||||||
\begin{figure}[H]
|
|
||||||
\centering
|
|
||||||
\includegraphics[width=0.6\textwidth, height=0.6\textheight, keepaspectratio]{figures/normal-coin.png}
|
|
||||||
\caption{Distribución Normal de Densidad de Probabilidad vs. Moneda}
|
|
||||||
\label{fig:normal-coin}
|
|
||||||
\end{figure}
|
|
||||||
|
|
||||||
En la figura anterior podemos ver en el eje X la cantidad de monedas obtenidos por el agente y/o jugador. En el eje Y tenemos la densidad de probabilidad. La densidad de probabilidad se puede interpretar como la probabilidad relativa en que un valor de un variable al azar seria igual a la muestra correspondiente.
|
|
||||||
|
|
||||||
La linea verde es el valor medio representado por la formula $ \bar{X} = \frac{\Sigma_{i=1}^{n} X_{i}}{n} $ donde $ \bar{X} $ es el valor medio, $ n $ es la cantidad de partidos y $ \Sigma_{i=1}^{n} X_{i} $ es la sumatorio de los monedas en cada partido. Las lineas rojas se llaman la desviación estándar. Para obtener la desviación estándar el primer paso es obtener la varianza que es representado por la formula $ \sigma^2 = \frac{\Sigma_{i=1}^{n} (X_{i})^2}{n} $ donde $ \sigma^2 $ es la varianza. Finalmente obtenemos la desviación estándar con la formula $ \sigma = \sqrt{\sigma^2} $ donde $ \sigma $ es la desviación estándar.
|
|
||||||
|
|
||||||
De 9 partidos del juego, tuvimos un valor medio de 4 monedas representado en la linea verde. Al calcular la desviación estándar obtuvimos el valor 3, por lo tanto la primera linea roja se coloca en 1 moneda porque 4 menos 3 es 1. Y en el caso de la segunda linea roja 4 mas 3 es 7.
|
|
||||||
|
|
||||||
Si la densidad de probabilidad en un punto X es muy grande, eso significa que un valor de un variable es probablemente cerca el punto X. En el caso de la figura \ref{fig:normal-coin}, se puede interpretar eso en que es muy probable que el valor de un variable sería entre 1 a 7 acercando a 4.
|
|
||||||
|
|
||||||
\begin{figure}[H]
|
|
||||||
\centering
|
|
||||||
\includegraphics[width=0.6\textwidth, height=0.6\textheight, keepaspectratio]{figures/normal-time.png}
|
|
||||||
\caption{Distribución Normal de Densidad de Probabilidad vs. Tiempo}
|
|
||||||
\label{fig:normal-time}
|
|
||||||
\end{figure}
|
|
||||||
|
|
||||||
En la figura anterior analizamos 28 partidos con un valor medio de 22 segundos y un desviación estándar de 14 segundos. Podemos concluir que en un valor de un variable es muy probable ser entre 8 segundos y 36 segundos un partido, acercando a 22 segundos.
|
|
||||||
|
|
||||||
\subsubsection{Serie de Tiempo}
|
|
||||||
Utilizamos el serie de tiempo para analizar el comportamiento del agente durante un partido. Eso nos permite entender mejor sus decisiones y cuanto demoró en lograr los objetivos.
|
|
||||||
|
|
||||||
\begin{figure}[H]
|
|
||||||
\centering
|
|
||||||
\includegraphics[width=0.6\textwidth, height=0.6\textheight, keepaspectratio]{figures/time-series.png}
|
|
||||||
\caption{Serie de Tiempo de Monedas en un Partido}
|
|
||||||
\label{fig:serie-tiempo}
|
|
||||||
\end{figure}
|
|
||||||
|
|
||||||
En la figura \ref{fig:serie-tiempo} analizamos un partido de un jugador humano que demoró 58 segundos y recolectó 7 monedas. En el eje X tenemos el tiempo medido en segundos, y en el eje Y tenemos la cantidad de monedas que tiene el agente y/o jugador en cualquier momento.
|
|
||||||
|
|
||||||
La mejor forma de utilizar un gráfico de serie tiempo en el análisis del agente y/o jugador es compararlo con otros partidos para ver que tan cerca el gráfico es con los demás. Idealmente el agente bien desarrollado debe demorar menos tiempo en completar sus objetivos y estar similar a un partido de jugador humano por cantidad de monedas obtenidas y en el tiempo que demoró en llegar a la meta final.
|
|
@ -1,44 +0,0 @@
|
|||||||
\section{Implementación del Lenguaje}
|
|
||||||
\subsection{Arquitectura}
|
|
||||||
|
|
||||||
\subsubsection{Compilador}
|
|
||||||
El compilador de Obelisk tiene como propósito leer el código fuente usando un \textit{Lexer}. Luego se utiliza el \textit{Parser} para analizar y aplicar operaciones en base a los tokens que encontró el \textit{Lexer}.
|
|
||||||
Tras esto, se crea una Base de Conocimiento, el cual posee la característica de poder consultar su información utilizando un software externo. Finalmente, el código relevante se transforma en código intermedio(IR) que luego será transformado en binario.
|
|
||||||
|
|
||||||
\begin{figure}[H]
|
|
||||||
\centering
|
|
||||||
\includegraphics[width=0.8\textwidth, height=0.6\textheight, keepaspectratio]{figures/diagramacompi.png}
|
|
||||||
\caption{Diagrama Estructural del Compilador}
|
|
||||||
\label{fig:diagramacompi}
|
|
||||||
\end{figure}
|
|
||||||
|
|
||||||
\subsubsection{Base de Conocimiento}
|
|
||||||
La Base de Conocimiento es donde se almacena toda la lógica proveniente de los hechos, reglas y acciones. Su implementación fue hecha utilizando SQLite, el cual permite hacer consultas a la base de conocimiento utilizando el lenguaje SQL.
|
|
||||||
|
|
||||||
\begin{figure}[H]
|
|
||||||
\centering
|
|
||||||
\includegraphics[width=0.6\textwidth, height=0.6\textheight, keepaspectratio]{figures/kb_diagram.png}
|
|
||||||
\caption{Estructura del Base de Conocimiento}
|
|
||||||
\label{fig:kbstructure}
|
|
||||||
\end{figure}
|
|
||||||
|
|
||||||
La estructura de la base de conocimientos consiste en seis tablas principales las cuales se describen en el siguiente cuadro:
|
|
||||||
|
|
||||||
\begin{table}[H]
|
|
||||||
\centering
|
|
||||||
\begin{tabular}{|l|p{0.66\linewidth}|}
|
|
||||||
\hline
|
|
||||||
\multicolumn{1}{|c|}{\textbf{Nombre de Tabla}} & \multicolumn{1}{c|}{\textbf{Propósito}} \\ \hline
|
|
||||||
entity & La tabla entity es usada para almacenar los nombres de las entidades presentes en los hechos, reglas y acciones. \\ \hline
|
|
||||||
verb & Se utiliza la tabla verb para almacenar los nombres de los verbos usados en los hechos y reglas. \\ \hline
|
|
||||||
action & La tabla action almacena los nombres de las posibles acciones que se pueden tomar. \\ \hline
|
|
||||||
fact & La tabla fact tiene los hechos verdaderos. Si existe una fila en esta tabla, la relación entre las dos entidades es verdadera. \\ \hline
|
|
||||||
rule & La tabla rule contiene reglas. Si una regla resulta ser verdad, se inserta el hecho en la tabla fact. \\ \hline
|
|
||||||
suggest\_action & La tabla suggest\_action contiene las dos posibles acciones que se pueden tomar dependiendo si el hecho es verdadero o falso. \\ \hline
|
|
||||||
\end{tabular}
|
|
||||||
\caption{Estructura del Base de Conocimiento}
|
|
||||||
\label{tab:kb-structure}
|
|
||||||
\end{table}
|
|
||||||
|
|
||||||
\subsubsection{Librería}
|
|
||||||
La librería de Obelisk permite interactuar y consultar a la Base de Conocimiento de Obelisk, con el uso de un software externo. Hay dos tipos de datos que puede devolver la consulta. Un string que representa la acción a tomar o un numero entre 0 y 1 que representa su valor de verdad.
|
|
@ -1,10 +0,0 @@
|
|||||||
\section{Incorporación del Lenguaje en el Motor de Videojuegos}
|
|
||||||
Para incorporar el lenguaje en un software externo, como por ejemplo nuestro videojuego \textit{Alai}, se utiliza la librería de Obelisk, la cual permite hacer consultas a la base de conocimientos y tomar acciones respectivamente.
|
|
||||||
|
|
||||||
\subsection{Toma de decisiones}
|
|
||||||
Tomando en cuenta la utilización de un software \textit{Third-Party}, cuando uno de estos utilice el idioma \textit{Obelisk}, se hacen consultas a la base de conocimientos, la cual maneja la lógica proveniente de las palabras claves implementadas. Es importante mencionar que la base de conocimientos es compilada antes de ejecutar cualquier consulta a esta.
|
|
||||||
|
|
||||||
Después de esto, son dos los posibles resultados que pueden tener las consultas, los cuales son las acciones que debe tomar el agente o un valor numérico que represente su resultado y que se puede interpretarse en el software externo.
|
|
||||||
|
|
||||||
\subsection{Puesta en marcha}
|
|
||||||
Basado en las decisiones elegidas en relación a la consulta de la base de conocimiento, es la responsabilidad del software poner la acción en marcha. Esto se puede llevar a cabo utilizando varias técnicas de inteligencia artificial tales como A*, máquina de estado, planificación de acciones orientados a metas, etc.
|
|
@ -1,16 +0,0 @@
|
|||||||
\section{Introducción}
|
|
||||||
|
|
||||||
La Inteligencia Artificial (I.A) es una disciplina científica relativamente nueva que nació de los avances en las ciencias de la computación y/o la informática. De forma intuitiva, la I.A puede entenderse como la capacidad que tiene una máquina para tomar decisiones de forma autónoma a partir de los estímulos que recibe del exterior. Una de las primeras veces que se hablo de la I.A como disciplina fue en el año 1956 cuando John McCarthy, importante matemático e informático, la definió como \textit{''La ciencia e ingeniería de hacer máquinas inteligentes, con mayor énfasis en programas computacionales inteligentes''} \cite{mccarthy2007artificial}.
|
|
||||||
|
|
||||||
Actualmente, la I.A puede considerarse también una tecnología moderna que ha ido evolucionando y se ha ido incorporando a nuestra vida cotidiana. Un ejemplo bastante importante es el reconocimiento facial introducido en teléfonos inteligentes, lo que permite agregar ciertos efectos a las imágenes, ajustar automáticamente el enfoque o balancear la exposición en condiciones de poca luz.
|
|
||||||
|
|
||||||
En el campo del desarrollo de videojuegos, la I.A. ha tenido mucha presencia. Por ejemplo, se ha empleado con éxito para programar los personajes o las entidades de los mismos, haciéndolos más dinámicos y divertidos. Permitiendo que el juego aprenda del usuario y creando adversarios artificiales del nivel de un humano.
|
|
||||||
|
|
||||||
Dado lo anterior, tendrá un gran beneficio implementar una I.A. empleando un lenguaje declarativo ya que no indicamos el cómo se debe computar, solo indicamos el qué se quiere computar. Esto último es importante ya que conseguimos que la tarea de programación se aproxime a la lógica humana y el programador pueda implementar el comportamiento de los agentes de una forma más intuitiva y empleando reglas simples. En este sentido, uno de los lenguajes declarativos más importantes ha sido Prolog demostrando sus buenas propiedades en este ámbito. Por lo tanto, vamos a crear un lenguaje declarativo inspirado en Prolog que se pueda utilizar para programar el comportamiento de los personajes de un videojuego que tienen, al menos, que realizar las siguientes acciones: i) superar obstáculos, con la opción de saltar con velocidad; ii) recoger items que se agregan a la puntuación final; iii) evitar enemigos.
|
|
||||||
|
|
||||||
|
|
||||||
\input{sections/composicion.tex}
|
|
||||||
|
|
||||||
\input{sections/introduccion/objetivos}
|
|
||||||
|
|
||||||
\input{sections/introduccion/metodologia}
|
|
@ -1,13 +0,0 @@
|
|||||||
\subsection{Metodología de Trabajo}
|
|
||||||
La metodología utilizada es la de investigación experimental, la cual consiste en ir implementando ciertas acciones en la que el agente debe reaccionar de acuerdo a los estímulos que están presentes en un ambiente preestablecido, que en este caso es un nivel presente en el videojuego.
|
|
||||||
Con esto, se harán pruebas de causa y efecto, recompilando información en su efectivo comportamiento racional frente a obstáculos para posterior comparación con un jugador real.
|
|
||||||
|
|
||||||
Por tanto, el plan de trabajo consistirá en las siguientes fases:
|
|
||||||
|
|
||||||
\begin{itemize}
|
|
||||||
\item \textbf{\underline{Fase 1:}} Revisión y descarte de bibliografía relacionada al desarrollo de videojuegos con implementación de inteligencia artificial inspirado en Prolog y motor Godot.
|
|
||||||
\item \textbf{\underline{Fase 2:}} Estudio de la información recopilada para posible implementación en el software.
|
|
||||||
\item \textbf{\underline{Fase 3:}} Creación del lenguaje de programación lógico de tipo declarativo inspirado en Prolog.
|
|
||||||
\item \textbf{\underline{Fase 4:}} Implementación de un videojuego en el motor Godot de estilo plataforma formado por agentes inteligentes programados usando el lenguaje recientemente creado.
|
|
||||||
\item \textbf{\underline{Fase 5:}} Evaluación del correcto desempeño del agente en la superación de obstáculos, con el cumplimiento del nivel versus un jugador real y la comparación de la puntuación y tiempo total en completar el juego entre ambos jugadores.
|
|
||||||
\end{itemize}
|
|
@ -1,5 +0,0 @@
|
|||||||
\subsection{Objetivos}
|
|
||||||
|
|
||||||
\input{sections/introduccion/objetivos/general}
|
|
||||||
|
|
||||||
\input{sections/introduccion/objetivos/especificos}
|
|
@ -1,9 +0,0 @@
|
|||||||
\subsubsection*{Objetivos Específicos}
|
|
||||||
|
|
||||||
\begin{enumerate}
|
|
||||||
\item Se revisó la bibliografía sobre Prolog, el motor Godot y programación de videojuegos.
|
|
||||||
\item Se analizó la información recopilada de la bibliografía investigada.
|
|
||||||
\item Se creó el lenguaje de programación de tipo declarativo inspirado en Prolog, de nombre \textit{''Obelisk''}.
|
|
||||||
\item Se implementó una inteligencia artificial basada en el lenguaje anteriormente creado en un videojuego.
|
|
||||||
\item Se evalúo del desempeño del lenguaje inventado, con verificación del cumplimiento exitoso de la superación de obstáculos por parte del agente.
|
|
||||||
\end{enumerate}
|
|
@ -1,3 +0,0 @@
|
|||||||
\subsubsection*{Objetivo General}
|
|
||||||
|
|
||||||
Hemos desarrollado un lenguaje de programación de tipo declarativo inspirado en Prolog que permite modelar el comportamiento de los agentes de un videojuego empleando reglas lógicas.
|
|
@ -1,5 +0,0 @@
|
|||||||
\section{Marco Teórico}
|
|
||||||
|
|
||||||
Este capítulo tiene como propósito describir los conceptos más importantes en el desarrollo del lenguaje de programación para control de la Inteligencia Artificial, así como también su implementación en un juego estilo plataformas.
|
|
||||||
|
|
||||||
\input{sections/marco_teorico/desarollo_videojuego}
|
|
@ -1,9 +0,0 @@
|
|||||||
\subsection{Desarrollo de Videojuegos}
|
|
||||||
|
|
||||||
\input{sections/marco_teorico/desarrollo_videojuegos/motor}
|
|
||||||
|
|
||||||
\input{sections/marco_teorico/desarrollo_videojuegos/game_ia}
|
|
||||||
|
|
||||||
\input{sections/marco_teorico/desarrollo_videojuegos/llvm}
|
|
||||||
|
|
||||||
\input{sections/marco_teorico/desarrollo_videojuegos/maquina_estado}
|
|
@ -1,21 +0,0 @@
|
|||||||
\subsubsection{Game I.A.}
|
|
||||||
El lenguaje responsable que se encargará de la toma de decisiones por parte de la Inteligencia Artificial será de carácter propio e inspirado en Prolog, un lenguaje declarativo y lógico. \cite{Logic-Programming}
|
|
||||||
|
|
||||||
\begin{figure}[H]
|
|
||||||
\centering
|
|
||||||
\includegraphics[width=0.9\textwidth, height=0.9\textheight, keepaspectratio]{figures/Programacion logica vs funcional.png}
|
|
||||||
\caption{Programación Lógica vs. Funcional}
|
|
||||||
\label{fig:prolog_logic_functional}
|
|
||||||
\end{figure}
|
|
||||||
|
|
||||||
Al hablar de programación funcional, la idea principal es que todos los elementos sean funciones y estos sean capaces de poder ejecutarse de manera secuencial, por lo cual se utiliza la lógica paso por paso para resolver el problema.
|
|
||||||
|
|
||||||
Por otro lado, la programación lógica utiliza una base de conocimiento para hacer preguntas y recibir respuestas que se utilizarán para resolver el problema. \cite{Prolog-Tutorial}
|
|
||||||
|
|
||||||
|
|
||||||
El lenguaje tipo Prolog debe tener 3 elementos importantes para funcionar:
|
|
||||||
\begin{enumerate}
|
|
||||||
\item \textbf{Hechos:} Son datos verdaderos, como por ejemplo ''el español es una idioma''.
|
|
||||||
\item \textbf{Reglas:} Son cláusulas condicionales que conectan los hechos. Un ejemplo es: ''si vives en Chile hablas el español''.
|
|
||||||
\item \textbf{Preguntas:} Son necesarias para tener una respuesta por parte de la base de conocimiento. Un ejemplo sería ''¿El español es una idioma?''.
|
|
||||||
\end{enumerate}
|
|
@ -1,14 +0,0 @@
|
|||||||
\subsubsubsection{Lenguaje de Programación Compilado}
|
|
||||||
El proyecto LLVM es un conjunto de tecnologías de compilador y toolchain, el cual permite crear un lenguaje propio de programación. \cite{LLVM-Cookbook}
|
|
||||||
|
|
||||||
\begin{figure}[H]
|
|
||||||
\centering
|
|
||||||
\includegraphics[width=0.3\textwidth, height=0.3\textheight, keepaspectratio]{figures/llvm.png}
|
|
||||||
\caption{LLVM}
|
|
||||||
\label{fig:llvm}
|
|
||||||
\end{figure}
|
|
||||||
|
|
||||||
LLVM consiste de varios sub-proyectos, pero el que será utilizado principalmente es ''LLVM Core''.
|
|
||||||
Este sub-proyecto contiene un optimizador y generador de código, siendo este último llamado LLVM Intermediate Representation(LLVM IR). La funcionalidad es similar a una Virtual Machine de bytecode que es portátil y se puede correr en cualquier sistema que posee el LLVM.
|
|
||||||
|
|
||||||
Otro aspecto importante de LLVM es que se puede utilizar el LLVM IR, que fue generado anteriormente y así compilarlo a lenguaje máquina para la arquitectura computacional que se desee. Luego, el código objeto generado se puede utilizar con un linker para crear librerías y binarios, lo que tendrá importancia al querer integrar el código que compila nuestra compilador en el motor Godot.
|
|
@ -1,13 +0,0 @@
|
|||||||
\subsubsubsection{Máquinas de Estados}
|
|
||||||
Una máquina de estados se utiliza para controlar el estado de un objeto que tiene una ramificación compleja y estados mutables. \cite{Game-Programming-Patterns} La máquina de estados finita es parte de una rama de la ciencia de la computación llamado \textit{''teoría de autómata''}, la cual incluye la famosa máquina de Turing. La máquina de estados es usualmente usada en programación de I.A., videojuegos y en la creación de compiladores de código, sin embargo fuera de estas 3 áreas, posee muy poca adopción y uso.
|
|
||||||
|
|
||||||
\begin{figure}[H]
|
|
||||||
\centering
|
|
||||||
\includegraphics[width=0.5\textwidth, height=0.5\textheight, keepaspectratio]{figures/state_machine.png}
|
|
||||||
\caption{Máquina de Estado Finito}
|
|
||||||
\label{fig:state_machine}
|
|
||||||
\end{figure}
|
|
||||||
|
|
||||||
La máquina de estado finita posee un conjunto concreto de estados en la que es capaz de estar, como ''saltar'' o ''caminar'', la cual también posee una restricción importante que consiste en que esta máquina solo puede presentar un solo estado en un instante de tiempo, es decir, el jugador no es capaz de saltar y caminar en un mismo momento.
|
|
||||||
|
|
||||||
El funcionamiento de la máquina de estados finita consiste en una secuencia de entradas y eventos que son enviadas a esta, los cuales se usan para cambiar entre estados. Cada máquina tiene un conjunto de transiciones y cada una de ellas está asociada con una entrada o un evento, lo que finalmente apunta a otro estado. Por tanto, cuando llega la entrada o evento y este coincide con una transición dentro del estado actual, la máquina cambiará al estado al que apunta la transición, por lo que si un jugador se encuentra ubicado en el estado ''caminar'', se puede presionar el botón de saltar para cambiar al estado de ''saltar''.
|
|
@ -1,34 +0,0 @@
|
|||||||
\subsubsection{Motor de Videojuegos}
|
|
||||||
El motor de videojuegos es, en pocas palabras, un conjunto de rutinas de programación que tiene como objetivo proveer facilidad en la utilización de elementos gráficos, sonidos, físicas, redes y varios otros aspectos que ayudan en el desarrollo de un videojuego.
|
|
||||||
El motor que se utilizará en este proyecto tiene por nombre ''Godot''. \cite{Desarrollo-de-Videojuegos,Game-Mechanics,Godot-Projects}
|
|
||||||
|
|
||||||
\begin{figure}[H]
|
|
||||||
\centering
|
|
||||||
\includegraphics[width=0.2\textwidth, height=0.2\textheight, keepaspectratio]{figures/godot.png}
|
|
||||||
\caption{Godot Engine}
|
|
||||||
\label{fig:godot_engine}
|
|
||||||
\end{figure}
|
|
||||||
|
|
||||||
El motor Godot tiene un gran valor en el desarrollo del proyecto, principalmente por su capacidad para implementar arte de píxel, en comparación con motores similares como Unity o Unreal, porque en Godot las distancias entre los elementos es medido en píxeles mientras que otros motores utilizan metros, lo que facilita en gran medida la eficacia al trabajar en juegos 2D, que usualmente para este género de videojuegos se usan píxeles y no metros.
|
|
||||||
|
|
||||||
Otro factor importante es que Godot posee un elemento llamado ''GDNative''. GDNative es una API que permite usar lenguajes de programación como C, C++ o Rust, siendo estos capaces de compilar código a lenguaje máquina. Lo anterior posee especial importancia para el proyecto, puesto que permite implementar otro lenguaje de programación que hará interacción con GDNative. \cite{Godot-24-Hours}
|
|
||||||
|
|
||||||
También la API permite que la Inteligencia Artificial tome decisiones con mayor rapidez dado que el código objeto de máquina se ejecuta con mayor velocidad que un lenguaje interpretado o que se compila a bytecode, como se puede notar en los siguientes gráficos que comparan GDScript(lenguaje interpretado similar a Python), C\# y C++.
|
|
||||||
|
|
||||||
\begin{figure}[H]
|
|
||||||
\centering
|
|
||||||
\includegraphics[width=0.5\textwidth, height=0.5\textheight, keepaspectratio]{figures/godot-gdnative.png}
|
|
||||||
\caption{Voxel Data Generation} \cite{GDNative-Benchmarks}
|
|
||||||
\label{fig:godot_gdnative_comparison}
|
|
||||||
\end{figure}
|
|
||||||
|
|
||||||
En el caso de generación de datos voxel, existe un aumento de 8,846.43\% en su rendimiento comparando GDNative con GDScript y un aumento de 651.07\% comparando GDNative con C\#.
|
|
||||||
|
|
||||||
\begin{figure}[H]
|
|
||||||
\centering
|
|
||||||
\includegraphics[width=0.5\textwidth, height=0.5\textheight, keepaspectratio]{figures/godot-gdnative2.png}
|
|
||||||
\caption{Mesh Generation} \cite{GDNative-Benchmarks}
|
|
||||||
\label{fig:godot_gdnative_comparison2}
|
|
||||||
\end{figure}
|
|
||||||
|
|
||||||
En la prueba de generación de mallas, GDNative tiene un aumento de 651.07\% en su rendimiento comparado con C\#.
|
|
@ -1,13 +0,0 @@
|
|||||||
\phantomsection
|
|
||||||
\printbibliography[heading=bibintoc,title={Referencias}]
|
|
||||||
|
|
||||||
%\phantomsection
|
|
||||||
%\printbibliography[heading=subbibintoc,keyword={prolog},title={Prolog}]
|
|
||||||
%\phantomsection
|
|
||||||
%\printbibliography[heading=subbibintoc,keyword={ai},title={Inteligencia Artificial}]
|
|
||||||
%\phantomsection
|
|
||||||
%\printbibliography[heading=subbibintoc,keyword={llvm},title={LLVM}]
|
|
||||||
%\phantomsection
|
|
||||||
%\printbibliography[heading=subbibintoc,keyword={godot},title={Godot}]
|
|
||||||
%\phantomsection
|
|
||||||
%\printbibliography[heading=subbibintoc,keyword={videojuegos},title={Videojuegos}]
|
|
@ -1,17 +0,0 @@
|
|||||||
\section*{Resumen}
|
|
||||||
\addcontentsline{toc}{section}{\protect{}Resumen}
|
|
||||||
|
|
||||||
Unas de las áreas nuevas que existen dentro de las muchas disciplinas científicas presentes en la era moderna es la Inteligencia Artificial (I.A).
|
|
||||||
La I.A. es básicamente la aptitud que puede tener una máquina para tomar decisiones, de forma independiente, en base a información proveniente del exterior.
|
|
||||||
|
|
||||||
Un área de bastante interés es la de videojuegos. El uso de esta tecnología ha permitido darle un lado más autónomo y humando a estos, así como también la posibilidad de experimentar en bastantes aspectos, dado que los videojuegos contienen muchos subgéneros y por tanto, las elecciones son incontables.
|
|
||||||
|
|
||||||
En base a esto, hemos decidido implementar una inteligencia artificial con la creación de un lenguaje de programación de tipo declarativo, que sólo indica qué procesos hacer, sin ir en detalles. Todo esto es con la finalidad de que sea más afín a la lógica del ser humano y que también contribuya a que la implementación del agente sea mas instintivo.
|
|
||||||
|
|
||||||
Para la creación del idioma, este ha sido escrito en C++ y también se utilizó LLVM, una tecnología que nos permitió generar y compilar nuestro lenguaje a uno máquina.
|
|
||||||
Por otro lado, la parte lógica del lenguaje fue construida utilizando una base de conocimientos hecha en SQLite, que almacena todas los elementos contenidos en las palabras claves implementadas, tales como acciones, hechos y reglas.
|
|
||||||
|
|
||||||
Finalmente se hizo la implementación del agente en un videojuego de estilo plataformas 2D, creado en el motor de videojuegos Godot, siendo el ultimo el más adecuado debido a su facilidad de manejo y gran compatibilidad con nuestra linea de trabajo.
|
|
||||||
\\[32pt]
|
|
||||||
|
|
||||||
\noindent\textbf{Palabras Claves:} Inteligencia Artificial, Lenguaje Lógico, Base de Conocimiento, Palabras Clave, Compilación a Lenguaje Máquina
|
|
Loading…
Reference in New Issue
Block a user