-
Notifications
You must be signed in to change notification settings - Fork 47
/
Copy pathRandomHypergraph.m
80 lines (66 loc) · 2.76 KB
/
RandomHypergraph.m
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
Package["SetReplace`"]
PackageImport["GeneralUtilities`"]
PackageExport["RandomHypergraph"]
(* Documentation *)
SetUsage @ "
RandomHypergraph[size$] generates a random hypergraph, where size$ is the total of its hyperedge arities.
RandomHypergraph[{count$, arity$}] generates a random hypergraph with count$ hyperedges of a specified arity$.
RandomHypergraph[{{count$(1), arity$(1)}, {count$(2), arity$(2)}, $$}] generates a random hypergraph with hyperedges \
of multiple arities.
RandomHypergraph[sizeSpec$, maxVertices$] also limits the vertex count.
";
(* SyntaxInformation *)
SyntaxInformation[RandomHypergraph] =
{"ArgumentsPattern" -> {sizeSpec_, maxVertices_.}};
(* Argument count *)
RandomHypergraph[args___] := 0 /;
!Developer`CheckArgumentCount[RandomHypergraph[args], 1, 2] && False;
(* Main entry *)
expr : RandomHypergraph[sizeSpec_, maxVertices_ : Automatic] := ModuleScope[
res = Catch[randomHypergraph[HoldForm @ expr, sizeSpec, maxVertices]];
res /; res =!= $Failed
];
(* Error messages *)
RandomHypergraph::invalidSig = "\
The argument at position `2` in `1` should be a positive integer or a hypergraph size specification.";
(* Support functions *)
randomPartition[n_] :=
randomPartition[n, RandomInteger[{1, n}]];
randomPartition[n_Integer ? Positive, nparts_Integer ? Positive] :=
RandomVariate @ MultinomialDistribution[n, ConstantArray[1 / nparts, nparts]];
$sizeSpecPattern = {_Integer ? NonNegative, _Integer ? NonNegative};
(*
In[]:= RandomHypergraph[8]
Out[]= {{4, 4, 3}, {2, 7}, {5, 6}, {8}}
*)
randomHypergraph[caller_, complexity_Integer ? Positive, n : (_Integer ? Positive | Automatic)] :=
With[{maxVertices = Replace[n, Automatic -> complexity]},
RandomInteger[{1, maxVertices}, #] & /@ DeleteCases[randomPartition @ complexity, 0]
];
(*
In[]:= RandomHypergraph[{{5, 2}, {4, 3}}, 10]
Out[]= {{1, 2}, {2, 9}, {6, 1}, {3, 3}, {10, 10}, {1, 8, 8}, {5, 7, 9}, {5, 7, 6}, {5, 3, 3}}
*)
randomHypergraph[caller_, sizeSpec : {$sizeSpecPattern ..}, n : (_Integer ? Positive | Automatic)] :=
ModuleScope[
If[n === Automatic,
(* Maximum possible number of atoms *)
maxVertices = Total[Times @@@ sizeSpec]
,
maxVertices = n
];
Catenate[RandomInteger[{1, maxVertices}, #] & /@ sizeSpec]
];
(*
In[]:= RandomHypergraph[{5, 2}, 10]
Out[]= {{5, 4}, {7, 8}, {7, 6}, {7, 1}, {4, 9}}
*)
randomHypergraph[caller_, sizeSpec : $sizeSpecPattern, n_] :=
randomHypergraph[caller, {sizeSpec}, n];
(* Incorrect arguments messages *)
randomHypergraph[caller_, sizeSpec : Except[$sizeSpecPattern | {$sizeSpecPattern ..} | _Integer ? Positive], _] :=
(Message[RandomHypergraph::invalidSig, caller, 1];
Throw[$Failed]);
randomHypergraph[caller_, sizeSpec_, maxVertices_] :=
(Message[RandomHypergraph::intpa, caller, 2];
Throw[$Failed]);