2025-05-01 22:30:59 +00:00
|
|
|
/*
|
|
|
|
Copyright © 2025 filifa
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
2025-05-04 03:37:58 +00:00
|
|
|
"errors"
|
2025-05-02 05:04:00 +00:00
|
|
|
"fmt"
|
2025-05-01 22:30:59 +00:00
|
|
|
"os"
|
|
|
|
|
2025-05-02 01:00:45 +00:00
|
|
|
"scm.dairydemon.net/filifa/dptdist/cmd/internal/markov"
|
|
|
|
|
2025-05-01 22:30:59 +00:00
|
|
|
"github.com/spf13/cobra"
|
2025-05-02 01:00:45 +00:00
|
|
|
"gonum.org/v1/gonum/graph/encoding/dot"
|
2025-05-02 05:04:00 +00:00
|
|
|
"gonum.org/v1/gonum/mat"
|
2025-05-01 22:30:59 +00:00
|
|
|
)
|
|
|
|
|
2025-05-02 01:00:45 +00:00
|
|
|
var file string
|
2025-05-01 22:30:59 +00:00
|
|
|
|
2025-05-04 02:42:09 +00:00
|
|
|
var matlabFmt bool
|
|
|
|
var pythonFmt bool
|
|
|
|
|
|
|
|
var oneline bool
|
|
|
|
|
2025-05-04 03:37:58 +00:00
|
|
|
var nodeOrder []string
|
|
|
|
|
2025-05-01 22:30:59 +00:00
|
|
|
// rootCmd represents the base command when called without any subcommands
|
|
|
|
var rootCmd = &cobra.Command{
|
|
|
|
Use: "dptdist",
|
|
|
|
Short: "A brief description of your application",
|
|
|
|
Long: `A longer description that spans multiple lines and likely contains
|
|
|
|
examples and usage of using your application. For example:
|
|
|
|
|
|
|
|
Cobra is a CLI library for Go that empowers applications.
|
|
|
|
This application is a tool to generate the needed files
|
|
|
|
to quickly create a Cobra application.`,
|
|
|
|
// Uncomment the following line if your bare application
|
|
|
|
// has an action associated with it:
|
2025-05-02 01:00:45 +00:00
|
|
|
Run: parse,
|
|
|
|
}
|
|
|
|
|
|
|
|
func parse(cmd *cobra.Command, args []string) {
|
|
|
|
data, err := os.ReadFile(file)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
graph := markov.NewAbsorbingMarkovChain()
|
|
|
|
err = dot.Unmarshal(data, graph)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !graph.IsValid() {
|
2025-05-03 01:02:05 +00:00
|
|
|
panic("not a Markov chain!")
|
|
|
|
}
|
|
|
|
|
|
|
|
if !graph.IsAbsorbing() {
|
2025-05-02 01:00:45 +00:00
|
|
|
panic("not an absorbing Markov chain!")
|
|
|
|
}
|
2025-05-02 05:04:00 +00:00
|
|
|
|
2025-05-04 03:42:02 +00:00
|
|
|
matrix, err := orderedAdjMatrix(graph)
|
2025-05-04 03:37:58 +00:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2025-05-04 03:42:02 +00:00
|
|
|
outputMatrix(matrix)
|
2025-05-01 22:30:59 +00:00
|
|
|
}
|
|
|
|
|
2025-05-04 03:37:58 +00:00
|
|
|
func orderedAdjMatrix(g *markov.AbsorbingMarkovChain) (*mat.Dense, error) {
|
|
|
|
matrix := g.AdjacencyMatrix()
|
|
|
|
if len(nodeOrder) == 0 {
|
|
|
|
return matrix, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
nodeIndexes := make(map[string]int)
|
|
|
|
for i := 0; i < len(nodeOrder); i++ {
|
|
|
|
nodeIndexes[nodeOrder[i]] = i
|
|
|
|
}
|
|
|
|
|
|
|
|
nodes := g.Nodes()
|
|
|
|
newOrder := make([]int, nodes.Len())
|
|
|
|
for nodes.Next() {
|
|
|
|
node := nodes.Node().(*markov.Node)
|
|
|
|
id := node.DOTID()
|
|
|
|
|
|
|
|
var ok bool
|
|
|
|
newOrder[node.ID()], ok = nodeIndexes[id]
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.New("node '" + id + "' not in given order")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
matrix.PermuteRows(newOrder, true)
|
|
|
|
matrix.PermuteCols(newOrder, true)
|
|
|
|
|
|
|
|
return matrix, nil
|
|
|
|
}
|
|
|
|
|
2025-05-04 03:42:02 +00:00
|
|
|
func outputMatrix(matrix mat.Matrix) {
|
|
|
|
fmtOptions := make([]mat.FormatOption, 0)
|
|
|
|
if matlabFmt {
|
|
|
|
fmtOptions = append(fmtOptions, mat.FormatMATLAB())
|
|
|
|
} else if pythonFmt {
|
|
|
|
fmtOptions = append(fmtOptions, mat.FormatPython())
|
|
|
|
}
|
|
|
|
|
|
|
|
out := mat.Formatted(matrix, fmtOptions...)
|
|
|
|
|
|
|
|
// for matlab and python formats, %#v outputs as matrix and %v is
|
|
|
|
// oneline, but for standard format, it's the opposite, so we xor
|
|
|
|
if (matlabFmt || pythonFmt) != oneline {
|
|
|
|
fmt.Printf("%#v\n", out)
|
|
|
|
} else {
|
|
|
|
fmt.Printf("%v\n", out)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-05-01 22:30:59 +00:00
|
|
|
// Execute adds all child commands to the root command and sets flags appropriately.
|
|
|
|
// This is called by main.main(). It only needs to happen once to the rootCmd.
|
|
|
|
func Execute() {
|
|
|
|
err := rootCmd.Execute()
|
|
|
|
if err != nil {
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
// Here you will define your flags and configuration settings.
|
|
|
|
// Cobra supports persistent flags, which, if defined here,
|
|
|
|
// will be global for your application.
|
|
|
|
|
|
|
|
// rootCmd.PersistentFlags().StringVar(&cfgFile, "config", "", "config file (default is $HOME/.dptdist.yaml)")
|
|
|
|
|
|
|
|
// Cobra also supports local flags, which will only run
|
|
|
|
// when this action is called directly.
|
2025-05-02 01:00:45 +00:00
|
|
|
rootCmd.Flags().StringVarP(&file, "file", "f", "", "dot file with absorbing Markov chain")
|
|
|
|
rootCmd.MarkFlagRequired("file")
|
2025-05-04 02:42:09 +00:00
|
|
|
|
|
|
|
rootCmd.Flags().BoolVar(&matlabFmt, "matlab", false, "format output as MATLAB array")
|
|
|
|
rootCmd.Flags().BoolVar(&pythonFmt, "python", false, "format output as python array")
|
|
|
|
rootCmd.MarkFlagsMutuallyExclusive("matlab", "python")
|
|
|
|
|
|
|
|
rootCmd.Flags().BoolVar(&oneline, "oneline", false, "output on one line")
|
2025-05-04 03:37:58 +00:00
|
|
|
|
|
|
|
rootCmd.Flags().StringSliceVarP(&nodeOrder, "order", "o", nil, "order of nodes in rows/columns of output")
|
2025-05-02 01:00:45 +00:00
|
|
|
}
|