How to Write a Steemit Blogging Bot Step by Step Using JavaScript

in #utopian-io7 years ago (edited)

What Will I Learn?

  • Basics of JavaScript syntax, Node.js projects, and npm usage
  • Use Microsoft VSCode as IDE
  • Grab HTML data from Internet web pages
  • Use Cheerio to parse, query and manipulate DOM tree from HTML
  • Basic usage of steemit-js library
  • Use Cron and Node.js cron library to schedule tasks

Stemmit provides APIs for us to do a lot of awesome things -- this is what makes it the best blogging platform in my mind. Now, how to use them to develop our own bot? Just follow my step.

Which Programming Language?

Well, this is also the best part of it: not only did Steemit provide APIs, but also in so many popular programming languages. So, how to choose?

Basically, choose which ever you are good at.

My first choice was Python. But its support for Windows is not that good (I know Windows is not the best for developing, but I kind of need it so much, and I cannot successfully install the steem-python package there, at all). So I just go to JavaScript.

So this tutorial shows a JavaScript example

Anything to Prepare?

Yes, a JavaScript engine is required to run you code. Any browser can do it, but it's not convenient at all.

A. Node.js is a good recommendation to run the JavaScript project.

To Install Node.js, check here to find whatever suits you OS.

B. A good IDE makes things a lot easier, although it's optional.

IDE refers to "Integrated Development Environment", a piece of software to help you write, debug and manage your project codes. Here I am using Microsoft's VSCode to organize project code. It natively supports Node.js projects, including GIT stuffs. If you need more functionality such as Python support or TODO parser, just search relative extensions. VSCode support plenty of such kind of extensions, which cover almost any need for any programmers.

Shall We Start?

Of course, let's do it!

A. Setup the shell Node.js project -- "SteemitBlogger"

  1. Open the VSCode, and open the root directory of the project, let's say, D:\Projects\SteemitBlogger\.

