-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.js
172 lines (148 loc) · 4.79 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
'use strict'
// predict version based on changes in the css ast
// the general idea here is that is that if unique selecors in prev version
// are equal to the changed version, then the change is a 'patch', if there
// are more selectors in the new versio the change is a 'minor' and if there
// are some selectors from the old version not present in the changed version
// the change is a 'major'.
var css = require('css');
var colors = require("colors/safe");
var _ = require('underscore');
var deepEqual = require('deep-equal')
const DIFF_ADDED = 0;
const DIFF_REMOVED = 1;
const DIFF_FORMATS= [{color:colors.green, prefix: '+ '}, {color:colors.red, prefix:'- '}]
module.exports = function (old, changed, options){
try{
var oldAst = css.parse(old || '');
var changedAst = css.parse(changed || '');
if(areProgramaticallyEqual(oldAst.stylesheet.rules,changedAst.stylesheet.rules)){
return null;
} else {
var prediction = predictChangeType(oldAst, changedAst, options || {});
return prediction;
}
} catch(e){
console.log('Oh no, something went wrong during css comparison.');
// console.error(e, e.stack.split("\n"));
}
};
function areProgramaticallyEqual(oldRuleSet, changedRuleSet){
var oldMapped = oldRuleSet.map(ruleDiffMap)
var changedMapped = changedRuleSet.map(ruleDiffMap)
return deepEqual(oldMapped, changedMapped)
}
function ruleDiffMap(elem){
// returns a subset of the css needed for successfull deep comparison even if the
// rules have different whitespace
return {
'type': elem.type,
'declarations': elem.declarations ? elem.declarations.map(declarationDiffMap) : null,
'selectors': elem.selectors || null
}
}
function declarationDiffMap(elem){
return {
'type': elem.type,
'property': elem.property,
'value': elem.value
}
}
function predictChangeType(oldAst, changedAst, options){
var oldUniques = extractUniqueCssSelectors(oldAst);
var changedUniques = extractUniqueCssSelectors(changedAst);
// note that underscores difference is not symetric
// but only removes x elements from the y array in difference(y,x)
var oldsNotInChanged = _.difference(oldUniques, changedUniques);
var changedsNotInOld = _.difference(changedUniques, oldUniques);
// log changes that are detected
if(changedsNotInOld.length > 0){
changedsNotInOld.forEach(function(elem){
logElem(DIFF_ADDED, elem, options)
});
}
if(oldsNotInChanged.length > 0){
oldsNotInChanged.forEach(function(elem){
logElem(DIFF_REMOVED, elem, options)
});
}
logChanged(oldAst, changedAst, options)
if( oldsNotInChanged.length === 0 && changedsNotInOld.length === 0){
// the files contain an eqaul amount of rules
return 'patch';
}
if(oldsNotInChanged.length == 0 && changedsNotInOld.length > 0){
// there are more rules in the changed file
return 'minor';
}
if (oldsNotInChanged.length > 0){
// there is something defined in the old file not present in the new file
return 'major';
}
throw "Unable to predict version";
}
function jsonLog(thing){
console.log(JSON.stringify(thing))
}
function logChanged(a, b, options){
try{
if(!options.verbose) return
var aRules = a.stylesheet.rules
var bRules = b.stylesheet.rules
var aBySelectors = getDelclerationsBySelectors(aRules)
var bBySelectors = getDelclerationsBySelectors(bRules)
var selectors = Object.keys(aBySelectors)
selectors.forEach( selector =>{
// we only care about changed rules
if(!bBySelectors[selector]) return
var eq = deepEqual( aBySelectors[selector], bBySelectors[selector])
if(eq) return
console.log(colors.yellow('~ '+ selector))
compareDeclerations(aBySelectors[selector][0], bBySelectors[selector][0])
})
} catch(e){
console.log('Error in css-semver verbose mode: ', e)
}
}
function compareDeclerations(a, b){
var aKeys = Object.keys(a)
var bKeys = Object.keys(b)
var allKeys = new Set(aKeys.concat(bKeys))
allKeys.forEach(key =>{
if(a[key] === b[key]) return;
console.log(colors.yellow(` ${key}: ${a[key]} -> ${b[key]}`))
})
}
function getDelclerationsBySelectors(rules){
if(!rules) return {}
var bySelectors = {}
rules.forEach((rule)=>{
if(!rule.selectors) return
rule.selectors.forEach(selector =>{
if(!bySelectors[selector]){
bySelectors[selector] = []
}
var elem = {}
rule.declarations.forEach(d => {
elem[d.property] = d.value
})
bySelectors[selector].push(elem)
})
})
return bySelectors
}
function extractUniqueCssSelectors(ast) {
var rules = ast.stylesheet.rules.filter(r => {return r.type === 'rule'});
var selectorsList = rules.map(r => {return r.selectors});
var selectors = _.flatten(selectorsList);
var uniques = _.uniq(selectors);
return uniques;
}
function logElem(type, elem, options){
if(options.verbose){
let format = DIFF_FORMATS[type];
let text = format.prefix + elem;
let output = format.color(text)
console.log(output);
}
}