如何从prolog网页查询prolog数据库?

时间:2016-08-31 03:15:58

标签: http prolog swi-prolog

我正在使用swipl。我正在尝试从网页查询下面的parts.pl数据库文件,但我不确定如何发布查询并让它“查询”数据库并返回这些结果。

即,我期望能够做的就是在swipl命令行中输入查询,例如'part(jeep,100,A,B)。并在页面上将相同的结果返回给我。

任何人都可以就如何做到这一点给出一些指导。这是我到目前为止所做的。

:- use_module(library(http/thread_httpd)).
:- use_module(library(http/http_dispatch)).
:- use_module(library(http/http_error)).
:- use_module(library(http/html_write)).
:- use_module(library(http/http_client)).

:- use_module(parts).

:- http_handler('/', my_query_form, []).

server(Port) :-
    http_server(http_dispatch, [port(Port)]).

my_query_form(Request) :-
    reply_html_page(
        my_style,
        [title('My Test')],
        [\page_content(Request)]).

page_content(_Request) -->
    html(
        [
        form([action='/landing', method='POST'], [
            p([], [
                label([for=model], 'Model '),
                input([name=model, type=textarea])
                    ]),
            p([], [
                label([for=major], 'Major '),
                input([name=major, type=textarea])
                    ]),
            p([], [
                label([for=minor], 'Minor '),
                input([name=part, type=textarea])
                    ]),
            p([], input([name=submit, type=submit, value='Search'], []))
        ])]).

:- http_handler('/landing', landing_pad, []).

landing_pad(Request) :-
    member(method(post), Request), !,
    http_read_data(Request, Data, []),
    format('Content-type: text/html~n~n', []),
    format('<p>', []),
    portray_clause(Data),
    format('</p<p>=======~n', []),
    portray_clause(Request),
    format('</p>').

:- multifile
    user:body//2.

user:body(my_style, Body) -->
    html(body([
        div(id(top), h3('Parts Query')),
        div(id(beta), p(i('(beta)'))),
        div(id(content), Body)
        ])).

这是我的parts.pl文件。

:- module(parts,[part/4]).
% part(model, major, minor, description).
part(jeep, 100, 1000, 'description of 100-1000').
part(jeep, 100, 1001, 'description of 100-1001').
part(jeep, 100, 1002, 'description of 100-1002').
part(jeep, 101, 1000, 'description of 101-1000').
part(jeep, 101, 1001, 'description of 101-1001').
part(jeep, 101, 1002, 'description of 101-1002').
part(ford, 101, 1000, 'description of 101-1000').

更新:(我的最新实际查询,但返回整个数据库)

landing_pad(Request) :-
    member(method(post), Request), !,
    http_read_data(Request, Data, []),
    format('Content-type: text/html~n~n', []),    
    format('<p>', []),
    memberchk(model=Model, Data),
    findall(p(Model, Major, Minor, Description),
        part(Model, Major, Minor, Description), Descriptions),
    maplist(description, Descriptions),
    format('</p>').   

description(p(M,A,I,D)) :- format("~q ~q:~q - ~q</br>", [M,A,I,D]).

1 个答案:

答案 0 :(得分:1)

非常关闭。

剩下的就是使用指定的组件实际查询您的数据库。我给你这个查询的一部分,并留下其余的作为练习:

landing_pad(Request) :-
    member(method(post), Request), !,
    http_read_data(Request, Data, []),
    format('Content-type: text/html~n~n', []),
    format('<p>', []),
    memberchk(model=Model, Data),
    findall(p(Major,Description), part(Model, Major, _, Description), Descriptions),
    maplist(description, Descriptions),
    format('</p>').

我在粗体中突出显示了新部分。

findall/3收集所有匹配的部分。在这种情况下,只从表单数据中使用Model组件。你可以填写剩下的部分。

要显示说明,您可以使用例如:

description(p(M,D)) :- format("<p>~q: ~q</p>\n", [M,D]). 

作为description/1的定义。

提示:请注意使用~q格式说明符。它使调试变得更容易。如果您将其他功能添加为查询的一部分,您将看到我的意思,因为您需要区分整数和原子。