1.gif

  1. Now press Ctrl+` (the one above CapsLock and Tab) to show up the terminal panel, run npm init in it, and follow the guide to type in the information, including package name, version, etc. These operations created the package.json. In the example I accepted all defaults.

2.gif

  1. In the package.json, the entry was default set to be index.js. This file needs to be manually created, and it's where we code the blogger.

B. Coding I: what to write in the blog

A Steemit blog contains a title, an author, several keywords as tags, and a body. These are the things we should prepare. Remember this is an autobot, composing and publishing blogs repeatedly. So you want different titles and bodies, maybe different tags, too. Let's do it like this:

// put date into the fields so they are different each day
var now = new Date().toISOString();    // the current time in UTC time zone, in format of "2017-12-31T23:59:59Z",
                                       // where the trailing 'Z' refers to the UTC time zone.
var title = 'Hello World on ' + now.split('T')[0];    // this 'split' separates the time string by 'T', and  takes
                                                      // the first part (JavaScript lists start with index 0)
var author = 'user1';
var body = 'Hello to everyone in the world at' + now + '\n\n$PLACE_HOLDER\n';

I want fill something later to the body, so I define the above $PLACE_HOLDER for now.

C. Coding II: grab data from Internet and put them in

Sure. Let's say, I want to find what Microsoft did in GitHub, i.e., the repository names.
The link involved is https://github.com/Microsoft.

C.1 Get the html data and parse DOM

DOM refers to "Document Object Model", which stores all information of a web page into a tree structure, with standard APIs to find and modify information in it. We use Cheerio to parse HTML strings to DOM trees, it operates the same way as jQuery. It's OK if you don't know Cheerio/jQuery, just follow my example.

To install Cheerio, type the following command in terminal:

npm install --save cheerio    # '--save' writes cheerio into package.json dependency records

3.gif

Above shows a short version of the install command. After it, dependency of cheerio appended to package.json, and a folder node_modules added to the project containing all dependencies.

Node.js provides http.get and https.get to read data from web pages. We just use it like this:

var cheerio = require('cheerio');
var https = require('https');
function grabData() {                                        // Put this job in a function
    https.get('https://github.com/Microsoft', function(res) {    // The 'get' procedure is standard,
        res.on('error', function(err) {                          // so the code snippet applies all similar cases 
            throw err;
        }); // res.on('error', function(err) { ... });
        var html = '';
        res.on('data', function(data) {
            html += data;
        }); // res.on('data', function(data) { ... });
        res.on('end', function() {
            var $ = cheerio.load(html);                         // This line parses HTML string to DOM with root '$'
            // Do something with the DOM
            // ... ...
        }); // res.on('end', function() { ... });
    }); // https.get('https://github.com/Microsoft', function(res) { ... });
} // function grabData() { ... }

C.2 Ming the DOM

Use Google Chrome's (similar with other browsers) Developer Tools, we can dig DOM information directly.
Just press F12 and follow this:

4.gif

In the Developer Tools, choose "Elements" tab, and toggle the left top button on, then you can quickly navigate the corresponding
DOM element when you hover the mouse cursor on anything in the web page. We can see each of the target text locates in an <a> with the attribute itemprop value of name codeRepository, so we grab such data in the following way:

// ... ...
// Do something with the DOM
var data = '';
$('[itemprop="name codeRepository"]')    // use '[attr="value"]' to select all element with target attribute/value
        .each(function(i, e) {           // '.each' iterates every selected element
    data += ' + ' + e.children[0].data.trim() + '\n';    // the texts are stored in its first child's 'data' field
}); // $( ... ).each( ... );
body = body.replace('$PLACE_HOLDER', data);     // put the data to the correct place in body
// ... ...

C.3 Prepare and publish the blog to Steemit

As been said, a Steemit blog contains title, author, tags and body, but they are not all. The full list contains:

  • title: simply the above title;
  • author: simply the above author;
  • body: simply the above body, in markdown format (see below);
  • permlink: a permanent link for this blog, could be concatenated by author and title, with spaces being properly dealt with:
    // Prepare and publish the blog
    // '/\s+/g' is a regular expression to match all continuous spaces
    var permlink = (author + '-' + title).replace(/\s+/g, '-');
    
  • json_metadata: this is a JSON object that contains tags, for Steemit blogs, it could always be set to:
    var json_metadata = {
        "tags":     ["tag1", "tag2", "tag3", "tag4"],    // the tag list you want add
        "app":      "steemit/0.1",                       // always be like this
        "format":   "markdown"                           // as mentioned above, body in markdown format
    }; // var json_metadata = { ... };
    
  • tags: they are included in json_metadata; and of course
  • posting_key: you need this to publish the blog, it must match author.

steem.broadcast.comment is the function to publish it. So you need steem-js library:

npm i -S [email protected]    # Specify the stable version of steem-js

With the above parameters, you publish a blog like this:

var steem = require('steem');
var posting_key = 'user1\'s-posting-key';
steem.broadcast.comment(
        posting_key,      // must be the private key that matches the author
        '',               // parent's author. Blogs to be empty string, replies to be the blog replied to
        'tutorial',       // parent’s permlink. Blogs to be the category; replies to be the blog replied to
        author,           // Steemit user name
        permlink,         // permanent link of this blog
        title,            // blog title; replies to be empty string
        body,             // blog body in markdown format; replies to be the reply message
        json_metadata,    // json_metadata contains tags; replies could be empty string
        function(err, re) {
    if (err) {
        throw err;
    } // if (err)
}); // steem.broadcast.comment( ... );

D. Coding III: schedule and repeat the fantastic job*

We use cron to schedule it:

npm i -S cron

It uses a string in the form of 6 fields: minute, hour, day of month, day of week, month, day of week. In each field, * means match every value, */5 means match every 5 times, and a specific value means match the value. So 00 05 01 * * * means match every day at 01:05:00AM. This is what we want -- to repeat everyday.

var CronJob = require('cron').CronJob;
new CronJob(
    '00 05 01 * * *',    // set the job to repeat at 01:05:00AM everyday
    function() {         // what to do at that time
        grabData();
    },
    null,                // what to do when stopped -- nothing
    true,                // shall we start the job right now? Of course
    'UTC'                // 01:05:00AM of which time zone
); // new CronJob( ... );

E. A need-to-know

Add this line to change the URL of steem API, otherwise when the task repeats an error will be raised by steem, causing the program crashed.

steem.api.setOptions({url: 'https://api.steemit.com'});

F. Run it

Go to menu Debug=>Add Configuration ..., then choose Node.js in the quick panel, the file .vscode/launch.json will be created automatically storing the launching configurations.

5.gif

Now press F5 to run it. After running, press F6 to pause it, Shift+F5 to stop it, and Ctrl+Shift+F5 to restart it.

To Conclude?

Nothing. Just check the full code, and remember to do npm i for the first run. Wish you good luck! :)

// ---- package.json ----
{
  "name": "steemitblogger",
  "version": "1.0.0",
  "description": "",
  "main": "index.js",
  "scripts": {
    "test": "echo \"Error: no test specified\" && exit 1"
  },
  "author": "",
  "license": "ISC",
  "dependencies": {
    "cheerio": "^1.0.0-rc.2",
    "cron": "^1.3.0",
    "steem": "^0.6.4"
  }
}
// ---- End of package.json ----
// ---- index.js ----
var cheerio = require('cheerio');
var CronJob = require('cron').CronJob;
var https = require('https');
var steem = require('steem');
// put date into the fields so they are diffrent each day
var now = new Date().toISOString();    // It's the current time in UTC timezone, in format of "2017-12-31T23:59:59Z",
                                       // where the trailing 'Z' refers to the UTC timezone.
var title = 'Hello World on ' + now.split('T')[0];    // this 'split' seperates the time string by 'T', and  takes
                                                      // the first part (JavaScript lists start with index 0)
var author = 'user1';
var body = 'Hello to everyone in the world at' + now + '\n\n$PLACE_HOLDER\n';
function grabData() {                                        // Put this job in a function
    https.get('https://github.com/Microsoft', function(res) {    // The 'get' procedure is standard,
        res.on('error', function(err) {                          // so the code snippet applies all similar cases 
            throw err;
        }); // res.on('error', function(err) { ... });
        var html = '';
        res.on('data', function(data) {
            html += data;
        }); // res.on('data', function(data) { ... });
        res.on('end', function() {
            var $ = cheerio.load(html);                         // This line parses HTML string to DOM with root '$'
            // Do something with the DOM
            var data = '';
            $('[itemprop="name codeRepository"]')    // use '[att="value"]' to select all element with target attribute/value
                    .each(function(i, e) {           // '.each' iterates every selected element
                data += ' + ' + e.children[0].data.trim() + '\n';    // the texts are stored in its first child's 'data' field
            }); // $( ... ).each( ... );
            body = body.replace('$PLACE_HOLDER', data);     // put the data to the correct place in body
            // Prepare and publish the blog
            // '/\s+/g' is a regular expression to match all continuous spaces
            var permlink = (author + '-' + title).replace(/\s+/g, '-');
            var json_metadata = {
                "tags":     ["tag1", "tag2", "tag3", "tag4"],    // the tag list you want add
                "app":      "steemit/0.1",                       // always be like this
                "format":   "markdown"                           // as mentioned above, body in markdown format
            }; // var json_metadata = { ... };
            var posting_key = 'user1\'s-posting-key';
            steem.broadcast.comment(
                    posting_key,      // must be the private key that matches the author
                    '',               // parent's author. Blogs to be empty string, replies to be the blog replied to
                    'tutorial',       // perent's permlink. Blogs to be the category; replies to be the blog replied to
                    author,           // Steemit user name
                    permlink,         // permanent link of this blog
                    title,            // blog title; replies to be empty string
                    body,             // blog body in markdown format; replies to be the reply message
                    json_metadata,    // json_metadata contains tags; replies could be empty string
                    function(err, re) {
                if (err) {
                    throw err;
                } // if (err)
            }); // steem.broadcast.comment( ... );
        }); // res.on('end', function() { ... });
    }); // https.get('https://github.com/Microsoft', function(res) { ... });
} // function grabData() { ... }
new CronJob(
    '00 05 01 * * *',    // set the job to repeat at 01:05:00AM everyday
    function() {         // what to do at that time
        grabData();
    },
    null,                // what to do when stopped -- nothing
    true,                // shall we start the job right now? Of course
    'UTC'                // 01:05:00AM of which timezone
); // new CronJob( ... );
// ---- End of index.js ----

All images and texts are created by @xuzhen. Thanks for your reading.



Posted on Utopian.io - Rewarding Open Source Contributors

Sort:  

Hey @xuzhen I am @utopian-io. I have just upvoted you!

Achievements

  • You have less than 500 followers. Just gave you a gift to help you succeed!
  • Seems like you contribute quite often. AMAZING!

Suggestions

  • Contribute more often to get higher and higher rewards. I wish to see you often!
  • Work on your followers to increase the votes/rewards. I follow what humans do and my vote is mainly based on that. Good luck!

Get Noticed!

  • Did you know project owners can manually vote with their own voting power or by voting power delegated to their projects? Ask the project owner to review your contributions!

Community-Driven Witness!

I am the first and only Steem Community-Driven Witness. Participate on Discord. Lets GROW TOGETHER!

mooncryption-utopian-witness-gif

Up-vote this comment to grow my power and help Open Source contributions like this one. Want to chat? Join me on Discord https://discord.gg/Pc8HG9x

your post is very interesting

Thank you for the contribution. It has been approved.

  • Please remove the last part where you ask for follow, upvote and resteem. Asking for follow, upvote, resteem can lead to rejection.
    ** Also remember not to add it in your next post.
    You can contact us on Discord.
    [utopian-moderator]

Thanks for your reminder. It has been removed.

@xuzhen 很羡慕你辛勤獲得的SP。請多多指教。
自上個星期開始玩steem之後,就定立目標向你門那些成功個案前進

-要發掘那些VOTE平均分佈的steemian post. 本人有編程底子,應該不難.
-不錯過任何機會向朋友推廣steemit.
-推廣在溫哥華華人對steemit認識.

本人剛發放了第一篇文章,可否點贊?
https://steemit.com/crytocurrency/@rksleung/price-target-take-it-with-a-grain-of-salt

已赞!:-)