-
-
Notifications
You must be signed in to change notification settings - Fork 27
/
Copy pathelsa-analyser.el
1201 lines (1109 loc) · 50.4 KB
/
elsa-analyser.el
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
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
;; -*- lexical-binding: t -*-
(require 'elsa-explainer)
(require 'elsa-reader)
(require 'elsa-check)
(require 'elsa-error)
(require 'elsa-types)
(require 'elsa-type-helpers)
(require 'elsa-type-algebra)
(require 'elsa-english)
(require 'elsa-state)
(require 'elsa-scope)
(require 'elsa-functions)
;; (elsa--arglist-to-arity :: (function ((or (list (or symbol (cons symbol mixed))) t string)) (cons int (or int (const many) (const undefined)))))
(defun elsa--arglist-to-arity (arglist)
"Return minimal and maximal number of arguments ARGLIST supports.
If there is a &rest argument we represent the upper infinite
number by symbol 'many.
If the arglist is `t', use a fallback 0 to infinity and return
'undefined for max. This has the same semantics as 'many but
adds the information that a fallback was used."
(let ((min 0)
(max 0))
(cond
;; TODO: this is a dumb fallback to basically mixed... -> mixed
;; This should be solved once we do the recursive `require' walks.
((or (eq arglist t) (stringp arglist))
(cons 0 'undefined))
(t
(while (and arglist (not (memq (car arglist) '(&optional &rest))))
(cl-incf min)
(!cdr arglist))
(when (eq (car arglist) '&optional)
(!cdr arglist))
(setq max min)
(while (and arglist (not (eq (car arglist) '&rest)))
(cl-incf max)
(!cdr arglist))
(when (eq (car arglist) '&rest)
(setq max 'many))
(cons min max)))))
;; (elsa-fn-arity :: (function ((class elsa-state) (or (list (or symbol (cons symbol mixed))) symbol)) (cons int (or int (const many) (const unevalled) (const undefined)))))
(defun elsa-fn-arity (state def)
(cond
;; directly provided arglist
((listp def)
(elsa--arglist-to-arity def))
;; symbol can point to a lambda, closure or subr
((symbolp def)
(cond
((elsa-state-get-defun state def)
(let ((dfn (elsa-state-get-defun state def)))
(if (slot-boundp dfn 'arglist)
(elsa--arglist-to-arity (oref (elsa-state-get-defun state def) arglist))
(elsa--arglist-to-arity t))))
((indirect-function def)
(let ((fn (indirect-function def)))
(cond
((subrp fn) (subr-arity fn))
((and (byte-code-function-p fn) (integerp (aref fn 0)))
(func-arity fn))
((eq (car-safe fn) 'lambda)
(elsa--arglist-to-arity (nth 1 fn)))
((eq (car-safe fn) 'closure)
(elsa--arglist-to-arity (nth 2 fn)))
(t (elsa--arglist-to-arity t)))))
;; fallback
(t (elsa--arglist-to-arity t))))))
(defun elsa--analyse-float (_form _scope _state)
nil)
(defun elsa--analyse-integer (_form _scope _state)
nil)
(defun elsa--analyse-keyword (_form _scope _state)
nil)
;; (elsa--analyse-symbol :: (function ((class elsa-form-symbol) mixed (class elsa-state)) mixed))
(defun elsa--analyse-symbol (form scope state)
(let* ((name (oref form name))
(type (cond
((elsa-state-quoted-p state)
(elsa-const-type
:type (elsa-type-symbol)
:value name))
((eq name t) (elsa-make-type t))
((eq name nil) (elsa-make-type nil))
((-when-let (var (elsa-scope-get-var scope form))
(clone (oref var type))))
((and (elsa-state-get-defvar state name)
(elsa-get-type (elsa-state-get-defvar state name))))
(t (elsa-make-type unbound)))))
(oset form type type)
(unless (or (memq name '(t nil))
(elsa-state-quoted-p state))
(oset form narrow-types
(list (elsa-variable :name name
:type (elsa-type-make-non-nullable type)))))))
(defun elsa--analyse-vector (_form _scope _state)
nil)
(defun elsa--analyse-string (_form _scope _state)
nil)
(defun elsa--analyse-variable-from-binding (binding scope state)
"Analyze let-like BINDING and return the resulting variable.
The BINDING should have one of the following forms:
- place ; initial is nil
- (place) ; initial is nil
- (place initial-value)"
(let* ((annotation (when (elsa-form-p binding)
(oref binding annotation)))
(annotation-type (and annotation
(elsa--make-type (nth 2 annotation)))))
(cond
((or (listp binding)
(elsa-form-list-p binding))
(-let [(var source) (elsa-form-sequence binding)]
(when source
(elsa--analyse-form source scope state))
(when (elsa-form-symbol-p var)
(if (not source)
(elsa-variable
:name (oref var name) :type (elsa-type-nil))
(elsa-variable
:name (oref var name) :type (or annotation-type
(oref source type)))))))
((elsa-form-symbol-p binding)
(elsa-variable :name (oref binding name) :type (elsa-type-nil)))
(t nil))))
(defun elsa--analyse:let (form scope state)
(let* ((new-vars nil)
(bindings-maybe (elsa-cadr form))
(bindings
;; If the binding list argument to let is not a list, we are
;; probably dealing with some pcase macro or another special
;; use-case. So we simply skip the binding. It can also be
;; empty (that is nil).
(when (elsa-form-sequence-p bindings-maybe)
(elsa-form-sequence bindings-maybe)))
(body (cddr (oref form sequence))))
;; TODO: move this to extension?
(-each bindings
(lambda (binding)
(push (elsa--analyse-variable-from-binding binding scope state) new-vars)))
(-each new-vars (lambda (v) (elsa-scope-add-var scope v)))
(if (not body)
(oset form type (elsa-type-nil))
(elsa--analyse-body body scope state)
(oset form type (oref (-last-item body) type)))
(-each new-vars (lambda (v) (elsa-scope-remove-var scope v)))))
(defun elsa--analyse:let* (form scope state)
(let* ((new-vars nil)
(bindings-maybe (elsa-cadr form))
(bindings
;; If the binding list argument to let is not a list, we are
;; probably dealing with some pcase macro or another special
;; use-case. So we simply skip the binding. It can also be
;; empty (that is nil).
(when (elsa-form-sequence-p bindings-maybe)
(elsa-form-sequence bindings-maybe)))
(body (cddr (oref form sequence))))
(-each bindings
(lambda (binding)
(let ((variable (elsa--analyse-variable-from-binding binding scope state)))
(push variable new-vars)
(elsa-scope-add-var scope variable))))
(if (not body)
(oset form type (elsa-type-nil))
(elsa--analyse-body body scope state)
(oset form type (oref (-last-item body) type)))
(-each new-vars (lambda (v) (elsa-scope-remove-var scope v)))))
;; TODO: add analysis of the reachability of body
(defalias 'elsa--analyse:when-let* 'elsa--analyse:let*)
(defun elsa--analyse:if (form scope state)
(let ((condition (elsa-nth 1 form))
(true-body (elsa-nth 2 form))
(false-body (elsa-nthcdr 3 form))
(mutated-vars-true nil)
(mutated-vars-false nil))
(elsa--analyse-form condition scope state)
(elsa-with-reachability state (elsa-type-is-non-nil condition)
(elsa-save-scope scope
(elsa-scope-narrow-var scope (oref condition narrow-types))
(elsa--analyse-form true-body scope state)
(setq mutated-vars-true (elsa-scope-get-assigned-vars scope))))
(elsa-with-reachability state (elsa-type-is-nil condition)
(elsa-save-scope scope
(elsa-scope-narrow-var scope (oref condition narrow-types)
'elsa-variable-diff)
(elsa--analyse-body false-body scope state)
(setq mutated-vars-false (elsa-scope-get-assigned-vars scope))))
(let ((condition-is-nil (elsa-type-is-nil condition))
(to-merge))
(cond
((trinary-true-p condition-is-nil)
(setq to-merge (--each (-map 'clone mutated-vars-false)
(oset it assigned (trinary-maybe)))))
((trinary-false-p condition-is-nil)
(setq to-merge (--each (-map 'clone mutated-vars-true)
(oset it assigned (trinary-maybe)))))
((trinary-maybe-p condition-is-nil)
(-let (((true false both) (elsa--variables-partition mutated-vars-true mutated-vars-false)))
(-each (-concat true false)
(lambda (var)
(push (elsa-variable
:name (elsa-get-name var)
:type (-if-let (scope-var (elsa-scope-get-var scope var))
(elsa-type-sum var scope-var)
(clone (elsa-get-type var)))
:assigned (trinary-add-maybe (oref var assigned))
:read (trinary-add-maybe (oref var read)))
to-merge)))
(setq to-merge (-concat both to-merge)))))
(elsa-variables-merge-to-scope to-merge scope))
(let ((true-result-type (elsa-get-type true-body))
(false-result-type (if false-body
(elsa-get-type (-last-item false-body))
(elsa-type-nil))))
(oset form type
(cond
((trinary-true-p (elsa-type-is-non-nil condition))
true-result-type)
((trinary-true-p (elsa-type-is-nil condition))
false-result-type)
(t (elsa-type-sum true-result-type false-result-type)))))))
(defun elsa--analyse:setq (form scope state)
(let* ((args (elsa-cdr form))
(assignments (-partition 2 args)))
(-each assignments
(lambda (assignment)
(let* ((place (car assignment))
(val (cadr assignment))
(var (elsa-scope-get-var scope place))
(special-var (elsa-state-get-defvar state (elsa-get-name place)))
(special-var-type (and special-var (elsa-get-type special-var))))
(elsa--analyse-form val scope state)
(elsa-scope-assign-var scope
(elsa-variable :name (elsa-get-name place)
:type (oref val type)))
(unless (or var special-var)
(elsa-state-add-message state
(elsa-make-warning place
"Assigning to free variable %s"
(symbol-name (elsa-get-name place)))))
(-when-let (type (or
(and var (elsa-get-type var))
special-var-type))
(if (elsa-readonly-type-p type)
(elsa-state-add-message state
(elsa-make-error place
"Assignment to read-only variable %s"
(symbol-name (elsa-get-name place))))
(let ((var-type (cond
((elsa-const-type-p type)
(oref type type))
((elsa-type-nil-p type)
(elsa-type-mixed))
(t type))))
(unless (elsa-type-accept var-type val)
(elsa-state-add-message state
(elsa-make-error place
(elsa-with-temp-explainer explainer
(elsa-explain-and-indent explainer
("Variable %s expects `%s', got `%s'"
(symbol-name (elsa-get-name place))
(elsa-type-describe var-type)
(elsa-type-describe (elsa-get-type val)))
(elsa-type-accept var-type val explainer))
explainer)
:code "invalid-assignment")))))))))
(oset form type (oref (-last-item args) type))
(oset form narrow-types
(list (elsa-variable :name (elsa-get-name (car (-last-item assignments)))
:type (elsa-type-make-non-nullable
(elsa-get-type (cadr (-last-item assignments)))))))))
(defun elsa--analyse:cond (form scope state)
(let ((branches (cdr (oref form sequence)))
(return-type (elsa-type-empty))
(condition-reachable (trinary-true)))
(elsa-save-scope scope
(-each branches
(lambda (branch)
(let* ((branch-seq (elsa-form-sequence branch))
(head (car branch-seq))
(body (cdr branch-seq)))
(elsa-with-reachability state condition-reachable
(when head
(elsa-save-scope scope
(elsa--analyse-form head scope state)
(elsa-scope-narrow-var scope (oref head narrow-types))
(elsa-with-reachability state (elsa-type-is-non-nil head)
(elsa--analyse-body body scope state)))
(elsa-scope-narrow-var scope (oref head narrow-types)
'elsa-variable-diff)
(when (trinary-possible-p condition-reachable)
(setq return-type
(elsa-type-sum return-type (-last-item branch-seq))))
(setq condition-reachable
(trinary-and
condition-reachable
(elsa-type-is-nil head)))))))))
(when (trinary-possible-p condition-reachable)
(setq return-type (elsa-type-make-nullable return-type)))
(oset form type return-type)))
(defun elsa--analyse:condition-case (form scope state)
(let* ((seq (oref form sequence))
(var (nth 1 seq))
(body (nth 2 seq))
(handlers (nthcdr 3 seq))
(return-type))
(elsa--analyse-form body scope state)
(setq return-type (oref body type))
(unless (eq (elsa-get-name var) 'nil)
(elsa-scope-add-var scope
(elsa-variable :name (elsa-get-name var))))
(--each handlers
(elsa--analyse-form it scope state)
(let ((last (-last-item (elsa-form-sequence it))))
(setq return-type (elsa-type-sum return-type last))))
(unless (eq (elsa-get-name var) 'nil)
(elsa-scope-remove-var scope var))
(oset form type return-type)))
(defun elsa--analyse:unwind-protect (form scope state)
(let* ((seq (oref form sequence))
(body (nth 1 seq))
(unwind-forms (nthcdr 2 seq))
(return-type))
(elsa--analyse-form body scope state)
(setq return-type (oref body type))
(elsa--analyse-body unwind-forms scope state)
(when-let ((last (-last-item unwind-forms)))
(setq return-type (elsa-type-sum return-type last))
(-when-let (grouped (elsa-variables-group-and-sum
(-non-nil
(-concat
(oref body narrow-types)
(oref last narrow-types)))))
(oset form narrow-types grouped)))
(oset form type return-type)))
(defun elsa--analyse:progn (form scope state)
(let* ((body (elsa-cdr form))
(last (-last-item (elsa-form-sequence form))))
(elsa--analyse-body body scope state)
(if body
(progn
(oset form type (elsa-get-type last))
(oset form narrow-types (oref last narrow-types)))
(oset form type (elsa-type-nil)))))
(defun elsa--analyse:save-excursion (form scope state)
(elsa--analyse:progn form scope state))
(defun elsa--analyse:save-restriction (form scope state)
(elsa--analyse:progn form scope state))
(defun elsa--analyse:save-current-buffer (form scope state)
(elsa--analyse:progn form scope state))
(defun elsa--analyse:track-mouse (form scope state)
(elsa--analyse:progn form scope state))
(defun elsa--analyse:prog1 (form scope state)
(let* ((body (elsa-cdr form))
(first (car body)))
(elsa--analyse-body body scope state)
(if first
(progn
(oset form type (elsa-get-type first))
(oset form narrow-types (oref first narrow-types)))
(oset form type (elsa-type-unbound)))))
(defun elsa--analyse:prog2 (form scope state)
(let* ((body (elsa-cdr form))
(second (cadr body)))
(elsa--analyse-body body scope state)
(if second
(progn
(oset form type (elsa-get-type second))
(oset form narrow-types (oref second narrow-types)))
(oset form type (elsa-type-unbound)))))
;; TODO: add reachability analysis
(defun elsa--analyse:or (form scope state)
(let* ((body (elsa-cdr form))
(return-type (elsa-type-nil))
(condition-reachable (trinary-true)))
(elsa-save-scope scope
(-each body
(lambda (arg)
(elsa-with-reachability state condition-reachable
(elsa--analyse-form arg scope state)
(when (trinary-possible-p condition-reachable)
(setq return-type
(elsa-type-sum return-type (oref arg type))))
(elsa-scope-narrow-var scope (oref arg narrow-types)
'elsa-variable-diff)
(setq condition-reachable
(trinary-and
condition-reachable
(elsa-type-is-nil arg)))))))
(-when-let (grouped (elsa-variables-group-and-sum
(->> body
(--filter (trinary-possible-p (elsa-form-reachable it)))
(--mapcat (oref it narrow-types))
(-non-nil))))
(oset form narrow-types grouped))
;; If the last form is definitely never reachable, the form always
;; succeds and we make it non-nullable.
(when (trinary-false-p condition-reachable)
(setq return-type (elsa-type-make-non-nullable return-type)))
(oset form type return-type)))
;; TODO: add tests for reachability
(defun elsa--analyse:and (form scope state)
(let* ((body (elsa-cdr form))
(return-type (elsa-type-t))
(condition-reachable (trinary-true)))
(elsa-save-scope scope
(-each body
(lambda (arg)
(elsa-with-reachability state condition-reachable
(elsa--analyse-form arg scope state)
(elsa-scope-narrow-var scope (oref arg narrow-types)
'elsa-variable-intersect)
(setq condition-reachable
(trinary-and
condition-reachable
(elsa-type-is-non-nil arg)))))))
(let ((narrowed-vars
;; get a union of all narrowed variables
(-uniq
(--map
(elsa-get-name it)
(-non-nil
(--mapcat (oref it narrow-types) body))))))
;; update narrowing of all the forms by the variables they don't
;; have assigned to their narrow-types
(-each body
(lambda (arg)
(let ((nil-always-accepts (elsa-type-could-accept
(elsa-type-nil)
(elsa-get-type arg)))
(narrow-types (oref arg narrow-types)))
;; add uncertainty to narrowing of all narrowed variables
;; if this form can fail.
(when (trinary-maybe-p nil-always-accepts)
(-each narrowed-vars
(lambda (var-name)
(when (--none? (eq var-name (elsa-get-name it)) narrow-types)
(push (elsa-variable
:name var-name
;; intersect with mixed will keep the
;; original type, but the (un)certainty
;; will propagate
:type (elsa-type-mixed)
:certainty (trinary-maybe))
narrow-types))))
(oset arg narrow-types narrow-types))))))
;; Group all the narrowing variables by name and intersect the
;; narrowing types, because we require all forms to be
;; simultaneously true. Set the resulting intersections on this
;; form.
(-when-let (grouped
(elsa-variables-group-and-intersect
(->> body
(--filter (trinary-possible-p (elsa-form-reachable it)))
(--mapcat (oref it narrow-types))
(-non-nil))))
(oset form narrow-types grouped))
(cond
((trinary-false-p condition-reachable)
(setq return-type (elsa-type-nil)))
(body
(setq return-type (oref (-last-item body) type))))
;; If the last form is not definitely reachable, it means we
;; could've failed somewhere inside the and form and we need to
;; make the return type nullable.
(unless (trinary-necessary-p condition-reachable)
(setq return-type (elsa-type-make-nullable return-type)))
(oset form type return-type)))
;; (elsa--get-default-function-types :: (function ((list (class elsa-form))) (list (class elsa-type))))
(defun elsa--get-default-function-types (args)
"Return a default list of types based on ARGS.
ARGS should be a list of `elsa-form's.
This function skips over special &optional and &rest markers and
collects all the arguments, turns &optional arguments into
nullables and the &rest argument into a variadic."
(-let ((optionalp nil)
(restp nil)
(types nil))
(elsa-form-foreach args
(lambda (arg)
(cond
((eq (elsa-get-name arg) '&optional)
(setq optionalp t))
((eq (elsa-get-name arg) '&rest)
(setq restp t)
(setq optionalp nil))
(t
(let ((type (or (elsa-get-type arg)
(elsa-type-mixed))))
(cond
(optionalp (push (elsa-type-sum
(elsa-make-type nil)
type)
types))
(restp (push (elsa-variadic-type
:item-type (elsa-type-sum
(elsa-make-type nil)
type))
types))
(t (push type types))))))))
(nreverse types)))
(defun elsa--make-function-type-from-annotation (annotation)
(and annotation
(if (symbolp (nth 2 annotation))
(elsa--make-type
`(function () ,(nth 2 annotation)))
(elsa--make-type (nth 2 annotation)))))
(defun elsa--analyse-defun-like-form (name args body form scope state)
"Analyse function or macro definition.
A definition is a form like `defun', `cl-defun', `cl-defmethod'
and similar forms.
Care needs to be taken to not use the `form' argument to extract
structural information, such as function name or arglist, because
this function can be called from different contexts. The
required information is instead passed as arguments.
This function tries to infer and validate the return type and the
argument types.
This function does not perform the call-site analysis, that is
handled by `elsa--analyse-function-like-invocation'."
(let* ((annotation (oref form annotation))
(annotation-type (elsa--make-function-type-from-annotation annotation))
(vars))
(when args
(-each-indexed (--remove
(memq (elsa-get-name it) '(&rest &optional))
args)
(lambda (index arg)
(let ((var
(elsa-variable
:name (elsa-get-name arg)
:type (cond
((not (elsa-type-mixed-p (elsa-get-type arg)))
(elsa-get-type arg))
((not annotation-type)
(elsa-type-mixed))
(t
(let ((expected-type (elsa-function-type-nth-arg
annotation-type
index)))
(or expected-type
(progn
(elsa-state-add-message state
(elsa-make-warning arg
"Argument no %d. `%s' is present in function declaration but is missing in Elsa type signature. Use of top-level implicit mixed is discouraged."
(1+ index)
(elsa-get-name arg)))
(elsa-type-mixed)))))))))
(push var vars)
(elsa-scope-add-var scope var)))))
;; FIXME: we must somehow add *this* overload before analyzing the
;; body, because it can be recursively called inside. But then we
;; won't know the return type...
(when body (elsa--analyse-body body scope state))
;; check if return type of defun corresponds with the last form of
;; the body
(let* ((body-return-type (if body (oref (-last-item body) type) (elsa-type-nil))))
;; Register the function
(let ((method (elsa-defun :name name
:defun-type (elsa-get-name form)
:type (or annotation-type
(elsa-function-type
:args (elsa--get-default-function-types args)
:return body-return-type))
:arglist (elsa-form-to-lisp args))))
(elsa-state-add-method state method))
;; (elsa-log "defun type %s" (elsa-tostring (oref (elsa-state-get-defun state name) type)))
;; (elsa-log "defun defgeneric-type %s" (elsa-tostring (oref (elsa-state-get-defun state name) defgeneric-type)))
;; (elsa-log "defun defun-type %s" (oref (elsa-state-get-defun state name) defun-type))
(let ((expected-return-type (or (elsa-type-get-return annotation-type)
(elsa-type-get-generic-return
(elsa-state-get-defun state name)))))
;; (elsa-log "expected-return-type %s" (elsa-tostring expected-return-type))
(when expected-return-type
(unless (elsa-type-accept expected-return-type body-return-type)
(elsa-state-add-message state
(elsa-make-error (elsa-car form)
(elsa-with-temp-explainer explainer
(elsa-explain-and-indent explainer
("Function is expected to return `%s' but returns `%s'"
(elsa-type-describe expected-return-type)
(elsa-type-describe body-return-type))
(elsa-type-accept expected-return-type body-return-type explainer))
explainer)
:code "incompatible-return-type"
:compact t))))))
(--each vars (elsa-scope-remove-var scope it))))
(defun elsa--analyse:defun (form scope state)
(let* ((sequence (elsa-form-sequence form))
(name (elsa-get-name (nth 1 sequence)))
(args (elsa-form-sequence (nth 2 sequence)))
(body (nthcdr 3 sequence)))
(elsa--analyse-defun-like-form name args body form scope state)))
(defun elsa--analyse-register-functionlike-declaration (name args state)
"Register functionlike declaration NAME with ARGS to STATE.
The registered object can be a `defun', `defmacro', or
`cl-defgeneric'."
(cond
((elsa-state-get-defun state name)
(-when-let (def (elsa-state-get-defun state name))
(unless (slot-boundp def 'arglist)
(oset def arglist (elsa-form-to-lisp args)))))
(t
(elsa-state-add-defun state
(elsa-defun :name name
:type (elsa-function-type
:args (elsa--get-default-function-types args)
:return (elsa-type-mixed))
:arglist (elsa-form-to-lisp args))))))
(defun elsa--analyse:elsa--form (form scope state)
"Analyse special marker for macroexpanded forms."
(let ((real-form (elsa-nth 2 form)))
(elsa--analyse-form real-form scope state)
(oset form type (oref real-form type))
(oset form narrow-types (oref real-form narrow-types))))
(defun elsa--analyse:defmacro (form _scope state)
"just skip for now, it's too complicated."
(let ((name (elsa-get-name (elsa-cadr form)))
(args (elsa-nth 2 form)))
(elsa--analyse-register-functionlike-declaration name args state)))
(defun elsa--analyse:defvar (form scope state)
"Analyze `defvar'.
We infer the type of the variable by the type of the default
form. This might not always make sense, for example if the value
is `nil' we will infer the type to be always nil, but it might be
a list or something else.
The user can provide a type annotation over the `defvar' form to
make it explicit and precise."
(let* ((name (elsa-nth 1 form))
(value (elsa-nth 2 form))
(var-name (elsa-get-name name))
(def (elsa-state-get-defvar state var-name))
(var-type (and def (elsa-get-type def))))
(when value
(elsa--analyse-form value scope state))
(if var-type
(unless (or (not value)
(elsa-type-assignable-p var-type (elsa-get-type value)))
(elsa-state-add-message state
(elsa-make-error value
(elsa-with-temp-explainer explainer
(elsa-explain-and-indent explainer
("Variable %s expects `%s', got `%s'"
var-name
(elsa-type-describe var-type)
(elsa-type-describe (elsa-get-type value)))
(elsa-type-accept var-type (elsa-get-type value) explainer))
explainer)
:code "invalid-assignment"
:compact t)))
(if value
(elsa-state-add-defvar state
(elsa-defvar :name var-name :type (oref value type)))
(elsa-state-add-defvar state
(elsa-defvar :name var-name :type (elsa-make-type unbound)))))))
(defun elsa--analyse:defvar-local (form scope state)
(elsa--analyse:defvar form scope state))
(defun elsa--analyse:defcustom (form scope state)
"Analyze `defcustom'.
The analysis works the same way as `elsa--analyse:defvar' except
we take the :type property of the defcustom into account when
automatically deriving the type."
(let* ((name (elsa-nth 1 form))
(value (elsa-nth 2 form))
(var-name (elsa-get-name name))
(def (elsa-state-get-defvar state var-name))
(var-type (and def (elsa-get-type def))))
(when value
(elsa--analyse-form value scope state))
(if var-type
(unless (or (not value)
(elsa-type-assignable-p var-type (elsa-get-type value)))
(elsa-state-add-message state
(elsa-make-error value
(elsa-with-temp-explainer explainer
(elsa-explain-and-indent explainer
("Variable %s expects `%s', got `%s'"
var-name
(elsa-type-describe var-type)
(elsa-type-describe (elsa-get-type value)))
(elsa-type-accept var-type (elsa-get-type value) explainer))
explainer)
:code "invalid-assignment"
:compact t)))
;; TODO: check the `:type' form here and also compare if we
;; are doing a valid assignment.
(if value
(elsa-state-add-defvar state
(elsa-defvar :name var-name :type (oref value type)))
(elsa-state-add-defvar state
(elsa-defvar :name var-name :type (elsa-make-type unbound)))))))
(defun elsa--analyse:defconst (form scope state)
"Analyze `defconst'.
If no type annotation is provided, find the value type through
`elsa--analyse:defvar' and wrap it as read-only."
(elsa--analyse:defvar form scope state)
(let* ((name (elsa-nth 1 form))
(value (elsa-nth 2 form))
(var-name (elsa-get-name name))
;; TODO: remove this once we analyse annotations here and not
;; in the reader.
(def (elsa-state-get-defvar state var-name))
(var-type (and def (elsa-get-type def))))
(when var-type
(unless (elsa-readonly-type-p var-type)
(oset def type (elsa-readonly-type :type var-type))
;; need to re-add to update the symbol property
(elsa-state-add-defvar state def)))
;; Defconst can also declare an interface, in which case we add it
;; to the global "pool".
(when (string-prefix-p "elsa-interface-" (symbol-name var-name))
(let* ((iface-name (replace-regexp-in-string
"^elsa-interface-" ""
(symbol-name var-name)))
(iface-type-def (cadr (elsa-form-to-lisp value)))
(iface-type
(elsa--make-type `(interface
,(intern iface-name)
,@iface-type-def))))
(put (intern iface-name) 'elsa-interface iface-type)))))
(defun elsa--analyse:defsubst (form scope state)
(elsa--analyse:defun form scope state))
;; TODO: add support for reading type annotations for lambda forms
(defun elsa--analyse:lambda (form scope state)
(let* ((sequence (oref form sequence))
(args (nth 1 sequence))
(body (nthcdr 2 sequence))
;; TODO: this should use `elsa--get-default-function-types'
(arg-types (-repeat (length (elsa-form-sequence args))
(elsa-make-type mixed)))
(vars))
(when (elsa-form-list-p args)
(-each-indexed (elsa-form-sequence args)
(lambda (index arg)
(let ((var (elsa-variable
:name (elsa-get-name arg)
:type (nth index arg-types))))
(push var vars)
(elsa-scope-add-var scope var)))))
(when body
(elsa--analyse-body body scope state))
(--each vars (elsa-scope-remove-var scope it))
(oset form type (elsa-function-type
:args arg-types
:return (if body
(oref (-last-item body) type)
(elsa-make-type nil))))))
(defun elsa--analyse:function (form _scope _state)
(let* ((arg (elsa-cadr form))
(name (elsa-get-name arg))
(type (elsa-function-get-type name)))
(when (and type (elsa-form-symbol-p arg))
(oset form type type))))
(defun elsa--analyse:quote (form _scope state)
(oset state quoted (trinary-true))
(let ((arg (cadr (oref form sequence))))
(cond
((elsa-form-list-p arg)
;; TODO: make sure we analyze everything as static when quoted
;; (elsa--analyse-list arg scope state)
(oset form type
(elsa-type-list
:item-type
(-reduce-from
(lambda (acc form)
(elsa-type-sum acc (elsa-get-type form)))
(elsa-type-empty)
(elsa-form-sequence arg)))))
((elsa-form-symbol-p arg)
(let* ((funciton-type (elsa-function-get-type (elsa-get-name arg)))
(const-type (elsa-const-type
:type (elsa-type-symbol)
:value (elsa--quoted-symbol-name form)))
(final-type (if funciton-type
(elsa-intersection-type
:types (list const-type funciton-type))
const-type)))
(oset form type final-type)))
((elsa-form-keyword-p arg)
(oset form type (elsa-type-keyword)))
((elsa-form-string-p arg)
(oset form type (elsa-type-string)))
((elsa-form-integer-p arg)
(oset form type (elsa-type-int)))))
(oset state quoted (trinary-false)))
(defun elsa--analyse--validate-interactive-string (state arg)
(let ((str (elsa-form-sequence arg))
(allowed-codes "[abBcCdDefFGikKmMnNpPrsSUvxXzZ]")
(case-fold-search nil))
(with-temp-buffer
(insert str)
(goto-char (point-min))
(skip-chars-forward "@^*")
(unless (eolp)
(insert "\n")
(backward-char 1)
(while (search-forward "\n" nil t)
(unless (or (looking-at-p allowed-codes)
(looking-at-p "^$"))
(elsa-state-add-message state
(elsa-make-error
arg
"Unknown interactive code letter: %c"
(char-after (point))))))))))
(defun elsa--analyse:interactive (form scope state)
(let ((arg (elsa-cadr form)))
(cond
((null arg))
((elsa-form-string-p arg)
(elsa--analyse--validate-interactive-string state arg))
((elsa-form-list-p arg) (elsa--analyse-list arg scope state))
(t (elsa-state-add-message state
(elsa-make-error
arg
"Invalid interactive spec, expecting string or list form"))))))
(defun elsa--analyse-backquote (_form _scope _state)
nil)
(defun elsa--analyse-unquote (_form _scope _state)
nil)
(defun elsa--analyse-splice (_form _scope _state)
nil)
(defun elsa--analyse-arg-variable-p (arg)
"Check if ARG is a variable and return its name."
(and (elsa-form-symbol-p arg)
(not (oref arg quote-type))
(elsa-get-name arg)))
;; (elsa--analyse-normalize-spec :: (function ((or bool (list bool)) mixed) (list bool)))
(defun elsa--analyse-normalize-spec (spec form)
"Normalize evaluation SPEC for FORM."
(cond
((eq spec t)
(-repeat (1- (length (elsa-form-sequence form))) t))
((eq spec nil)
(-repeat (1- (length (elsa-form-sequence form))) nil))
((eq (-last-item spec) 'body)
(-concat (-butlast spec)
(-repeat (- (1- (length (elsa-form-sequence form)))
(1- (length spec)))
t)))
(t spec)))
(defun elsa--check-argument-for-index (index min max argument-form overloads state all-overloads overloads-errors)
(let ((good-overloads nil)
(new-overloads nil))
(when (elsa-type-is-empty-p (elsa-get-type argument-form))
(elsa-state-add-message state
(elsa-make-warning argument-form
"Domain of variable %s is empty"
:code "empty-domain"
(elsa-tostring argument-form))))
;; Check each argument against all possible overloads. Keep only
;; those which can match current arguments. If we run out of
;; overloads, report the last set of possible overloads as the
;; error.
(setq
good-overloads
(-keep
(-lambda ((overload . overload-index))
(let* ((last-type-is-keys (elsa-type-keys-p (-last-item (oref overload args))))
(index (cond
((< index min) index)
((and (symbolp max) last-type-is-keys)
;; this is most likely a keyword argument,
;; so we check if it's a keyword and if so
;; pass it forward
(cond
;; if this index is a keyword, we just
;; skip to the next argument form.
((and (elsa-form-keyword-p argument-form)
(cl-evenp (- index min)))
nil)
;; if the previous form was a keyword,
;; that's the thing we need to look up
((elsa-form-keyword-p
(oref argument-form previous))
(elsa-form-to-lisp
(oref argument-form previous)))
(t index)))
(t index))))
(if (not index)
(list overload overload-index index)
(let* ((expected (elsa-function-type-nth-arg overload index))
(expected-normalized (or expected (elsa-type-mixed)))
(actual (oref argument-form type))
(acceptablep (elsa-type-could-assign-p expected-normalized actual)))
(cond
((not expected)
(push (list overload overload-index
(if (numberp index) argument-form (oref argument-form previous))
(format
"Argument %s is present but the function signature does not define it. Missing overload?"
(if (numberp index) (1+ index) index)))
overloads-errors))
((trinary-false-p acceptablep)
(push (list overload overload-index
(if (numberp index) argument-form (oref argument-form previous))
;; we need to sanitize the % sign in the error
;; because it is later passed to format in
;; `elsa-make-error'. And the "but received"
;; string can contain arbitrary text as one
;; possible type is (const "whatever")
(elsa-with-temp-explainer explainer
(elsa-explain-and-indent explainer
("Argument %s accepts type `%s' but received `%s'"
(if (numberp index) (1+ index) index)
(elsa-type-describe expected-normalized)
(elsa-type-describe actual))
(elsa-type-accept expected-normalized actual explainer))
explainer))
overloads-errors)))
(when (and expected (trinary-possible-p acceptablep))
(list overload overload-index index))))))
overloads))
(if good-overloads
;; If we have multiple overloads where the argument is of a
;; concrete type, that is not a sum or intersection (where the
;; analysis is a bit more difficult for now), we can eliminate
;; all the "less specific" overloads and only pick the one
;; with the narrowest domain which still accepts the argument.
;; Because the `good-overloads' are only those which accept
;; the arguments, we can sort them by `elsa-type-accept' and
;; pick the last (smallest) one.
(mapcar
(-lambda ((overload overload-index))
(cons overload overload-index))
(if (= (length good-overloads) 1)
(setq new-overloads good-overloads)
(setq new-overloads (elsa--simplify-overloads good-overloads))))
(setq new-overloads nil)
(elsa-state-add-message state