From 1f6b18a1974c1b53a079b21b6be39af86deb2432 Mon Sep 17 00:00:00 2001 From: Kumar Rishabh Date: Thu, 29 Jun 2017 11:54:06 +0530 Subject: VNF_Catalogue Codebase Catalogue of Open Source VNFs consist in helping the end users to get information of the VNF we can deploy on top of an OPNFV solution [Deepak]: Removed all swp files. Change-Id: Ib2ea7330e964f1b684f32aedf631accd580df968 Signed-off-by: Kumar Rishabh Signed-off-by: Deepak S --- .../cronjob/3rd_party/wait-for-it/LICENSE | 20 +++ .../cronjob/3rd_party/wait-for-it/README.md | 75 ++++++++++ .../cronjob/3rd_party/wait-for-it/wait-for-it.sh | 161 +++++++++++++++++++++ VNF_Catalogue/cronjob/Dockerfile | 55 +++++++ VNF_Catalogue/cronjob/crontab | 3 + VNF_Catalogue/cronjob/database.js | 16 ++ VNF_Catalogue/cronjob/git_count_loc.sh | 11 ++ VNF_Catalogue/cronjob/github.js | 159 ++++++++++++++++++++ VNF_Catalogue/cronjob/package.json | 10 ++ VNF_Catalogue/cronjob/script.sh | 2 + 10 files changed, 512 insertions(+) create mode 100644 VNF_Catalogue/cronjob/3rd_party/wait-for-it/LICENSE create mode 100644 VNF_Catalogue/cronjob/3rd_party/wait-for-it/README.md create mode 100755 VNF_Catalogue/cronjob/3rd_party/wait-for-it/wait-for-it.sh create mode 100644 VNF_Catalogue/cronjob/Dockerfile create mode 100644 VNF_Catalogue/cronjob/crontab create mode 100644 VNF_Catalogue/cronjob/database.js create mode 100755 VNF_Catalogue/cronjob/git_count_loc.sh create mode 100644 VNF_Catalogue/cronjob/github.js create mode 100644 VNF_Catalogue/cronjob/package.json create mode 100644 VNF_Catalogue/cronjob/script.sh (limited to 'VNF_Catalogue/cronjob') diff --git a/VNF_Catalogue/cronjob/3rd_party/wait-for-it/LICENSE b/VNF_Catalogue/cronjob/3rd_party/wait-for-it/LICENSE new file mode 100644 index 00000000..bd18d0c4 --- /dev/null +++ b/VNF_Catalogue/cronjob/3rd_party/wait-for-it/LICENSE @@ -0,0 +1,20 @@ +The MIT License (MIT) +Copyright (c) 2016 Giles Hall + +Permission is hereby granted, free of charge, to any person obtaining a copy of +this software and associated documentation files (the "Software"), to deal in +the Software without restriction, including without limitation the rights to +use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies +of the Software, and to permit persons to whom the Software is furnished to do +so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/VNF_Catalogue/cronjob/3rd_party/wait-for-it/README.md b/VNF_Catalogue/cronjob/3rd_party/wait-for-it/README.md new file mode 100644 index 00000000..900b7573 --- /dev/null +++ b/VNF_Catalogue/cronjob/3rd_party/wait-for-it/README.md @@ -0,0 +1,75 @@ +# HELP NEEDED + +### March 13, 2017 update + +Since I posted this request for help, I've had a dozen or so responses which I am now sorting through. Applicants need to fill out [this](https://goo.gl/forms/GKCBFxaloaU47aky1) survey by March 17th. I will select, notify and announce the new volunteer(s) on March 19th. Once this is done, me and my team will work through the backlog of issues amd pull requests. Thanks for your paitence. + +### Old request follows: + +Hi there! I wrote `wait-for-it` in order to help me orchestrate containers I operate at my day job. I thought it was a neat little script, so I published it. I assumed I would be its only user, but that's not what happened! `wait-for-it` has received more stars then all of my other public repositories put together. I had no idea this tool would solicit such an audience, and I was equally unprepared to carve out the time required to address my user's issues and patches. I would like to solicit a volunteer from the community who would be willing to be a co-maintainer of this repository. If this is something you might be interested in, please email me at `waitforit@polymerase.org`. Thanks! + +## wait-for-it + +`wait-for-it.sh` is a pure bash script that will wait on the availability of a host and TCP port. It is useful for synchronizing the spin-up of interdependent services, such as linked docker containers. Since it is a pure bash script, it does not have any external dependencies. + +## Usage + +``` +wait-for-it.sh host:port [-s] [-t timeout] [-- command args] +-h HOST | --host=HOST Host or IP under test +-p PORT | --port=PORT TCP port under test + Alternatively, you specify the host and port as host:port +-s | --strict Only execute subcommand if the test succeeds +-q | --quiet Don't output any status messages +-t TIMEOUT | --timeout=TIMEOUT + Timeout in seconds, zero for no timeout +-- COMMAND ARGS Execute command with args after the test finishes +``` + +## Examples + +For example, let's test to see if we can access port 80 on www.google.com, and if it is available, echo the message `google is up`. + +``` +$ ./wait-for-it.sh www.google.com:80 -- echo "google is up" +wait-for-it.sh: waiting 15 seconds for www.google.com:80 +wait-for-it.sh: www.google.com:80 is available after 0 seconds +google is up +``` + +You can set your own timeout with the `-t` or `--timeout=` option. Setting the timeout value to 0 will disable the timeout: + +``` +$ ./wait-for-it.sh -t 0 www.google.com:80 -- echo "google is up" +wait-for-it.sh: waiting for www.google.com:80 without a timeout +wait-for-it.sh: www.google.com:80 is available after 0 seconds +google is up +``` + +The subcommand will be executed regardless if the service is up or not. If you wish to execute the subcommand only if the service is up, add the `--strict` argument. In this example, we will test port 81 on www.google.com which will fail: + +``` +$ ./wait-for-it.sh www.google.com:81 --timeout=1 --strict -- echo "google is up" +wait-for-it.sh: waiting 1 seconds for www.google.com:81 +wait-for-it.sh: timeout occurred after waiting 1 seconds for www.google.com:81 +wait-for-it.sh: strict mode, refusing to execute subprocess +``` + +If you don't want to execute a subcommand, leave off the `--` argument. This way, you can test the exit condition of `wait-for-it.sh` in your own scripts, and determine how to proceed: + +``` +$ ./wait-for-it.sh www.google.com:80 +wait-for-it.sh: waiting 15 seconds for www.google.com:80 +wait-for-it.sh: www.google.com:80 is available after 0 seconds +$ echo $? +0 +$ ./wait-for-it.sh www.google.com:81 +wait-for-it.sh: waiting 15 seconds for www.google.com:81 +wait-for-it.sh: timeout occurred after waiting 15 seconds for www.google.com:81 +$ echo $? +124 +``` + +## Thanks + +I wrote this script for my employer, [Ginkgo Bioworks](http://www.ginkgobioworks.com/), who was kind enough to let me release it as an open source tool. We are always looking to [hire](https://jobs.lever.co/ginkgobioworks) talented folks who are interested in working in the field of synthetic biology. diff --git a/VNF_Catalogue/cronjob/3rd_party/wait-for-it/wait-for-it.sh b/VNF_Catalogue/cronjob/3rd_party/wait-for-it/wait-for-it.sh new file mode 100755 index 00000000..eca6c3b9 --- /dev/null +++ b/VNF_Catalogue/cronjob/3rd_party/wait-for-it/wait-for-it.sh @@ -0,0 +1,161 @@ +#!/usr/bin/env bash +# Use this script to test if a given TCP host/port are available + +cmdname=$(basename $0) + +echoerr() { if [[ $QUIET -ne 1 ]]; then echo "$@" 1>&2; fi } + +usage() +{ + cat << USAGE >&2 +Usage: + $cmdname host:port [-s] [-t timeout] [-- command args] + -h HOST | --host=HOST Host or IP under test + -p PORT | --port=PORT TCP port under test + Alternatively, you specify the host and port as host:port + -s | --strict Only execute subcommand if the test succeeds + -q | --quiet Don't output any status messages + -t TIMEOUT | --timeout=TIMEOUT + Timeout in seconds, zero for no timeout + -- COMMAND ARGS Execute command with args after the test finishes +USAGE + exit 1 +} + +wait_for() +{ + if [[ $TIMEOUT -gt 0 ]]; then + echoerr "$cmdname: waiting $TIMEOUT seconds for $HOST:$PORT" + else + echoerr "$cmdname: waiting for $HOST:$PORT without a timeout" + fi + start_ts=$(date +%s) + while : + do + (echo > /dev/tcp/$HOST/$PORT) >/dev/null 2>&1 + result=$? + if [[ $result -eq 0 ]]; then + end_ts=$(date +%s) + echoerr "$cmdname: $HOST:$PORT is available after $((end_ts - start_ts)) seconds" + break + fi + sleep 1 + done + return $result +} + +wait_for_wrapper() +{ + # In order to support SIGINT during timeout: http://unix.stackexchange.com/a/57692 + if [[ $QUIET -eq 1 ]]; then + timeout $TIMEOUT $0 --quiet --child --host=$HOST --port=$PORT --timeout=$TIMEOUT & + else + timeout $TIMEOUT $0 --child --host=$HOST --port=$PORT --timeout=$TIMEOUT & + fi + PID=$! + trap "kill -INT -$PID" INT + wait $PID + RESULT=$? + if [[ $RESULT -ne 0 ]]; then + echoerr "$cmdname: timeout occurred after waiting $TIMEOUT seconds for $HOST:$PORT" + fi + return $RESULT +} + +# process arguments +while [[ $# -gt 0 ]] +do + case "$1" in + *:* ) + hostport=(${1//:/ }) + HOST=${hostport[0]} + PORT=${hostport[1]} + shift 1 + ;; + --child) + CHILD=1 + shift 1 + ;; + -q | --quiet) + QUIET=1 + shift 1 + ;; + -s | --strict) + STRICT=1 + shift 1 + ;; + -h) + HOST="$2" + if [[ $HOST == "" ]]; then break; fi + shift 2 + ;; + --host=*) + HOST="${1#*=}" + shift 1 + ;; + -p) + PORT="$2" + if [[ $PORT == "" ]]; then break; fi + shift 2 + ;; + --port=*) + PORT="${1#*=}" + shift 1 + ;; + -t) + TIMEOUT="$2" + if [[ $TIMEOUT == "" ]]; then break; fi + shift 2 + ;; + --timeout=*) + TIMEOUT="${1#*=}" + shift 1 + ;; + --) + shift + CLI="$@" + break + ;; + --help) + usage + ;; + *) + echoerr "Unknown argument: $1" + usage + ;; + esac +done + +if [[ "$HOST" == "" || "$PORT" == "" ]]; then + echoerr "Error: you need to provide a host and port to test." + usage +fi + +TIMEOUT=${TIMEOUT:-15} +STRICT=${STRICT:-0} +CHILD=${CHILD:-0} +QUIET=${QUIET:-0} + +if [[ $CHILD -gt 0 ]]; then + wait_for + RESULT=$? + exit $RESULT +else + if [[ $TIMEOUT -gt 0 ]]; then + wait_for_wrapper + RESULT=$? + else + wait_for + RESULT=$? + fi +fi + +if [[ $CLI != "" ]]; then + if [[ $RESULT -ne 0 && $STRICT -eq 1 ]]; then + echoerr "$cmdname: strict mode, refusing to execute subprocess" + exit $RESULT + fi + exec $CLI +else + exit $RESULT +fi diff --git a/VNF_Catalogue/cronjob/Dockerfile b/VNF_Catalogue/cronjob/Dockerfile new file mode 100644 index 00000000..350411aa --- /dev/null +++ b/VNF_Catalogue/cronjob/Dockerfile @@ -0,0 +1,55 @@ +############################################################### +# Docker container for VNF_Catalogue cronjob service +############################################################### +# Purpose: Don't run it from here! Use docker-compose(See README.md) +# +# Maintained by Kumar Rishabh :: penguinRaider +## +# All rights reserved. This program and the accompanying materials +# are made available under the terms of the Apache License, Version 2.0 +# which accompanies this distribution, and is available at +# http://www.apache.org/licenses/LICENSE-2.0 +# + +FROM node:boron +MAINTAINER KumarRishabh::penguinRaider +LABEL version="v0.0.1" description="Open Source VNF_Catalogue for OPNFV" + +RUN apt-get update +RUN apt-get install vim -y +RUN apt-get install cron -y + +ENV DB_HOST mysql +ENV DB_USER vnf_user +ENV DB_PASSWORD vnf_password +ENV DB_DATABASE vnf_catalogue + +RUN mkdir -p /usr/src/app +WORKDIR /usr/src/app + +COPY package.json /usr/src/app/ + +# RUN npm config set proxy http://10.4.20.103:8080 +# RUN npm config set https-proxy http://10.4.20.103:8080 + +RUN npm install + +# ADD crontab /etc/cron.d/simple-cron + +COPY . /usr/src/app + +RUN chmod +x git_count_loc.sh +RUN chmod +x script.sh + +RUN crontab crontab +RUN sed -i '/session required pam_loginuid.so/c\#session required pam_loginuid.so' /etc/pam.d/cron + +# Give execution rights on the cron job +# RUN chmod 0644 /etc/cron.d/simple-cron +# +# # Create the log file to be able to run tail +RUN touch /var/log/cron.log + +# The ordering of events should be coming up of mysql service and then running +# of cronjob. To enforce this causal relationship we use a 3rd_party script. +CMD [ "./3rd_party/wait-for-it/wait-for-it.sh", "mysql:3306", "-t", "0", "--", "cron", "&&", "tail", "-f", "/var/log/cron.log"] diff --git a/VNF_Catalogue/cronjob/crontab b/VNF_Catalogue/cronjob/crontab new file mode 100644 index 00000000..892f149d --- /dev/null +++ b/VNF_Catalogue/cronjob/crontab @@ -0,0 +1,3 @@ +0 * * * * /usr/src/app/script.sh +# * * * * * root /usr/local/bin/node github +# An empty line is required at the end of this file for a valid cron file. diff --git a/VNF_Catalogue/cronjob/database.js b/VNF_Catalogue/cronjob/database.js new file mode 100644 index 00000000..8c6d0bbf --- /dev/null +++ b/VNF_Catalogue/cronjob/database.js @@ -0,0 +1,16 @@ +var mysql = require('mysql'); + +// Hardcoding some variables for now as crontab does not seem to support +// environment variables. +var pool = mysql.createPool({ + host: 'mysql', + user: 'vnf_user', + password: 'vnf_password', + database: 'vnf_catalogue', + connectionLimit: 50, + supportBigNumbers: true, + multipleStatements: true, + dateStrings: 'date' +}); + +exports.pool = pool; diff --git a/VNF_Catalogue/cronjob/git_count_loc.sh b/VNF_Catalogue/cronjob/git_count_loc.sh new file mode 100755 index 00000000..3b316666 --- /dev/null +++ b/VNF_Catalogue/cronjob/git_count_loc.sh @@ -0,0 +1,11 @@ +folder_name=$(head /dev/urandom | tr -dc A-Za-z0-9 | head -c 13 ; echo ''); + +git clone $1 $folder_name 2> /dev/null +if [ "$?" -ne "0" ]; then + echo 0; + exit 1; +fi +cd $folder_name; +(git ls-files | xargs cat | wc -l) 2> /dev/null; +cd ../; +rm -rf $folder_name; diff --git a/VNF_Catalogue/cronjob/github.js b/VNF_Catalogue/cronjob/github.js new file mode 100644 index 00000000..cc6e35f6 --- /dev/null +++ b/VNF_Catalogue/cronjob/github.js @@ -0,0 +1,159 @@ +// Important Add your access token here default rate of github is limited to 60 API calls per hour +var access_token = '523825265638c8f2b5b9f1d1f5af31d71bceb49c'; +// Important set the delta threshold for repo details updation. For instance if the threshold is +// set to 1 day(60 * 60 * 24), the cronjob will only update the row if the difference between current +// time and last_updated time stamp of a repo is greater than one day +var delta = 60 * 60 * 24; + + +var github = require('octonode'); +db_pool = require('/usr/src/app/database').pool; +async = require('async'); + +var current_time = Math.floor(new Date().getTime() / 1000);//toISOString().slice(0, 19).replace('T', ' '); +console.log(current_time); + +var get_val_from_header = function(header_link) { + // small hack by parsing the header and setting per_page = 1, hence no pagination fetch required + result_intermediate = header_link.split(';'); + result_intermediate = result_intermediate[result_intermediate.length - 2]; + var reg = /&page=([0-9].*)>/g; + var match = reg.exec(result_intermediate); + return parseInt(match[1]); +} + +var get_stargazers = function(result, ghrepo, orig_repo, primary_callback, cb) { + ghrepo.stargazers({per_page: 1}, function(err, data, headers) { + //console.log(JSON.stringify(data)); + try { + result['no_of_stars'] = get_val_from_header(headers['link']); + cb(null, result, ghrepo, orig_repo, primary_callback); + } catch(err) { + result['no_of_stars'] = null; + cb(null, result, ghrepo, orig_repo, primary_callback); + } + }); +} + +var get_branches = function(result, ghrepo, orig_repo, primary_callback, cb) { + ghrepo.branches({per_page: 1}, function(err, data, headers) { + try { + result['versions'] = get_val_from_header(headers['link']); + cb(null, result, ghrepo, orig_repo, primary_callback); + } catch(err) { + result['versions'] = null; + cb(null, result, ghrepo, orig_repo, primary_callback); + } + }); +} + +var get_contributors = function(result, ghrepo, orig_repo, primary_callback, cb) { + ghrepo.contributors({per_page: 1}, function(err, data, headers) { + try { + result['no_of_developers'] = get_val_from_header(headers['link']); + cb(null, result, ghrepo, orig_repo, primary_callback); + } catch(err) { + result['no_of_developers'] = null; + cb(null, result, ghrepo, orig_repo, primary_callback); + } + }); +} + +var get_lines_of_code = function(result, ghrepo, orig_repo, primary_callback, cb) { + // #TODO + console.log('here'); + + console.log(orig_repo); + const + spawn = require( 'child_process' ).spawn, + ls = spawn( 'bash', ['/usr/src/app/git_count_loc.sh', orig_repo] ); + + ls.stdout.on( 'data', data => { + console.log( `stdout: ${data}` ); + result['lines_of_code'] = parseInt(`${data}`.replace(/\n$/, ''), 10); + //cb(null, result, primary_callback); + }); + + ls.stderr.on( 'data', data => { + //console.log( `stderr: ${data}` ); + //cb(null, result, primary_callback); + }); + + ls.on( 'close', code => { + console.log( `child process exited with code ${code}` ); + cb(null, result, primary_callback); + }); +} + +var secondary_callback = function (err, result, primary_callback) { + console.log(result); + if((result['last_updated'] == null) || (current_time - result['last_updated'] > delta)) { + db_pool.getConnection(function(err, connection) { + //Use the connection + var last_updated = current_time; + var no_of_stars = result['no_of_stars']; + var versions = result['versions']; + var no_of_developers = result['no_of_developers']; + var lines_of_code = result['lines_of_code']; + console.log('here'); + console.log(lines_of_code); + sql_query = 'update vnf set last_updated = FROM_UNIXTIME(' + last_updated; + sql_query += '), no_of_stars = ' + no_of_stars + ', versions = ' + versions; + sql_query += ', lines_of_code = ' + lines_of_code; + sql_query += ', no_of_developers = ' + no_of_developers + ' where vnf_id = '; + sql_query += result['vnf_id']; + console.log(sql_query); + connection.query(sql_query, function (error, results, fields) { + //if (error) throw error; + //And done with the connection. + connection.release(); + if(error) { + primary_callback(null, result['vnf_id'] + ' not updated'); + } else { + primary_callback(null, result['vnf_id'] + ' updated'); + } + // Handle error after the release. + // Don't use the connection here, it has been returned to the pool. + }); + }); + } else { + primary_callback(null, result['vnf_id'] + ' not updated'); + } +} + +var get_stats = function(vnf_details, callback) { + repo = vnf_details['repo_url']; + orig_repo = repo; + repo = repo.split("/"); + github_id = repo[repo.length - 2] + '/' + repo[repo.length - 1]; + + var async = require('async'); + var client = github.client(access_token); + var ghrepo = client.repo(github_id); + + result = {} + result['vnf_id'] = vnf_details['vnf_id']; + result['last_updated'] = vnf_details['last_updated']; + + async.waterfall([ + async.apply(get_stargazers, result, ghrepo, orig_repo, callback), + get_branches, + get_contributors, + get_lines_of_code, + ], secondary_callback); +} + +db_pool.getConnection(function(err, connection) { + sql_query = 'select vnf_id, repo_url, UNIX_TIMESTAMP(last_updated) last_updated from vnf'; + console.log(sql_query); + connection.query(sql_query, function (error, results, fields) { + if (error) throw error; + async.map(results, get_stats, function(error, results) { + //console.log(results); + console.log(results); + process.exit(); + + }); + }); +}); + diff --git a/VNF_Catalogue/cronjob/package.json b/VNF_Catalogue/cronjob/package.json new file mode 100644 index 00000000..1b7fb1b5 --- /dev/null +++ b/VNF_Catalogue/cronjob/package.json @@ -0,0 +1,10 @@ +{ + "name": "VNF_Catalogue_cron_jobs", + "version": "0.0.1", + "private": true, + "dependencies": { + "octonode": "*", + "async": "*", + "mysql": "*" + } +} diff --git a/VNF_Catalogue/cronjob/script.sh b/VNF_Catalogue/cronjob/script.sh new file mode 100644 index 00000000..4b5a8d26 --- /dev/null +++ b/VNF_Catalogue/cronjob/script.sh @@ -0,0 +1,2 @@ +#!/bin/bash +/usr/local/bin/node /usr/src/app/github >> /tmp/mybacklog 2>&1 -- cgit 1.2.3-korg