GoLang as Oracle external procedure


04.05.2021
by Kamil Stawiarski

This short article is a result of the following Twitter activity:

So let’s do it! First of all, you have to install GoLang: https://golang.org/doc/install

Once this is done, we can create a Go program to send emails. We will use "gopkg.in/mail.v2″ to make it simple.

Below you can find a simple GoLang code to send an email.

1package main
2import (
3    "C"
4    "crypto/tls"
5    gomail "gopkg.in/mail.v2"
6)
7 
8//Hardcoding those values may not be a good idea ;)
9const (
10    smtpHost string = "my.smtp.server.pl"
11    smtpPort int    = 465
12    smtpUser string = "my.user@ora-600.pl"
13    smtpPass string = "Xtr3mly53c23tP@ssw02d"
14)
15//The following comment with export is crucial for this to work!!! Without it won't be visible as a function
16//export sendEmail
17func sendEmail(mailSendTo *C.char, mailSubject *C.char, mailBody *C.char) *C.char {
18    //We have to use C package to give Oralce something it can understand when calling the function
19    var sendTo = C.GoString(mailSendTo) //Because of that we have to do a convertion to GO types
20    var subject = C.GoString(mailSubject)
21    var body = C.GoString(mailBody)
22    m := gomail.NewMessage()
23    // Set E-Mail sender
24    m.SetHeader("From", smtpUser)
25    // Set E-Mail receivers
26    m.SetHeader("To", sendTo)
27    // Set E-Mail subject
28    m.SetHeader("Subject", subject)
29    // Set E-Mail body. You can set plain text or html with text/html
30    m.SetBody("text/html", body)
31    // Settings for SMTP server
32    d := gomail.NewDialer(smtpHost, smtpPort, smtpUser, smtpPass)
33    // This is only needed when SSL/TLS certificate is not valid on server.
34    // In production this should be set to false.
35    d.TLSConfig = &tls.Config{InsecureSkipVerify: true}
36    // Now send E-Mail
37    if err := d.DialAndSend(m); err != nil {
38        return C.CString(err.Error())
39    }
40    return C.CString("ok")
41}
42func main() {}

I put explanation of the code in the comments.

Now, let’s build the package:

1[oracle@b3124c94c1c7 ~]$ go mod init sendemail
2go: creating new go.mod: module sendemail
3go: to add module requirements and sums:
4    go mod tidy
5[oracle@b3124c94c1c7 ~]$ go mod tidy
6go: finding module for package gopkg.in/mail.v2
7go: downloading gopkg.in/mail.v2 v2.3.1
8go: found gopkg.in/mail.v2 in gopkg.in/mail.v2 v2.3.1
9go: finding module for package gopkg.in/alexcesaro/quotedprintable.v3
10go: downloading gopkg.in/alexcesaro/quotedprintable.v3 v3.0.0-20150716171945-2caba252f4dc
11go: found gopkg.in/alexcesaro/quotedprintable.v3 in gopkg.in/alexcesaro/quotedprintable.v3 v3.0.0-20150716171945-2caba252f4dc
12[oracle@b3124c94c1c7 ~]$ go build -o sendemail.so -buildmode=c-shared

This operation should create the sendemail.so shared object and it’s header file sendemail.h:

1[oracle@b3124c94c1c7 ~]$ file sendemail.so
2sendemail.so: ELF 64-bit LSB shared object, x86-64, version 1 (SYSV), dynamically linked, BuildID[sha1]=f0eb77f5f5dd817b59837f7793d3afa43fe8fa0b, not stripped

We can copy sendemail.so to $ORACLE_HOME/lib or put it in some other directory and add the following line to the $ORACLE_HOME/hs/admin/extproc.ora file:

1SET EXTPROC_DLLS=ONLY:/home/oracle/extproc/sendemail.so

This line will let use use your shared SO file.

Now, we can create the appropriate database objects:

1SQL> create or replace library lib_sendemail as '/home/oracle/extproc/sendemail.so';
2  2  /
3 
4Library created.
5 
6SQL> create or replace function f_sendemail(p_to varchar2, p_subcject varchar2, p_body varchar2) return varchar2 as
7  2  external
8  name "sendEmail"
9  4  library lib_sendemail
10  5  language C;
11  6  /
12 
13Function created.

And we are ready to send the email from our external procedure! 😀

1SQL> select f_sendemail('kamil@ora-600.pl', 'jebacpis', 'And that''s it folks!') from dual;
2 
3F_SENDEMAIL('KAMIL@ORA-600.PL','JEBACPIS','ANDTHAT''SITFOLKS!')
4--------------------------------------------------------------------------------
5ok

Integrating the power of GoLang and Oracle together may bring some fun features 😉 Have fun with external procedures!


Contact us

Database Whisperers sp. z o. o. sp. k.
al. Jerozolimskie 200, 3rd floor, room 342
02-486 Warszawa
NIP: 5272744987
REGON:362524978
+48 508 943 051
+48 661 966 009
info@ora-600.pl

Newsletter Sign up to be updated