Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Remove global clients #1014

Open
wants to merge 40 commits into
base: main
Choose a base branch
from
Open

Remove global clients #1014

wants to merge 40 commits into from

Conversation

dido18
Copy link
Contributor

@dido18 dido18 commented Jan 27, 2025

Please check if the PR fulfills these requirements

  • The PR has no duplicates (please search among the Pull Requests
    before creating one)
  • Tests for the changes have been added (for bug fixes / features)
  • What kind of change does this PR introduce?
    The purpose of this PR is to improve the maintainability, testability, and clarity of the codebase by removing global variables and explicitly passing dependencies as arguments to functions or methods. Global variables can lead to tightly coupled code, making it harder to understand, test, and maintain.
  • What is the current behavior?
  • What is the new behavior?
    1. remove global variables, they are now created in the main function and passed explicitly to the components or functions that need them.
    2. rename the h into hub
    3. remove the serial.go and create the following new files:
      • serialhub.go contains the serialhub struct
      • serialportlist.go contains the serialPortList and the SpPortItem struct
    4. move spErr,spWrite and spClose from the serial.go to the hub.go file
    5. move the spHandlerOpen from the serialport.go to the hub.go file
    6. move the go hub.serialPortList.Run() from the maingo into the hub.run() because the hub is in charge of connecting serial events to websocket clients.
  • Does this PR introduce a breaking change?
    No.
  • Other information:

@per1234 per1234 added type: enhancement Proposed improvement topic: code Related to content of the project itself labels Jan 29, 2025
@dido18 dido18 changed the base branch from main to datarace-on-config-access January 29, 2025 10:34
@dido18 dido18 changed the base branch from datarace-on-config-access to main March 31, 2025 14:05
@codecov-commenter
Copy link

codecov-commenter commented Mar 31, 2025

Codecov Report

Attention: Patch coverage is 8.06100% with 422 lines in your changes missing coverage. Please review.

Project coverage is 20.77%. Comparing base (87f097b) to head (815a791).
Report is 1 commits behind head on main.

Files with missing lines Patch % Lines
hub.go 8.21% 197 Missing and 4 partials ⚠️
serialportlist.go 4.58% 125 Missing ⚠️
serialhub.go 19.35% 25 Missing ⚠️
main.go 0.00% 23 Missing ⚠️
conn.go 7.14% 13 Missing ⚠️
update.go 0.00% 13 Missing ⚠️
info.go 0.00% 10 Missing ⚠️
serialport.go 0.00% 10 Missing ⚠️
tools/download.go 71.42% 2 Missing ⚠️
Additional details and impacted files
@@            Coverage Diff             @@
##             main    #1014      +/-   ##
==========================================
+ Coverage   20.23%   20.77%   +0.53%     
==========================================
  Files          42       43       +1     
  Lines        3242     3298      +56     
==========================================
+ Hits          656      685      +29     
- Misses       2499     2522      +23     
- Partials       87       91       +4     
Flag Coverage Δ
unit 20.77% <8.06%> (+0.53%) ⬆️

Flags with carried forward coverage won't be shown. Click here to find out more.

☔ View full report in Codecov by Sentry.
📢 Have feedback on the report? Share it here.

🚀 New features to boost your workflow:
  • ❄️ Test Analytics: Detect flaky tests, report on failures, and find test suite problems.

@dido18 dido18 marked this pull request as ready for review April 1, 2025 13:45
@dido18 dido18 requested a review from a team April 2, 2025 09:44
Copy link
Contributor

@alessio-perugini alessio-perugini left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It looks good. Left some minor comments.
Probably we should do another round of manual testing to make sure no regression are being introduced. We can also do it before the next release, depends on how many changes we want to make.

serial.go Outdated
@@ -255,22 +265,22 @@ func (sp *SerialPortList) getPortByName(portname string) *SpPortItem {
return nil
}

