test_forward.py 3.55 KB
Newer Older
rusty1s's avatar
rusty1s committed
1
2
3
4
5
6
from itertools import product

import pytest
import torch
import torch_scatter

rusty1s's avatar
rusty1s committed
7
8
9
10
11
12
from .utils import dtypes, devices, tensor

tests = [{
    'name': 'add',
    'src': [[2, 0, 1, 4, 3], [0, 2, 1, 3, 4]],
    'index': [[4, 5, 4, 2, 3], [0, 0, 2, 2, 1]],
rusty1s's avatar
rusty1s committed
13
    'dim': -1,
rusty1s's avatar
rusty1s committed
14
    'fill_value': 0,
rusty1s's avatar
rusty1s committed
15
    'expected': [[0, 0, 4, 3, 3, 0], [2, 4, 4, 0, 0, 0]],
rusty1s's avatar
rusty1s committed
16
17
18
}, {
    'name': 'add',
    'src': [[5, 2], [2, 5], [4, 3], [1, 3]],
rusty1s's avatar
rusty1s committed
19
    'index': [0, 1, 1, 0],
rusty1s's avatar
rusty1s committed
20
21
    'dim': 0,
    'fill_value': 0,
rusty1s's avatar
rusty1s committed
22
    'expected': [[6, 5], [6, 8]],
rusty1s's avatar
rusty1s committed
23
24
25
26
27
28
}, {
    'name': 'sub',
    'src': [[2, 0, 1, 4, 3], [0, 2, 1, 3, 4]],
    'index': [[4, 5, 4, 2, 3], [0, 0, 2, 2, 1]],
    'dim': -1,
    'fill_value': 9,
rusty1s's avatar
rusty1s committed
29
    'expected': [[9, 9, 5, 6, 6, 9], [7, 5, 5, 9, 9, 9]],
rusty1s's avatar
rusty1s committed
30
31
32
}, {
    'name': 'sub',
    'src': [[5, 2], [2, 2], [4, 2], [1, 3]],
rusty1s's avatar
rusty1s committed
33
    'index': [0, 1, 1, 0],
rusty1s's avatar
rusty1s committed
34
35
    'dim': 0,
    'fill_value': 9,
rusty1s's avatar
rusty1s committed
36
    'expected': [[3, 4], [3, 5]],
rusty1s's avatar
rusty1s committed
37
38
39
40
41
42
}, {
    'name': 'mul',
    'src': [[2, 0, 1, 4, 3], [0, 2, 1, 3, 4]],
    'index': [[4, 5, 4, 2, 3], [0, 0, 2, 2, 1]],
    'dim': -1,
    'fill_value': 1,
rusty1s's avatar
rusty1s committed
43
    'expected': [[1, 1, 4, 3, 2, 0], [0, 4, 3, 1, 1, 1]],
rusty1s's avatar
rusty1s committed
44
45
46
}, {
    'name': 'mul',
    'src': [[5, 2], [2, 5], [4, 3], [1, 3]],
rusty1s's avatar
rusty1s committed
47
    'index': [0, 1, 1, 0],
rusty1s's avatar
rusty1s committed
48
49
    'dim': 0,
    'fill_value': 1,
rusty1s's avatar
rusty1s committed
50
    'expected': [[5, 6], [8, 15]],
rusty1s's avatar
rusty1s committed
51
52
53
54
55
56
}, {
    'name': 'div',
    'src': [[2, 1, 1, 4, 2], [1, 2, 1, 2, 4]],
    'index': [[4, 5, 4, 2, 3], [0, 0, 2, 2, 1]],
    'dim': -1,
    'fill_value': 1,
rusty1s's avatar
rusty1s committed
57
    'expected': [[1, 1, 0.25, 0.5, 0.5, 1], [0.5, 0.25, 0.5, 1, 1, 1]],
rusty1s's avatar
rusty1s committed
58
59
60
}, {
    'name': 'div',
    'src': [[4, 2], [2, 1], [4, 2], [1, 2]],
rusty1s's avatar
rusty1s committed
61
    'index': [0, 1, 1, 0],
rusty1s's avatar
rusty1s committed
62
63
    'dim': 0,
    'fill_value': 1,
rusty1s's avatar
rusty1s committed
64
    'expected': [[0.25, 0.25], [0.125, 0.5]],
rusty1s's avatar
rusty1s committed
65
66
67
68
}, {
    'name': 'mean',
    'src': [[2, 0, 1, 4, 3], [0, 2, 1, 3, 4]],
    'index': [[4, 5, 4, 2, 3], [0, 0, 2, 2, 1]],
rusty1s's avatar
rusty1s committed
69
    'dim': -1,
rusty1s's avatar
rusty1s committed
70
    'fill_value': 0,
rusty1s's avatar
rusty1s committed
71
    'expected': [[0, 0, 4, 3, 1.5, 0], [1, 4, 2, 0, 0, 0]],
rusty1s's avatar
rusty1s committed
72
73
74
}, {
    'name': 'mean',
    'src': [[5, 2], [2, 5], [4, 3], [1, 3]],
rusty1s's avatar
rusty1s committed
75
    'index': [0, 1, 1, 0],
rusty1s's avatar
rusty1s committed
76
77
    'dim': 0,
    'fill_value': 0,
rusty1s's avatar
rusty1s committed
78
    'expected': [[3, 2.5], [3, 4]],
rusty1s's avatar
rusty1s committed
79
80
81
82
83
84
85
}, {
    'name': 'max',
    'src': [[2, 0, 1, 4, 3], [0, 2, 1, 3, 4]],
    'index': [[4, 5, 4, 2, 3], [0, 0, 2, 2, 1]],
    'dim': -1,
    'fill_value': 0,
    'expected': [[0, 0, 4, 3, 2, 0], [2, 4, 3, 0, 0, 0]],
rusty1s's avatar
rusty1s committed
86
    'expected_arg': [[-1, -1, 3, 4, 0, 1], [1, 4, 3, -1, -1, -1]],
rusty1s's avatar
rusty1s committed
87
88
89
}, {
    'name': 'max',
    'src': [[5, 2], [2, 5], [4, 3], [1, 3]],
rusty1s's avatar
rusty1s committed
90
    'index': [0, 1, 1, 0],
rusty1s's avatar
rusty1s committed
91
92
93
    'dim': 0,
    'fill_value': 0,
    'expected': [[5, 3], [4, 5]],
rusty1s's avatar
rusty1s committed
94
    'expected_arg': [[0, 3], [2, 1]],
rusty1s's avatar
rusty1s committed
95
96
97
98
99
100
101
}, {
    'name': 'min',
    'src': [[2, 0, 1, 4, 3], [0, 2, 1, 3, 4]],
    'index': [[4, 5, 4, 2, 3], [0, 0, 2, 2, 1]],
    'dim': -1,
    'fill_value': 9,
    'expected': [[9, 9, 4, 3, 1, 0], [0, 4, 1, 9, 9, 9]],
rusty1s's avatar
rusty1s committed
102
    'expected_arg': [[-1, -1, 3, 4, 2, 1], [0, 4, 2, -1, -1, -1]],
rusty1s's avatar
rusty1s committed
103
104
105
}, {
    'name': 'min',
    'src': [[5, 2], [2, 5], [4, 3], [1, 3]],
rusty1s's avatar
rusty1s committed
106
    'index': [0, 1, 1, 0],
rusty1s's avatar
rusty1s committed
107
108
109
    'dim': 0,
    'fill_value': 9,
    'expected': [[1, 2], [2, 3]],
rusty1s's avatar
rusty1s committed
110
    'expected_arg': [[3, 0], [1, 2]],
rusty1s's avatar
rusty1s committed
111
112
113
114
115
116
117
}]


@pytest.mark.parametrize('test,dtype,device', product(tests, dtypes, devices))
def test_forward(test, dtype, device):
    src = tensor(test['src'], dtype, device)
    index = tensor(test['index'], torch.long, device)
rusty1s's avatar
rusty1s committed
118
    expected = tensor(test['expected'], dtype, device)
rusty1s's avatar
rusty1s committed
119
120

    op = getattr(torch_scatter, 'scatter_{}'.format(test['name']))
rusty1s's avatar
rusty1s committed
121
    out = op(src, index, test['dim'], fill_value=test['fill_value'])
rusty1s's avatar
rusty1s committed
122

rusty1s's avatar
rusty1s committed
123
124
125
126
127
    if isinstance(out, tuple):
        assert out[0].tolist() == expected.tolist()
        assert out[1].tolist() == test['expected_arg']
    else:
        assert out.tolist() == expected.tolist()