| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
# $Id: SGF.pm,v 1.2 2009/02/27 19:54:29 drhyde Exp $ |
|
2
|
|
|
|
|
|
|
|
|
3
|
1
|
|
|
1
|
|
530
|
use strict; |
|
|
1
|
|
|
|
|
2
|
|
|
|
1
|
|
|
|
|
31
|
|
|
4
|
1
|
|
|
1
|
|
5
|
use warnings; |
|
|
1
|
|
|
|
|
2
|
|
|
|
1
|
|
|
|
|
44
|
|
|
5
|
|
|
|
|
|
|
|
|
6
|
|
|
|
|
|
|
package WWW::Facebook::Go::SGF; |
|
7
|
|
|
|
|
|
|
|
|
8
|
1
|
|
|
1
|
|
5
|
use vars qw($VERSION @ISA @EXPORT_OK); |
|
|
1
|
|
|
|
|
4
|
|
|
|
1
|
|
|
|
|
85
|
|
|
9
|
|
|
|
|
|
|
|
|
10
|
|
|
|
|
|
|
require Exporter; |
|
11
|
|
|
|
|
|
|
|
|
12
|
1
|
|
|
1
|
|
1985
|
use LWP::Simple; |
|
|
1
|
|
|
|
|
267312
|
|
|
|
1
|
|
|
|
|
10
|
|
|
13
|
|
|
|
|
|
|
|
|
14
|
|
|
|
|
|
|
@ISA = qw(Exporter); |
|
15
|
|
|
|
|
|
|
@EXPORT_OK = qw(facebook2sgf); |
|
16
|
|
|
|
|
|
|
$VERSION = '1.0'; |
|
17
|
|
|
|
|
|
|
|
|
18
|
|
|
|
|
|
|
=head1 NAME |
|
19
|
|
|
|
|
|
|
|
|
20
|
|
|
|
|
|
|
WWW::Facebook::Go::SGF - convert a game of Go on Facebook into SGF. |
|
21
|
|
|
|
|
|
|
|
|
22
|
|
|
|
|
|
|
=head1 SYNOPSIS |
|
23
|
|
|
|
|
|
|
|
|
24
|
|
|
|
|
|
|
use WWW::Facebook::Go::SGF qw(facebook2sgf); |
|
25
|
|
|
|
|
|
|
|
|
26
|
|
|
|
|
|
|
my $sgf = facebook2sgf($game_id); |
|
27
|
|
|
|
|
|
|
|
|
28
|
|
|
|
|
|
|
=head1 DESCRIPTION |
|
29
|
|
|
|
|
|
|
|
|
30
|
|
|
|
|
|
|
A simple tool to extract a game record from the GoTheGame application |
|
31
|
|
|
|
|
|
|
on Facebook and convert it to SGF so that you can then manipulate it |
|
32
|
|
|
|
|
|
|
using other tools. |
|
33
|
|
|
|
|
|
|
|
|
34
|
|
|
|
|
|
|
=head1 FUNCTIONS |
|
35
|
|
|
|
|
|
|
|
|
36
|
|
|
|
|
|
|
=head2 facebook2sgf |
|
37
|
|
|
|
|
|
|
|
|
38
|
|
|
|
|
|
|
This can be exported if you wish. It takes a game ID as its only |
|
39
|
|
|
|
|
|
|
parameter, and returns a scalar representation of an SGF recording |
|
40
|
|
|
|
|
|
|
of the game. |
|
41
|
|
|
|
|
|
|
|
|
42
|
|
|
|
|
|
|
You can get game IDs by visiting L |
|
43
|
|
|
|
|
|
|
and clicking the "View Full Profile" link. |
|
44
|
|
|
|
|
|
|
|
|
45
|
|
|
|
|
|
|
=cut |
|
46
|
|
|
|
|
|
|
|
|
47
|
|
|
|
|
|
|
sub facebook2sgf { |
|
48
|
4
|
|
|
4
|
1
|
3454
|
my $gameid = shift(); |
|
49
|
4
|
|
|
|
|
19
|
my @moves = split(/[\n\r]+/, _download($gameid)); |
|
50
|
|
|
|
|
|
|
|
|
51
|
4
|
|
|
|
|
5832
|
(my $size = (grep { /^var board_size = '(9|13|19)'/ } @moves)[0]) |
|
|
1858
|
|
|
|
|
2979
|
|
|
52
|
|
|
|
|
|
|
=~ s/^var board_size = '(9|13|19)'.*/$1/; |
|
53
|
|
|
|
|
|
|
|
|
54
|
4
|
|
|
|
|
20
|
my $handicap = (grep { /HANDICAP/ } @moves)[0]; |
|
|
1858
|
|
|
|
|
3367
|
|
|
55
|
4
|
|
|
|
|
10
|
my @handicapstones = (); |
|
56
|
4
|
100
|
|
|
|
19
|
if($handicap) { |
|
57
|
3
|
|
|
|
|
23
|
$handicap =~ s/.*HANDICAP','([\d_,]+)'.*/$1/; |
|
58
|
3
|
|
|
|
|
17
|
@handicapstones = split(',', $handicap); |
|
59
|
|
|
|
|
|
|
} |
|
60
|
|
|
|
|
|
|
|
|
61
|
615
|
|
|
|
|
1872
|
@moves = map { |
|
62
|
1858
|
100
|
|
|
|
7459
|
/new goMove\((\d+),'([BW])','([^']+)'/; |
|
63
|
615
|
|
|
|
|
1063
|
[$1, $2, _fixcoords($3)]; |
|
64
|
4
|
|
|
|
|
16
|
} grep { /^moves\[\d+\] = new goMove/ && $_ !~ /START|HANDICAP/ } @moves; |
|
65
|
4
|
|
|
|
|
169
|
@moves = @moves[0 .. $#moves - 1]; # lop off last NEGOTIATE |
|
66
|
|
|
|
|
|
|
|
|
67
|
4
|
100
|
|
|
|
51
|
my $komi = 0.5 + (@handicapstones ? 0 : 6); # 0.5 or 6.5 |
|
68
|
4
|
|
|
|
|
75
|
my $board = q{(;GM[1]FF[4]AP[}.__PACKAGE__. |
|
69
|
|
|
|
|
|
|
qq{]ST[1]SZ[$size]HA[}. |
|
70
|
|
|
|
|
|
|
(1+$#handicapstones). |
|
71
|
|
|
|
|
|
|
qq{]KM[$komi]PW[White player]PB[Black player]}. |
|
72
|
|
|
|
|
|
|
"\n\n"; |
|
73
|
4
|
100
|
|
|
|
14
|
if(@handicapstones) { |
|
74
|
3
|
|
|
|
|
8
|
$board .= ';AB'; |
|
75
|
3
|
|
|
|
|
8
|
foreach my $stone (map { _fixcoords($_) } @handicapstones) { |
|
|
16
|
|
|
|
|
28
|
|
|
76
|
16
|
|
|
|
|
24
|
$board .= "[$stone]"; |
|
77
|
|
|
|
|
|
|
} |
|
78
|
3
|
|
|
|
|
8
|
$board .= "\n"; |
|
79
|
|
|
|
|
|
|
} |
|
80
|
|
|
|
|
|
|
|
|
81
|
4
|
|
|
|
|
10
|
foreach(@moves) { |
|
82
|
611
|
|
|
|
|
1046
|
$board .= ';'.$_->[1].'['.$_->[2].']'; |
|
83
|
|
|
|
|
|
|
} |
|
84
|
4
|
|
|
|
|
10
|
$board .= "\n)\n"; |
|
85
|
|
|
|
|
|
|
|
|
86
|
4
|
|
|
|
|
219
|
return $board; |
|
87
|
|
|
|
|
|
|
} |
|
88
|
|
|
|
|
|
|
|
|
89
|
|
|
|
|
|
|
sub _fixcoords { |
|
90
|
631
|
|
|
631
|
|
968
|
my $fbcoord = shift; |
|
91
|
631
|
|
|
|
|
1908
|
$fbcoord =~ s/(\d+)/substr('abcdefghijklmnopqrs', $1, 1)/eg; |
|
|
1234
|
|
|
|
|
3182
|
|
|
92
|
631
|
|
|
|
|
987
|
$fbcoord =~ y/_//d; |
|
93
|
631
|
|
|
|
|
760
|
$fbcoord =~ s/PASS|NEGOTIATE//; |
|
94
|
631
|
|
|
|
|
2852
|
$fbcoord; |
|
95
|
|
|
|
|
|
|
} |
|
96
|
|
|
|
|
|
|
|
|
97
|
|
|
|
|
|
|
# private function, wraps around LWP::Simple::get so we can mock it in |
|
98
|
|
|
|
|
|
|
# testing |
|
99
|
|
|
|
|
|
|
sub _download { |
|
100
|
|
|
|
|
|
|
my $url = 'http://facebook3.wx3.com/go/go_iframe_spectate.php?game_id='.shift(); |
|
101
|
|
|
|
|
|
|
my $content = get($url) || die("Couldn't fetch $url\n"); |
|
102
|
|
|
|
|
|
|
return $content; |
|
103
|
|
|
|
|
|
|
} |
|
104
|
|
|
|
|
|
|
|
|
105
|
|
|
|
|
|
|
=head1 BUGS/WARNINGS/LIMITATIONS |
|
106
|
|
|
|
|
|
|
|
|
107
|
|
|
|
|
|
|
This has only been tested on completed games. I assume that both players |
|
108
|
|
|
|
|
|
|
correctly identified all dead groups after passing and that play didn't |
|
109
|
|
|
|
|
|
|
have to resume. Please report any bugs that you find using |
|
110
|
|
|
|
|
|
|
L. Obviously you will need to include the game id |
|
111
|
|
|
|
|
|
|
in your bug report. |
|
112
|
|
|
|
|
|
|
|
|
113
|
|
|
|
|
|
|
=head1 FEEDBACK |
|
114
|
|
|
|
|
|
|
|
|
115
|
|
|
|
|
|
|
I welcome feedback about my code, including constructive criticism |
|
116
|
|
|
|
|
|
|
and bug reports. The best bug reports include files that I can add |
|
117
|
|
|
|
|
|
|
to the test suite, which fail with the current code in CVS and will |
|
118
|
|
|
|
|
|
|
pass once I've fixed the bug. |
|
119
|
|
|
|
|
|
|
|
|
120
|
|
|
|
|
|
|
Feature requests are far more likely to get implemented if you submit |
|
121
|
|
|
|
|
|
|
a patch yourself. |
|
122
|
|
|
|
|
|
|
|
|
123
|
|
|
|
|
|
|
=head1 CVS |
|
124
|
|
|
|
|
|
|
|
|
125
|
|
|
|
|
|
|
L |
|
126
|
|
|
|
|
|
|
|
|
127
|
|
|
|
|
|
|
=head1 AUTHOR, COPYRIGHT and LICENCE |
|
128
|
|
|
|
|
|
|
|
|
129
|
|
|
|
|
|
|
Copyright 2009 David Cantrell EFE |
|
130
|
|
|
|
|
|
|
|
|
131
|
|
|
|
|
|
|
This software is free-as-in-speech software, and may be used, |
|
132
|
|
|
|
|
|
|
distributed, and modified under the terms of either the GNU |
|
133
|
|
|
|
|
|
|
General Public Licence version 2 or the Artistic Licence. It's |
|
134
|
|
|
|
|
|
|
up to you which one you use. The full text of the licences can |
|
135
|
|
|
|
|
|
|
be found in the files GPL2.txt and ARTISTIC.txt, respectively. |
|
136
|
|
|
|
|
|
|
|
|
137
|
|
|
|
|
|
|
=head1 CONSPIRACY |
|
138
|
|
|
|
|
|
|
|
|
139
|
|
|
|
|
|
|
This module is also free-as-in-mason software. |
|
140
|
|
|
|
|
|
|
|
|
141
|
|
|
|
|
|
|
=cut |
|
142
|
|
|
|
|
|
|
|
|
143
|
|
|
|
|
|
|
1; |