Skip to content

Latest commit

 

History

History
272 lines (221 loc) · 15.8 KB

README.md

File metadata and controls

272 lines (221 loc) · 15.8 KB

GOARG

What is This?

  • This project is for those who use command line arguments in their own projects.
  • It's easy to use you can implement your project just 4 lines of code.

What can you do?

  • In just one line, you can enter the name of your option, the activity value, the usage information, and the error specific to it.
  • You can add the project title, examples of how it works. When added, they will all be displayed with auto usage writer.
  • This project does things for you that you don't want you to do. You just focus on your own project.

How to Install?

  • Open your project and write the code below in the terminal. The project will be added.

Structs

  • I will talk about the structs that should be known here.
    • Let's start with the Goarg Struct first

      • The "goarg" struct can accept a total of 4 values: Usage, Options, Mods, and ModeName.
      • You need to have an understanding of the fields:
        • Usage: The Usage struct in this field is created to hold a Usage address. It is a structure necessary to write the Usage information of your project. In the future, its contents will be explained in more detail.
        • Options: All the options you add are stored in an array-like structure known as the Options struct. You can refer to where I previously explained this struct for a better understanding.
        • Mods: This field is a combination of the goarg structure you're describing and the ModeName. When making a goarg.AddMode call, the first string argument you provide is used here as a key. This field becomes useful when searching for a specific mode.
        • ModeName: This field is a value you set when calling the AddMode function. It holds significant importance as it helps you properly direct incoming inputs. It's returned to you after processing within the Input struct. With this field, you can easily use a switch case structure to direct incoming inputs to their respective modes.
    • Let me explain the Usage Struct for you.

      • The Usage struct accepts 4 values.
        • Title: This title represents the name of the Goarg struct and appears at the beginning of the help message.
        • Description: This is where you explain the purpose of the current goarg and it appears just below the title.
        • Examples: This section is meant for explaining how Goarg works. It can include examples like using go run main.go -p .
        • Message: After combining all the above values to create a Help message, this variable holds that message.
    • Let me explain the Option Struct for you.

      • The Option struct accepts 4 values, and understanding each of them will significantly simplify your task:
        • PlaceHolder: When using AddOption, the first argument you add becomes the PlaceHolder. This allows you to enter the argument through the Command Line. You can separate multiple placeholders using commas. Additionally, they must begin with a '-' or '--' to be recognized.
        • Active: This value indicates whether the command line argument expects additional input. When set to false, it expects input. For instance, after typing '-p', you would need to provide another value separated by a space. When set to true, it only returns one value. You can use this as 'true' when appropriate.
        • Usage: Here, you can describe the purpose of your Option. In Automatic Usage, this description will be visible.
        • Global: This determines whether the option is global across all modes or specific to a single mode. This distinction can be extremely useful.
    • Finally let me explain Input Struct for you.

      • The Input struct accepts 2 values.
        • Argument: This field holds the placeholder of the input from the user. for better input routing
        • Value: This field holds the value of the placeholder above
        • ModeName: As I previously explained in the context of the goarg struct, this field indicates which mode's option the input corresponds to. It helps you associate the input with the relevant mode. When we write args := Setup.Start(), the variable args receives the incoming values as a slice of Input structs, i.e., []Input.

How to Use?

  • First, let me explain the methods.
    • Goarg := cla.Init(): Initializes the struct
    • Goarg.SetTitle(string): Addes the name of the project to the usage
    • Goarg.SetExamples(string): Adds examples of how your project works to the usage.
    • Goarg.SetMessage(string): You can set your own usage message.
    • Goarg.SetUsage(title string, description string, examples []string): You can set all the value of usage in on line.
    • Goarg.AddOption(args string, active bool, usage string): Addes a option to the goarg for later to use.
      • args: the option name like -h --host. You must separate the setting names with (,).
      • active: Indicates whether the setting takes input. if true it takes no input it's just a "1"
      • usage: this is just a usage for the option. You can see on automatic usage.
      • error: you can add errors to the option for later to use.
    • Goarg.AutomaticUsage(): You can activate the automatic usage. This code MUST be after AddOptions.
    • Goarg.Start(string) []Input: This is the last piece of code you need. This code returns us array of struct Input. Input has (for now) three variables.
      • Arguments: Here you can see which option is returning.
      • Value: The value of the option that returned.
      • Error: Custom errors that you add for option.
    • This is it.
  • Now a setup example for the project
package main

import (
	"fmt"

	"github.com/cetinboran/goarg/cla"
)

func main() {
	Goarg := cla.Init()
	// Goarg.SetTitle("cetinboran")
	// Goarg.SetDescription("This is a Description")
	//Goarg.SetExamples([]string{"Example 1", "Example 2}})

	// This is equivalent to the three codes above
	Goarg.SetUsage("cetinboran", "This is a Description", []string{"Example 1", "Example 2"})

	Goarg.AddOption("-h,--host", false, "Enter your host.")
	Goarg.AddOption("-p,--port", false, "Enter your Port.")
	Goarg.AddOption("--fast", true, "Makes it faster.")

	NewMode := cla.ModInit()
	NewMode.SetUsage("NewMode", "This is a Description", []string{"Example 3", "Example 4"})
	NewMode.AddOption("-a,--aa", false, "Enter your aa.")
	NewMode.AddOption("-b,--bb", false, "Enter your bb.")

	// Main Mode Init
	Goarg.AddMode("NewMode", &NewMode)

	// Usage
	NewMode.AutomaticUsage()
	Goarg.AutomaticUsage()

	args := Goarg.Start()

	for _, arg := range args {
		fmt.Println(arg.Argument, arg.Value, arg.ModeName)
	}
}
  • If we write the following code to the terminal go run .\main.go -h 127.0.0.1 --port 22 --fast.

  • we can see the result below.

    • h 127.0.0.1 Main
    • port 22 Main
    • fast 1 Main
  • The main text here states where the inputs are returned.Main writes that this is the first created Goarg Struct.

  • If it just wrote mode, that means there would be a Goarg mod that hasn't been added to the Main goarg struct yet.

  • If we add it to the main struct with the AddMode function, the first argument we write there will change to our ModeName. This allows us to send the inputs to the right places.

  • Below you can see the auto-generated help. You can see this message with the code below.

    • go run .\main.go --help
    cetinboran
    Enter your host. [-h --host]     
    Enter your Port. [-p --port]        
    Makes it faster. [--fast]

    Examples:
    1. go run main.go -h 127.0.0.1 -p 22
    2. go run main.go -p 192.168.1.* 

Better Goarg Structure

  • If you intend to create a well-organized goarg structure, there are several important steps to keep in mind.
    • Main Goarg Initialization: When initializing the main goarg, you can use functions like SetTitle, SetDescription, SetExamples, or SetUsage if necessary.
    • Adding Modes to the Main Goarg: If you plan to add modes to the main goarg, you can replicate the steps you performed in the first step for each individual mode. If you wish to add multiple modes, you can stack them one after another.
    • Adding Modes to the Main Goarg: You should use the AddMode function to add all modes to the main goarg.
    • Adding Global Options: If you're considering adding global options, do this after adding the modes. Global options are used to apply the same options to all modes.
    • Generating Automatic Usage Information: Once you complete these steps, you can call the AutomaticUsage() function if you want to generate automatic usage information. If you prefer to add a custom message, you can use SetMessage to include your own.
    • Starting the Main Goarg: Lastly, you can call the Start() function of the main goarg to begin waiting for inputs.
  • These aforementioned steps are also applicable to nested modes. This means that your mode can itself contain another mode. By following the same steps, you can avoid issues.