func spErr(err string) {
func (h *hub) spErr(err string) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we reorganize those files with one package per struct?

  • hub
  • serialPortList
  • serialHub

It isn't very easy to follow this code

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I thought the same thing, but I think it would be easier for the reviewers if we open a new PR dedicated to refactoring those components. WDYT?

Copy link
Contributor Author

@dido18 dido18 Apr 2, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I tried to move the least as possible to facilitate reviews with the following changes:

  • split the serial.go file into
    • serialhub.go with the serialhub struct
    • serialportlist.go with the serialPortList and the SpPortItem struct
  • move spErr,spWrite and spClose from the serial.go to the hub.go file
  • move the spHandlerOpen from the ßerialport.go to the hub.go file

@dido18
Copy link
Contributor Author

dido18 commented Apr 3, 2025

Tested on linux with MKR 1010

  • upload the echo-skecth and test that the serial monitor works as expected
  • provision an IOT device + update NINA
    • downgrade the NINA with ~/.arduino-create/arduino/arduino-fwuploader/2.4.1/arduino-fwuploader firmware flash -b arduino:samd:mkrwifi1010 -a /dev/ttyACM0 --module [email protected]
    • follow the create a device with iot provisioning

@@ -73,51 +110,53 @@ const commands = `{
]
}`

func (h *hub) unregisterConnection(c *connection) {
if _, contains := h.connections[c]; !contains {
func (hub *hub) unregisterConnection(c *connection) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it is more canonical to use single letter for local object mapping

Suggested change
func (hub *hub) unregisterConnection(c *connection) {
func (h *hub) unregisterConnection(c *connection) {

Comment on lines +287 to +289
type logWriter struct {
onWrite func([]byte)
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would create an utility that transform a channel to a writer, for instance something like that

// ChanWriter is a simple io.Writer that sends data to a channel.
type ChanWriter struct {
	Ch chan<- []byte
}

func (u *ChanWriter) Write(p []byte) (n int, err error) {
	u.Ch <- p
	return len(p), nil
}

then you can directly use it as follows

multiWriter := io.MultiWriter(&ChanWriter{Ch: hub.broadcastSys}, os.Stderr)

}

func (hub *hub) spErr(err string) {
//log.Println("Sending err back: ", err)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
//log.Println("Sending err back: ", err)

// Register serial ports from the connections.
func (sh *serialhub) Register(port *serport) {
sh.mu.Lock()
//log.Print("Registering a port: ", p.portConf.Name)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
//log.Print("Registering a port: ", p.portConf.Name)

// Unregister requests from connections.
func (sh *serialhub) Unregister(port *serport) {
sh.mu.Lock()
//log.Print("Unregistering a port: ", p.portConf.Name)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
//log.Print("Unregistering a port: ", p.portConf.Name)

}

// Register serial ports from the connections.
func (sh *serialhub) Register(port *serport) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would use a single char variable

Suggested change
func (sh *serialhub) Register(port *serport) {
func (s *serialhub) Register(port *serport) {

Comment on lines +27 to +28
ports map[*serport]bool
mu sync.Mutex
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I usually see the other way around first the mutex and then the resource that is protected

Suggested change
ports map[*serport]bool
mu sync.Mutex
mu sync.Mutex
ports map[*serport]bool

for port := range sh.ports {
if strings.EqualFold(port.portConf.Name, portname) {
// we found our port
//spHandlerClose(port)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

should we remove this?

Comment on lines +389 to +392
p.OnClose = func(port *serport) {
hub.serialPortList.MarkPortAsClosed(p.portName)
hub.serialPortList.List()
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think you can remove the OnClose callback and move those two lines of code in the spClose method below after calling p.Close(). AFAIK this struct is the only one closing ports, so moving the code there should cover all the cases. The following should be the updated spClose function:

func (hub *hub) spClose(portname string) {
	if myport, ok := hub.serialHub.FindPortByName(portname); ok {
		hub.broadcastSys <- []byte("Closing serial port " + portname)
		myport.Close()
		hub.serialPortList.MarkPortAsClosed(myport.portName)
		hub.serialPortList.List()
	} else {
		hub.spErr("We could not find the serial port " + portname + " that you were trying to close.")
	}
}

Comment on lines +376 to +388
p := &serport{
sendBuffered: make(chan string, 256000),
sendNoBuf: make(chan []byte),
sendRaw: make(chan string),
portConf: conf,
portIo: sp,
portName: portname,
BufferType: buftype,
}

p.OnMessage = func(msg []byte) {
hub.broadcastSys <- msg
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Instead of this call back I think you could use the ChanWriter I suggested above

Suggested change
p := &serport{
sendBuffered: make(chan string, 256000),
sendNoBuf: make(chan []byte),
sendRaw: make(chan string),
portConf: conf,
portIo: sp,
portName: portname,
BufferType: buftype,
}
p.OnMessage = func(msg []byte) {
hub.broadcastSys <- msg
}
p := &serport{
sendBuffered: make(chan string, 256000),
sendNoBuf: make(chan []byte),
sendRaw: make(chan string),
portConf: conf,
portIo: sp,
portName: portname,
BufferType: buftype,
Writer: &ChanWriter{Ch: hub.broadcastSys},
}

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
topic: code Related to content of the project itself type: enhancement Proposed improvement
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants