-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathAdventOfCode.cfc
More file actions
168 lines (147 loc) · 5.74 KB
/
AdventOfCode.cfc
File metadata and controls
168 lines (147 loc) · 5.74 KB
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
<cfcomponent output="false">
<cffunction name="isPrime" access="public" returntype="boolean" output="false">
<cfargument name="n" type="numeric" required="true" />
<cfif arguments.n lte 1>
<cfreturn false />
<cfelseif arguments.n lte 3>
<cfreturn true />
<cfelseif arguments.n % 2 eq 0 || arguments.n % 3 eq 0>
<cfreturn false />
</cfif>
<cfset var i = 5 />
<cfloop condition="i * i lte arguments.n">
<cfif arguments.n % i eq 0 || arguments.n % (i + 2) eq 0>
<cfreturn false />
</cfif>
<cfset i += 6 />
</cfloop>
<cfreturn true />
</cffunction>
<cffunction name="arrayOfNumbersToHexString" access="public" returntype="string" output="false">
<cfargument name="arrayOfNumbers" type="array" required="true" />
<cfset var hexString = '' />
<cfset var number = '' />
<cfset var hs = '' />
<cfloop array="#arguments.arrayOfNumbers#" item="number">
<cfset hs = FormatBaseN(number, 16) />
<cfif Len(hs) eq 1>
<cfset hs = '0' & hs />
</cfif>
<cfset hexString &= hs />
</cfloop>
<cfreturn LCase(hexString) />
</cffunction>
<cffunction name="getKnotHash" access="public" returntype="string" output="false">
<cfargument name="input" required="true" />
<cfargument name="numberOfNumbers" default="256" />
<cfargument name="numberOfRounds" default="64" />
<cfargument name="numberOfBlocks" default="16" />
<cfset var numbers = [] />
<cfset var i = '' />
<cfloop from="0" to="#arguments.numberOfNumbers - 1#" index="i">
<cfset ArrayAppend(numbers, i) />
</cfloop>
<cfset var lengths = [] />
<cfset var inputIndex = '' />
<cfloop from="1" to="#Len(input)#" index="inputIndex">
<cfset ArrayAppend(lengths, Asc(Mid(arguments.input, inputIndex, 1))) />
</cfloop>
<cfset ArrayAppend(lengths, ListToArray('17,31,73,47,23'), true) />
<cfset var pos = 0 />
<cfset var skipSize = 0 />
<cfset var roundIndex = '' />
<cfset var length = '' />
<cfset var offset = '' />
<cfset var indexFromStart = '' />
<cfset var indexFromEnd = '' />
<cfloop from="1" to="#arguments.numberOfRounds#" index="roundIndex">
<cfloop array="#lengths#" item="length">
<cfloop from="0" to="#Floor(length / 2) - 1#" index="offset">
<cfset indexFromStart = (pos + offset) % arguments.numberOfNumbers + 1 />
<cfset indexFromEnd = (pos + length - 1 - offset) % arguments.numberOfNumbers + 1 />
<cfset ArraySwap(numbers, indexFromStart, indexFromEnd) />
</cfloop>
<cfset pos = (pos + length + skipSize) % arguments.numberOfNumbers />
<cfset skipSize++ />
</cfloop>
</cfloop>
<cfset var numbersPerBlock = arguments.numberOfNumbers / arguments.numberOfBlocks />
<cfset var denseHash = [] />
<cfset var blockIndex = '' />
<cfset var numberIndex = '' />
<cfset var blockValue = '' />
<cfset var number = '' />
<cfloop from="1" to="#arguments.numberOfBlocks#" index="blockIndex">
<cfset blockValue = 0 />
<cfloop from="1" to="#numbersPerBlock#" index="numberIndex">
<cfset number = numbers[(blockIndex - 1) * numbersPerBlock + numberIndex] />
<cfset blockValue = BitXOR(blockValue, number) />
</cfloop>
<cfset ArrayAppend(denseHash, blockValue) />
</cfloop>
<cfreturn arrayOfNumbersToHexString(denseHash) />
</cffunction>
<cffunction name="dance" access="public" returntype="string" output="false">
<cfargument name="programs" type="string" required="true" />
<cfargument name="moves" type="string" required="true" />
<cfset var numberOfPrograms = Len(arguments.programs) />
<cfset var programsByName = {} />
<cfset var programsByPos = {} />
<cfset var pos = '' />
<cfset var program = '' />
<cfloop from="0" to="#numberOfPrograms - 1#" index="pos">
<cfset program = { pos = pos, name = Mid(arguments.programs, pos + 1, 1) } />
<cfset programsByName[program.name] = program />
<cfset programsByPos[program.pos] = program />
</cfloop>
<cfset var move = '' />
<cfset var name = '' />
<cfset var pos1 = '' />
<cfset var pos2 = '' />
<cfset var program1 = '' />
<cfset var program2 = '' />
<cfset var name1 = '' />
<cfset var name2 = '' />
<cfloop list="#arguments.moves#" item="move">
<cfswitch expression="#Left(move, 1)#">
<cfcase value="s"><!--- shift to right --->
<cfloop collection="#programsByName#" item="name">
<cfset program = programsByName[name] />
<cfset program.pos = (program.pos + Mid(move, 2, 100)) mod numberOfPrograms />
<cfset programsByPos[program.pos] = program />
</cfloop>
</cfcase>
<cfcase value="x"> <!--- swap by pos --->
<cfset pos1 = ListGetAt(Mid(move, 2, 100), 1, '/') />
<cfset pos2 = ListGetAt(Mid(move, 2, 100), 2, '/') />
<cfset program1 = programsByPos[pos1] />
<cfset program2 = programsByPos[pos2] />
<cfset program1.pos = pos2 />
<cfset program2.pos = pos1 />
<cfset programsByPos[program1.pos] = program1 />
<cfset programsByPos[program2.pos] = program2 />
</cfcase>
<cfcase value="p"> <!--- swap by name --->
<cfset name1 = ListGetAt(Mid(move, 2, 100), 1, '/') />
<cfset name2 = ListGetAt(Mid(move, 2, 100), 2, '/') />
<cfset program1 = programsByName[name1] />
<cfset program2 = programsByName[name2] />
<cfset pos1 = program1.pos />
<cfset pos2 = program2.pos />
<cfset program1.pos = pos2 />
<cfset program2.pos = pos1 />
<cfset programsByPos[program1.pos] = program1 />
<cfset programsByPos[program2.pos] = program2 />
</cfcase>
<cfdefaultcase>
<cfthrow message="Unexpected move: #move#" />
</cfdefaultcase>
</cfswitch>
</cfloop>
<cfset var programsAfterDance = '' />
<cfloop from="0" to="#numberOfPrograms - 1#" index="pos">
<cfset programsAfterDance &= programsByPos[pos].name />
</cfloop>
<cfreturn programsAfterDance />
</cffunction>
</cfcomponent>