Errors

  • there are four error messages you may encounter
    • not an option -> If you didn't add such a option but you are using it, this message will appear.
    • missing input for ... -> If you have added the option with the false activity option but forgot to add the input, this message will appear.
    • no need input for ... -> If you added the option with true activity option and add the input, this message will appear. ( You don't need the input )
    • you don't need more than one of this option -> If you try to use the same option more than once, this message will appear.

What's new

  • 19.08.2023
    • now you can add mods to your project
    • it allows you to spread the options in your project more regularly.
    • Your project can be more organized.
    • First, let's explain the added methods.
      • hello := cla.ModInit(): Initializes the mode struct
      • Goarg.AddMode("helloMod", &hello): This method adds a mod to the main goarg struct.
        • The first argument specifies how to call the mode, the other specifies the mod's struct
        • You must be the address of that struct when using the other argument.
    • The rest is the same. You can add settings, add examples, add title as in the main struct.
    • At the end just run the start of the main struct
    • Reminder: Automatic Usage code must be at the end.
    • With the code on the below. We access the mode you added with AddMode() and we can use it just like we did in main goarg.
    • go run .\main.go helloMod
    • After typing the first argument you give in the addMode function to the terminal, access to the functions of that mode is provided.
package main

import (
	"fmt"

	cla "github.com/cetinboran/goarg/CLA"
)

func main() {
	Goarg := cla.Init()

	Goarg.SetTitle("cetinboran")
	Goarg.SetExamples([]string{"go run main.go -h 127.0.0.1 -p 22", "go run main.go -p 192.168.1.*"})

	Goarg.AddOption("--host", false, "Enter your host.")
	Goarg.AddOption("-p,--port", false, "Enter your Port.")
	Goarg.AddOption("--fast", true, "Makes it faster.")

	hello := cla.ModInit()
	Goarg.AddMode("helloMod", &hello)

	hello.SetTitle("Hello Mod")
	hello.SetExamples([]string{"Example 1", "Example 2"})
	hello.AddOption("--hello", true, "Says hello")
	hello.AddOption("-sc,--scream", false, "You can scream!")

	hello.AutomaticUsage()
	Goarg.AutomaticUsage()

	args := Goarg.Start()

	for _, arg := range args {
		fmt.Println(arg.Argument, arg.Value, arg.ModeName)
	}
}
  • 20.08.2023
    • Now we can understand from which mode the input came from, from the return of the arguments.
    • args.ModeName carries this information.
    • This information is the first input value we entered in the AddMode section. Accordingly, it became easier to process incoming inputs.
    • AddGlobalOption func added. Now, when you define a option with this function in the main goarg, this option will come to all the modes below.
    • This code must be under AddMode() Function
  • 23.08.2023
    • Fixed 4 medium bugs.
    • Goarg.SetDescription(string): Added setdescription function. You can now write a description about your goarg.It will be added directly thanks to automatic usage.
  • 25.08.2023
    • Now we can add mods inside mods. like 'config create'
    • A few minor bugs fixed
    • GoargSetUsage(title string, description string, examples []string): With this code, instead of setting the title description or example one by one, you can set them all at once.
    • Goarg struct has undergone a minor change. Instead of being kept in the title examples description goargin, a struct called Usage is opened and kept in it. Goarg only keeps the Usage struct.
  • 30.08.2023
    • The goarg start function now returns two distinct values. The first one is a args structure containing the arguments, and the second one is an error structure that includes error messages added by you corresponding to the input values. With this update, your code will have become more comprehensible and manageable.
    • By using Goarg.AddError(argument string, errors []string), you can add errors specific to argument names to this OptionError struct or to a goarg module.
    • We will handle all necessary processes for you and return the error to you.
    • The OptionError struct comes with a GetErrors method. You can utilize this method to retrieve the error message for the desired option in your project, allowing you to display the error message and terminate the program.
    • The GetErrors method takes one arguments. The first one is the number of the error you want to display on the screen. This number corresponds to the order of the Error string array that you added using AddError. If you want to see the first error you added, you'll write 1.
    • An example is provided below.
    • It might seem a hard to use, but if you look at the structure of the project I recently worked on, you'll easily understand how useful it can be. Nonetheless, I'm still looking forward to your feedback, whether it's positive or negative. gosec
package main

import (
	"fmt"

	cla "github.com/cetinboran/goarg/CLA"
)

func main() {
	Goarg := cla.Init()

	Goarg.SetTitle("cetinboran")
	Goarg.SetExamples([]string{"go run main.go -h 127.0.0.1 -p 22", "go run main.go -p 192.168.1.*"})

	Goarg.AddOption("--host", false, "Enter your host.")
	Goarg.AddOption("-p,--port", false, "Enter your Port.")
	Goarg.AddOption("--fast", true, "Makes it faster.")

    Goarg.AddError("--host", []string{"Please Enter an ip not url"})
    Goarg.AddError("-p,--port", []string{"Please enter between 0 - 65535"})

	hello := cla.ModInit()
	Goarg.AddMode("helloMod", &hello)

	hello.SetTitle("Hello Mod")
	hello.SetExamples([]string{"Example 1", "Example 2"})
	hello.AddOption("--hello", true, "Says hello")
	hello.AddOption("-sc,--scream", false, "You can scream!")

    hello.AddError("--hello", []string{"I am shy", "My voice was muted"})
	hello.AddError("-sc,--scream", []string{"My voice was muted"})

	hello.AutomaticUsage()
	Goarg.AutomaticUsage()

	args, errors := Goarg.Start()
    errors["--hello"].GetErrors(1) // => I am shy.
}
  • 23.09.2023
    • Goarg.AddOptionTitle(string) Added.
    • This can be used to add a title to the beginning of your options in the usage section to make them more organized.

Contact

github linkedin instagram twitter