Initial commit
commit
121d88747a
|
@ -0,0 +1,148 @@
|
||||||
|
package microdata
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"code.google.com/p/go-html-transform/h5"
|
||||||
|
"io"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
type ValueList []interface{}
|
||||||
|
type PropertyMap map[string]ValueList
|
||||||
|
|
||||||
|
type Item struct {
|
||||||
|
properties PropertyMap
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewItem() *Item {
|
||||||
|
return &Item{
|
||||||
|
properties: make(PropertyMap, 10),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (self *Item) SetString(property string, value string) {
|
||||||
|
self.properties[property] = append(self.properties[property], value)
|
||||||
|
}
|
||||||
|
|
||||||
|
type Microdata struct {
|
||||||
|
items []*Item
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewMicrodata() *Microdata {
|
||||||
|
return &Microdata{
|
||||||
|
items: make([]*Item, 0),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type Parser struct {
|
||||||
|
p *h5.Parser
|
||||||
|
data *Microdata
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewParser(r io.Reader) *Parser {
|
||||||
|
return &Parser {
|
||||||
|
p : h5.NewParser(r),
|
||||||
|
data: NewMicrodata(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (self *Parser) Parse() (*Microdata, error) {
|
||||||
|
err := self.p.Parse()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
tree := self.p.Tree()
|
||||||
|
|
||||||
|
self.scanForItem(tree)
|
||||||
|
|
||||||
|
return self.data, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (self *Parser) scanForItem(node *h5.Node) {
|
||||||
|
if node == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
hasItemscope := false
|
||||||
|
|
||||||
|
for _, a := range node.Attr {
|
||||||
|
if a.Name == "itemscope" {
|
||||||
|
hasItemscope = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if hasItemscope {
|
||||||
|
item := NewItem()
|
||||||
|
self.data.items = append(self.data.items, item)
|
||||||
|
|
||||||
|
|
||||||
|
if len(node.Children) > 0 {
|
||||||
|
for _, child := range node.Children {
|
||||||
|
self.readItem(item, child)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
if len(node.Children) > 0 {
|
||||||
|
for _, child := range node.Children {
|
||||||
|
self.scanForItem(child)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func (self *Parser) readItem(item *Item, node *h5.Node) {
|
||||||
|
if propertyName, exists := getAttr("itemprop", node); exists {
|
||||||
|
var propertyValue string
|
||||||
|
|
||||||
|
switch node.Data() {
|
||||||
|
|
||||||
|
case "img","audio", "source", "video", "embed", "iframe", "track":
|
||||||
|
if urlValue, exists := getAttr("src", node); exists {
|
||||||
|
propertyValue = urlValue
|
||||||
|
}
|
||||||
|
case "a", "area", "link":
|
||||||
|
if urlValue, exists := getAttr("href", node); exists {
|
||||||
|
propertyValue = urlValue
|
||||||
|
}
|
||||||
|
case "data":
|
||||||
|
if urlValue, exists := getAttr("value", node); exists {
|
||||||
|
propertyValue = urlValue
|
||||||
|
}
|
||||||
|
case "time":
|
||||||
|
if urlValue, exists := getAttr("datetime", node); exists {
|
||||||
|
propertyValue = urlValue
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
var text bytes.Buffer
|
||||||
|
node.Walk( func(n *h5.Node) {
|
||||||
|
if n.Type == h5.TextNode {
|
||||||
|
text.WriteString(n.Data())
|
||||||
|
}
|
||||||
|
|
||||||
|
})
|
||||||
|
propertyValue = text.String()
|
||||||
|
}
|
||||||
|
|
||||||
|
item.SetString(propertyName, propertyValue)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(node.Children) > 0 {
|
||||||
|
for _, child := range node.Children {
|
||||||
|
self.readItem(item, child)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func getAttr(name string, node *h5.Node) (string, bool) {
|
||||||
|
for _, a := range node.Attr {
|
||||||
|
if a.Name == name {
|
||||||
|
return a.Value, true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return "", false
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,236 @@
|
||||||
|
package microdata
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func ReadOneItem(html string, t *testing.T) *Item {
|
||||||
|
p := NewParser(strings.NewReader(html))
|
||||||
|
|
||||||
|
data, err := p.Parse()
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("Expected no error but got %d", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if data == nil {
|
||||||
|
t.Errorf("Expected non-nil data")
|
||||||
|
}
|
||||||
|
|
||||||
|
return data.items[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func TestRead(t *testing.T) {
|
||||||
|
html := `
|
||||||
|
<div itemscope>
|
||||||
|
<p>My name is <span itemprop="name">Elizabeth</span>.</p>
|
||||||
|
</div>`
|
||||||
|
|
||||||
|
item := ReadOneItem(html, t)
|
||||||
|
|
||||||
|
if item.properties["name"][0].(string) != "Elizabeth" {
|
||||||
|
t.Errorf("Property value not found")
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func TestReadActuallyParses(t *testing.T) {
|
||||||
|
html := `
|
||||||
|
<div itemscope>
|
||||||
|
<p>My name is <span itemprop="name">Daniel</span>.</p>
|
||||||
|
</div>`
|
||||||
|
item := ReadOneItem(html, t)
|
||||||
|
|
||||||
|
if item.properties["name"][0].(string) != "Daniel" {
|
||||||
|
t.Errorf("Property value not found")
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func TestReadThreeProps(t *testing.T) {
|
||||||
|
html := `
|
||||||
|
<div itemscope>
|
||||||
|
<p>My name is <span itemprop="name">Neil</span>.</p>
|
||||||
|
<p>My band is called <span itemprop="band">Four Parts Water</span>.</p>
|
||||||
|
<p>I am <span itemprop="nationality">British</span>.</p>
|
||||||
|
</div>`
|
||||||
|
|
||||||
|
item := ReadOneItem(html, t)
|
||||||
|
|
||||||
|
if item.properties["name"][0].(string) != "Neil" {
|
||||||
|
t.Errorf("Property value not found")
|
||||||
|
}
|
||||||
|
|
||||||
|
if item.properties["band"][0].(string) != "Four Parts Water" {
|
||||||
|
t.Errorf("Property value not found")
|
||||||
|
}
|
||||||
|
|
||||||
|
if item.properties["nationality"][0].(string) != "British" {
|
||||||
|
t.Errorf("Property value not found")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func TestReadImgSrc(t *testing.T) {
|
||||||
|
html := `
|
||||||
|
<div itemscope>
|
||||||
|
<img itemprop="image" src="google-logo.png" alt="Google">
|
||||||
|
</div>`
|
||||||
|
|
||||||
|
item := ReadOneItem(html, t)
|
||||||
|
|
||||||
|
if item.properties["image"][0].(string) != "google-logo.png" {
|
||||||
|
t.Errorf("Property value not found")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestReadAHref(t *testing.T) {
|
||||||
|
html := `
|
||||||
|
<div itemscope>
|
||||||
|
<a itemprop="image" href="google-logo.png">foo</a>
|
||||||
|
</div>`
|
||||||
|
|
||||||
|
item := ReadOneItem(html, t)
|
||||||
|
|
||||||
|
if item.properties["image"][0].(string) != "google-logo.png" {
|
||||||
|
t.Errorf("Property value not found")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestReadAreaHref(t *testing.T) {
|
||||||
|
html := `
|
||||||
|
<div itemscope><map name="shapes">
|
||||||
|
<area itemprop="foo" href="target.html" shape=rect coords="50,50,100,100">
|
||||||
|
|
||||||
|
</map></div>`
|
||||||
|
|
||||||
|
item := ReadOneItem(html, t)
|
||||||
|
|
||||||
|
if item.properties["foo"][0].(string) != "target.html" {
|
||||||
|
t.Errorf("Property value not found")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestReadLinkHref(t *testing.T) {
|
||||||
|
html := `
|
||||||
|
<div itemscope>
|
||||||
|
<link itemprop="foo" rel="author" href="target.html">
|
||||||
|
</div>`
|
||||||
|
|
||||||
|
item := ReadOneItem(html, t)
|
||||||
|
|
||||||
|
if item.properties["foo"][0].(string) != "target.html" {
|
||||||
|
t.Errorf("Property value not found")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestReadAudioSrc(t *testing.T) {
|
||||||
|
html := `
|
||||||
|
<div itemscope>
|
||||||
|
<audio itemprop="foo" src="target"></audio>
|
||||||
|
</div>`
|
||||||
|
|
||||||
|
item := ReadOneItem(html, t)
|
||||||
|
|
||||||
|
if item.properties["foo"][0].(string) != "target" {
|
||||||
|
t.Errorf("Property value not found")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestReadSourceSrc(t *testing.T) {
|
||||||
|
html := `
|
||||||
|
<div itemscope>
|
||||||
|
<source itemprop="foo" src="target"></source>
|
||||||
|
</div>`
|
||||||
|
|
||||||
|
item := ReadOneItem(html, t)
|
||||||
|
|
||||||
|
if item.properties["foo"][0].(string) != "target" {
|
||||||
|
t.Errorf("Property value not found")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func TestReadVideoSrc(t *testing.T) {
|
||||||
|
html := `
|
||||||
|
<div itemscope>
|
||||||
|
<video itemprop="foo" src="target"></video>
|
||||||
|
</div>`
|
||||||
|
|
||||||
|
item := ReadOneItem(html, t)
|
||||||
|
|
||||||
|
if item.properties["foo"][0].(string) != "target" {
|
||||||
|
t.Errorf("Property value not found")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestReadEmbedSrc(t *testing.T) {
|
||||||
|
html := `
|
||||||
|
<div itemscope>
|
||||||
|
<embed itemprop="foo" src="target"></embed>
|
||||||
|
</div>`
|
||||||
|
|
||||||
|
item := ReadOneItem(html, t)
|
||||||
|
|
||||||
|
if item.properties["foo"][0].(string) != "target" {
|
||||||
|
t.Errorf("Property value not found")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestReadTrackSrc(t *testing.T) {
|
||||||
|
html := `
|
||||||
|
<div itemscope>
|
||||||
|
<track itemprop="foo" src="target"></track>
|
||||||
|
</div>`
|
||||||
|
|
||||||
|
item := ReadOneItem(html, t)
|
||||||
|
|
||||||
|
if item.properties["foo"][0].(string) != "target" {
|
||||||
|
t.Errorf("Property value not found")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestReadIFrameSrc(t *testing.T) {
|
||||||
|
html := `
|
||||||
|
<div itemscope>
|
||||||
|
<iframe itemprop="foo" src="target"></iframe>
|
||||||
|
</div>`
|
||||||
|
|
||||||
|
item := ReadOneItem(html, t)
|
||||||
|
|
||||||
|
if item.properties["foo"][0].(string) != "target" {
|
||||||
|
t.Errorf("Property value not found")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestReadDataValue(t *testing.T) {
|
||||||
|
html := `
|
||||||
|
<h1 itemscope>
|
||||||
|
<data itemprop="product-id" value="9678AOU879">The Instigator 2000</data>
|
||||||
|
</h1>`
|
||||||
|
|
||||||
|
item := ReadOneItem(html, t)
|
||||||
|
|
||||||
|
if item.properties["product-id"][0].(string) != "9678AOU879" {
|
||||||
|
t.Errorf("Property value not found")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestReadTimeDatetime(t *testing.T) {
|
||||||
|
html := `
|
||||||
|
<h1 itemscope>
|
||||||
|
I was born on <time itemprop="birthday" datetime="2009-05-10">May 10th 2009</time>.
|
||||||
|
</h1>`
|
||||||
|
|
||||||
|
item := ReadOneItem(html, t)
|
||||||
|
|
||||||
|
if item.properties["birthday"][0].(string) != "2009-05-10" {
|
||||||
|
t.Errorf("Property value not found")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue