Monday, November 12, 2012

loop variable is leaked?

Python has scope rules, but there are a few minor exceptions. Example: the loop variable leaks.

I was learning about MongoDB tonight and the instructor went over some basic Python code. I would normally skip them but as a learn I was skimming through each Python lecture to make sure nothing strikes me (I don't want to miss any COOL tricks). Well, Python: for loops with lists caught my attention.

The loop variable item was accessible and used post-loop. HOW! I thought it was local. 
My first thought was "damn could this be a static thing? It doesn't look right!". I fired up a shell, typed in some code, and tried the same code and the instructor wasn't lying! 

I dissembled the code in the shell, and this is what I get:

>>> def f():
...     l = ['1', '2', '3']
...     l2 = []
...     for item in l:
...             pass
...     l2.append(item)
...     print l2
... 
>>> f()
['3']
>>> import dis
>>> dis.dis(f)
  2           0 LOAD_CONST               1 ('1')
              3 LOAD_CONST               2 ('2')
              6 LOAD_CONST               3 ('3')
              9 BUILD_LIST               3
             12 STORE_FAST               0 (l)

  3          15 BUILD_LIST               0
             18 STORE_FAST               1 (l2)

  4          21 SETUP_LOOP              14 (to 38)
             24 LOAD_FAST                0 (l)
             27 GET_ITER            
        >>   28 FOR_ITER                 6 (to 37)
             31 STORE_FAST               2 (item)

  5          34 JUMP_ABSOLUTE           28
        >>   37 POP_BLOCK           

  6     >>   38 LOAD_FAST                1 (l2)
             41 LOAD_ATTR                0 (append)
             44 LOAD_FAST                2 (item)
             47 CALL_FUNCTION            1
             50 POP_TOP             

  7          51 LOAD_FAST                1 (l2)
             54 PRINT_ITEM          
             55 PRINT_NEWLINE       
             56 LOAD_CONST               0 (None)
             59 RETURN_VALUE        

This doesn't really help me. I don't remember all the goodies about dis from Python conference last summer. So instead, I googled this first.
http://stackoverflow.com/questions/3611760/scoping-in-python-for-loops

I wasn't the first one. The accepted answer referenced a mailing-list discussion on mails.python.org.
The last value of the loop variable (in this case, item) will live in the surrounding scope.

l = ['1', '2', '3']
l2 = []

for item in l:
    print id(item)
    pass
l2.append(item)
print id(item)

The result is as follow:

yeukhon@yeukhon-P5E-VM-DO:~/tmp$ python wtf.py
3077889320
3077837144
3077839616
3077839616
The first three are the three iterations of the list l and we can see the last output matches the last iteration. This means the value persists. The value continues to reside.

I didn't stop here. I went on. I wanted to find out the underlying assembler code. So I use Cython to translate my python code into C code.

I removed the print statements from the previous code, and then I generated this C code. The translator did an amazing job. Sadly, for some strange reason I couldn't compile it because Python.h was not found. I probably have to setup the library so the compiler can discover that file? I already have python-dev (and python2.7-dev) installed. Probably....

In any case, a few things:
I know that python keeps a namespace, and they are keys/values pair. You can see the static char starting from line 480.

/* Implementation of 'wtf' */
static char __pyx_k__1[] = "1";
static char __pyx_k__2[] = "2";
static char __pyx_k__3[] = "3";
static char __pyx_k__l[] = "l";
static char __pyx_k__l2[] = "l2";
static char __pyx_k__item[] = "item";
static char __pyx_k____main__[] = "__main__";
static char __pyx_k____test__[] = "__test__";

Now I want to dive deeper and see how they keep the last value in the surrounding scope.

Starting from line 655 we enter the for loop. The code is a bit messy to read at first. Many of the Py_* are actually CPython's C API. I have to google some of them to understand what they do and what their return values are. My quest is to find the last reference to the object holding the value item is not garbage collected.

I explained the relevant portion of the code here. It's very hard to read, so make sure you are prepared to read it...

But to summarize, at the end of the loop,

if (PyObject_SetAttr(__pyx_m, __pyx_n_s__item, __pyx_t_1) < 0) {__pyx_filename = __pyx_f[0]; __pyx_lineno = 4; __pyx_clineno = __LINE__; goto __pyx_L1_error;}

This line saves the value of the last iteration into the static variable item in memory! After this, the append statement looks like this.

  __pyx_t_2 = __Pyx_GetName(__pyx_m, __pyx_n_s__l2); if (unlikely(!__pyx_t_2)) {__pyx_filename = __pyx_f[0]; __pyx_lineno = 6; __pyx_clineno = __LINE__; goto __pyx_L1_error;}
  __Pyx_GOTREF(__pyx_t_2);
  __pyx_t_1 = __Pyx_GetName(__pyx_m, __pyx_n_s__item); if (unlikely(!__pyx_t_1)) {__pyx_filename = __pyx_f[0]; __pyx_lineno = 6; __pyx_clineno = __LINE__; goto __pyx_L1_error;}
  __Pyx_GOTREF(__pyx_t_1);

As you see,  we get the value directly from __pyx_n_s__item. It's a static variable.

Lesson learned:
1. First thing I recognized was all the ugly branch statements in the code. It might be just the translation (btw, Cython did a great job though... I can't even dare to think how I would go about writing such translator.... geesh). But still, under the hook, there are lots of checks. Branch statements are very expensive. So if this was not optimized, we would waste lots of cycles!

2. Static is everywhere in the translated C code! Look at the object variable item. It's static.
static PyObject *__pyx_n_s__item;

3. It's a great challenge to read all these codes. I remember all these SET, INCREF stuff from my programming language paradigm class which I took last year. It's cool to see these guys again. Honestly, if I weren't doing this, I wouldn't be digging through the C API on python.org

Some references:
http://docs.python.org/2/c-api/refcounting.html
http://stackoverflow.com/questions/4657764/py-incref-decref-when
http://docs.python.org/2/c-api/structures.html
http://docs.python.org/2/c-api/list.html
http://wiki.cython.org/enhancements/refnanny
http://wiki.cython.org/enhancements/pypy
http://mail.python.org/pipermail/python-ideas/2008-October/002109.html
http://stackoverflow.com/questions/2525518/writing-code-translator-from-python-to-chttp://docs.python.org/2/library/dis.html


Sunday, November 11, 2012

static in Python - 1

My friend Dave (also my ex-TA) always points me to interesting posts on StackOverflow after I introduced it to him. He likes helping others to succeed whenever possible. I admire him a lot. He's a great mentor.

He was my instructor for my Computer Organization lab a year ago. Last night he linked me to What and where are the stack and heap?. That was his response. In our class, Dave went over scope and lifetime of static and automatic variables. I still remember most of the details. But he got very excited when I said Python programmers don't really need to worry about static allocation because for such a high-level language Python programmers just pass things around.

He wrote an example to illustrate how static variable(s) was used in Python. Later tonight he went over how he used this in his Database homework. He was writing a class for functional dependency and multi-value dependency. He subclass FD to make MVD. He wanted to show A -> B and A ->-> B. But instead of hardcoding the string -> and ->-> he thought of static variable.

Idea:

class FD:
	_ARROW_STRING = "->"

class MVD(FD):
        _ARROW_STRING = "->->"  # or even just * 2 I think that's fine too

I think that's the idea. When using static variable, all instances of FD and MVD will refer to the same _ARROW_STRING which saves memory. Imagine we have 10000 instances of FD. That's a lot of saving.

Sometimes, I as a Python programmer, overlook the old school C++ stuff because they look boring and confusing. But sometimes, a small trick like that will help. Global variables are okay too, but if we want to encapsulate it into the class, we might as well just make it as a static member variable. I mean, global variables are also statically allocated. It doesn't really make any difference in general.

In the next post I will mention more about statics in Python.... sometimes bugs me right now.

Sunday, October 14, 2012

Procedural or Class-based OOP for a simple python library?

Python is an OOP language. Everything in Python is an object. Functions are first-class objects which mean you can pass them around like objects. Try that in languages like C or C++. I am pretty sure you can do it (just google that on StackOverflow), but I am also pretty sure it would be very very very difficult to achieve. I actually asked a similar question at one point...

As a library designer, I am always seeking the best design patterns. I want my code to be clean, small and general. I don't want to make too much assumptions. I want it to be maintainable and easy to use.

One of the libraries we planned to release in the future as part of the Graphyte Project is called RepoMan. This libraries uses the Mercurial commands to perform init, clone, push, pull, add, addremove, and commit. My friend Jeremy did the core implementation and I did the external functionality such as write to a file, read a file, or rename a directory. 90% of what we did in this library were plain I/O operations using the os and shutil libraries. We didn't have a single class in the original design. Not at all.

Tonight I started implementing some hooks for SCM-Manager to talk to its REST apis. I used requests to communicate over HTTP. As usual, the first thing I did was did some initial coding and wrote a few lines of tests.

My initial codes were class-based. I thought about writing the entire thing in pure functions, but then I remember that our webclient will probably benefit from having an object instantiated and passed around, instead of having to reconstruct or saving those kwargs (stuff we need for the repo communication, such as username, password, repo url, etc) all the time.

Clearly, in that case, class-objects helps a lot. With functions, you really can't.You usually don't have a way to store the state. In fact, with functions, states should be one-time, and shouldn't be persistent.... closure is not a state-storage counterpart. Closure is just a form of lazy evaluation to me. Once you complete the function, the state is lost.

I thought about function vs class-object. I finally went with functions in my initial implementation. I realized that for my own libraries, the lowest level of the libraries (or the core) should be simple and straightforward. I could write these independent, individual functions and then implement the interface as I wish. Users can easily pull one of these functions and make use of it if they don't need a full-featured interface.

If I have all the REST apis written as functions, then I can make a very simple Python class-based interface that calls these apis individually. That interface is what people use. It should be simple to use. If I started writing all the cores into the methods, I am revealing the implementation. I am not abstracting it from the end-user. In the end, I ended up writing ugly codes in my methods.

I may be handling some logics after calling some functions in my methods, but that piece of logic code is specified to that method. If I don't like what I did for the core, I can easily killed the core and replaced that old function with a new implementation. My interface will usualy survice.

The conclusion is that when writing a library, stay focus on what the library intends to do. If the library is some complex data structure, the lowest level could be a data structure, which in C or C++ could be a class or struct. There is nothing wrong with that. But even in C or C++, the implementation and definition are always separated in best practices. Thus, the lowest-level of the implementation should be independent from the interface. Anyone who wish to use my libary can easily import a particular function (e.g. a user may be interested in reading the content of a file using the scm-hook). That import will give the user the full control of that piece of functionality.

Stop writing everything as classes in Python. Most of the stuff in Python should remain as functions. Classes usually make codes harder to debug and maintain.

Saturday, October 13, 2012

presentation is not so easy to make

I started leading a beginner Python workshop this past Tuesday and Thursday at school. It didn't go too well from what I can tell. I lost like half of the students on Thursday. I hope I can see them again on next Thursday.

I learned my lesson.

1. don't pay too much attention on pretty slides
They become useless if they don't help. I was attracted to Google HTML5 Slides  since I attended the 2011 I/O recap here in New York. It was really cool to see that. I thought why not make one for my workshop?
I spent so many hours on it and I thought I could write the scripts either in reStructureText or Markdown, but it turned out most of these libraries out there are sort of "broken". Their CSS don't work very nice. I ended up using Shower and did the entire thing by hands. Yeah. Hard-code all the htmls myself.
Next time, I will stick to traditional powerpoint.

2. limit the goals
At first, I wanted this talk to be as comprehensive as possible. Although I had a couple juniors and seniors from CS came to learn Python, but most of my students don't have programming experiences at all.
I modified my slides a few hours before the talk. I realized the goal must be lowered to introduction to programming. Enough to get by. I still ended up too much.

3. test your equipment
Don't rely on previous experiences. They break and they change over time. For example, the first workshop (which was actually my talk on Python and introduction to programming), I spent nearly 30 minutes trying to hook up the projector and fixing my slides. Freakin' mad....
On Thursday I had to deal with this thing called PyLab and I didn't know how to run a script (I can drag the .py file into PyLab to let it execute but it sucks..). I planned to write little script on IDLE and I couldn't.

4. get a better room
Seriously... 7/107 sucks. That projector requires me to turn off the light which is a terrible thing. I hate that. Students have to keep turning their heads over me.... that's bad for teaching.
None of the CS labs are setup such that the computers can face the projector. That's why I asked for the engineering room at first because it has a good setup...

Anyhow, I posted my stuff here
http://yeukhon.bitbucket.org/

R.I.P Amanda Todd



This is supposed to be a programming blog. But I am going to take a little detour. I was supposed to be refactoring Aurum right now. 

I just spoke to my girlfriend and she told me about this suicidal video. I just want to be very honest to the audience out there: my first thought was to check her out. I want to know why she killed herself.

It was morally wrong for me to check her out in the first place... my bad.

But I want to say that her connections were totally fucked up. First of all, that motherfucker player cheated on his girlfriend, and that dumbass bitch girlfriend thought it was worth the trouble to teach Amanda a little lesson. Both of them are stupid. A good girl knows a cheater is worthless. 

Secondly, all the observers in her life were totally fucked up too. Just because her nude photo was posted online, none of you should moved away from her. She just need a few real greeting to make her feel stronger. None of you did. 

If I were there, I am just being serious, I would stop the fucking fight and pull her aside. I would defend the shit out of her just because I think it was wrong to beat anyone up. She already paid the price.. it was a public humiliation. 

I believe her family was divorced. She probably needed a complete family. I don't want to make any more conjecture, but family support was very important. People in her life were generally bad and cruel.

Now you bullies have to live with this death mark forever. You guys will constantly remember Amanda from time to time. When you are at your wedding, you might remember Amanda. When you are making love with a girl, you fucking player, you are going to remember that your penis was once inside this beautiful, yet, dead, girl. She was only looking for a little love from people. Just a little bit, and you ruined it.

God. Please have mercy on Amanda. I know she killed herself, but please make her feel comfortable.

Friday, October 5, 2012

Learning SCRUM with LEGO

So Michael bought some legos to each scrum for this MIS software engineering course. It would be fun to do it in the lab so we scheduled the event today.

Actually I was late today so I missed the first 10 mins. But it was a video showing what scrum was about.

The whole game was divided into several sessions. Michael followed the rule very strictly by timing us. Since there was only five members, we made a big team instead of dividing into sub-teams.

We write down the user stories (the stuff we need to build) on Post-its. Then we put up a big poster paper on the wall. We first grouped them up by the level of difficulty. I actually disagreed with the arrangement we had but majority won.

Anyhow, then we put up another poster and drew three lines, each line is called a sprint (period), so there was three sessions for us to complete. I think each is 15 minutes.

We tried to construct all the vehicle in the first sprint, but we failed. SUV was the only one that actually looked usable, whereas the bus and the tractor were totally useless.


We spread the legos on the floor.

In any case, after we finished the first sprint, we had a short recap session. We were asked to think what went wrong. The first thing was resources weren't given ahead of time and we had to search for pieces which proved to be very difficult. Just look at the picture above. The only thing that was completed was the bridge. The bridge, like I said, would be the easiest....


That's Yuriy.

We continued to perfect our cars and build other stuff which I can't recall now. But  it didn't go very well either. The bus was totally broken when we showed it to the product owner (Michael) in this case.

Our final product.

When it came to the last phase, we still had to finish the bus. I begged to work on the tower crane because it was worth five points and I knew it was a simple task. 
We did most of it, but as you can see in the photo, we didn't manage to put covers over the house or the garage.  

The requirement said the garage must be able to protect the cars from bad weather. Jeremy said "well it can protect horizontal rains." LOL
Recap!
The final phase was reflection. We pointed out that resources weren't given ahead of time, and we should have collected the pieces and put them into groups in the first sprint. We also need to re-weigh each user stories. For example, houses are very easy to make and they should be done as soon as possible. But the dilemma was we need to need how big the cars are ahead of time so the garage and bus stop could be build. 

This nice exercise shows us several things:
1. we can't construct a perfect plan
2. we can't overestimate or underestimate any task
3. we need to be able to work as a team or things will fall apart

It's a nice exercise. We have a lot of fun.
This feels like one of thoses Google internal training event :P

Wednesday, September 26, 2012

Use Fabric more!

I have been really busy these days. I have many things I want to write about. Oh please good nature let me have some free times to write about things!

Anyway. I just happened to be on StackOverflow. There were two questions on sending files remotely. One was copying over to another machine, and another through FTP.

I am against using raw connection. Fabric is designed for deployment, so I am a strong advocate of using it.

How to automate the sending of files over the network using python?
Python Script Uploading files via FTP

I wrote the scripts very quickly.

    #!/usr/bin/python
    from fabric.api import run, env, sudo, put

    env.user = 'username'
    env.host = ['hostname,]
    #env.password = 'my_lovely_password'

    def copy():
        put('wong_8066.zip', '/home1/yeukhon/wong_8066.zip')

    def copy2():
        put('wong_8066.zip', '/www/public/wong_8066.zip')


You can save it as fabfile.py and run it via by calling fab task_name  or multiple task names all together.

If you need to automated it you should specify the path to the ssh key. Hardcoding the password (for FTP, for example) is OK if this is a personal thing.

Seriously... use Fabric more. Sadly, GLASSLAB needs a better automation system so we have to go with Chef.

Friday, September 14, 2012

Mocking sucks! I hate mocking!

I use Mock very frequently in my unittests because I need to isolate my system under test. But I hate mocking because it always give me problems. The documentation gives made-up examples that are independent from other sections in the documentation.

Fine. I am fine with that, but why the hell do I need to use .return_value all the time in my mock object?

I am consistently biting myself with mock library and I have wasted more time on mocking my objects more than me watching a two-hour movie.

The number one thing I hate about mocking is when it returns a Mock object when I call it. Like what the fuck man? Seriously.

Here is the latest issue I have with mocking: [TIP] Mock returns a mock using autospec

If this was such an easy question people would have answer it a long time ago. Why none? Because it is so god damn hard I think!

If there was a different library to do mocking I'd go with that AS LONG AS IT IS SIMPLE TO MOCK.

Monday, August 27, 2012

Session and cookies

Aurum uses Pyramid. I am not familiar with Pyramid but it's very flexible because unlike Django, Pyramid does not come with a User system by default when you activated the setup. It just doesn't. You can create a User model and use the Session abstraction built into Pyramid (and the Auth policy that comes with Pyramid).

As a Django user, I don't need to worry about how session was managed. It just works for each user. Now I knew I have to care about it.

I still don't know much about the whole architecture, but basically I was using pyramid-beaker and added a key to session. But the next time I access session and asks to read the key, it raises KeyError.

I made this post, and I am not embarrassed to show. I actually have several Stackoverflow accounts.

Why is that? As the answer pointed out, to get the same session back, it needs to have the same cookies. That's pretty much the way to identity "oh so I see you again."

Then I asked myself: what if I lose that cookies or cookies changed? Wouldn't that mean I will lose the session information? My requirement is to store user actions into the session / cache so I can retrieve them on demand.

When a user runs the code, we do short-polling at the moment to requests "build, run, generate report". Each of these steps will require / make some ids or status code. I need to keep track of these things. I can't afford asking GCS "oh by the way can you please give me that build_id again?" Instead, I can go right into session / cacher and ask "given this task id, what's the build_id if any?"

What about use memcache and forget about session. Make a key/value pair in memcache that uses the task id and when we request the resource could look like /aurum/api/task/status/<task_id>

That's how we do it currently. But that's really bad. Any one with this task id can actually query the status. If I knew my classmate is really smart, and I knew he did it right. I am going to steal that task id as he's running it, and then put it in my javascript, then when I make the next request, it will blindly asks for <my-friend-task-status> because the task id is his. What's next? When it's completed, I get all green on my test output and a big fat 100 for my homework!!!!

Session is the way to go. With cookies, there is a barrier over getting that cookies. I am not sure whether copying a cookies over to a new browser will actually invalidate the cookies. It might? But with session, I can guarantee that anyone who looks that this session and using the data from this session are the user who created this session last time.

So I am sticking with Session, although I still prefer to use Memcache because it looks more professional? I don't know.

invalid date format in specification: "2011-08-25 00:00:00.000000000Z"

This is an annoying problem with the outdated Ruby installation on many of the Debian/Ubuntu apt-get.

Invalid gemspec in [/var/lib/gems/1.8/specifications/tilt-1.3.3.gemspec]: invalid date format in specification: "2011-08-25 00:00:00.000000000Z"

One fix is to upgrade to ruby1.9 but I don't want to. What if it messes up my other configuration? A quick way is this:

sudo sed -i 's/ 00:00:00.000000000Z//' /var/lib/gems/1.8/specifications/*

This solution is taken from here. But every time when you install a new gem (pip package, in Python's world), the problem will appear. But it's safe to do it again.


openshift folks are helpful

So recently I was looking for free PHP instance online, similar to Heroku if possible. Then I found RedHat's OpenShift.

After signing up, I created a Wordpress right away. The interface looks simple and awesome actually. Here are two screenshots of the creating application console:
http://imgur.com/a/5CDyE

The OpenShift folks were very helpful, even though I was a free user. The IRC channel is not dead. People actually can respond, although they would point you to some blog posts about how to do things (written by the community / OpenShift people). But it's great.

The command-line client `rhc` looks cool too. It takes care of the ssh stuff for you. If you want to access the database for example, you could port-forward it and run a command-line option to call phpmyadmin, and the actual GUI runs very smoothly.

You also get to accessed to the local machine. I guess that's the trend these cloud companies are doing.

It also provide Jenkins! Oh wow.

I recommend it if you are considering a mini site.


Here are links I want to keep:

https://openshift.redhat.com/community/faq/how-can-i-test-my-database-connection-from-my-local-client
https://openshift.redhat.com/community/get-started

Wednesday, August 22, 2012

Chef's python cookbook

I am quite impressed with the Chef's Python cookbook. Initially I thought that python_pip does not come with the ability to install packages using requirements.txt.

It turns out it does:
[19:19] <jameson_> Guest94181: you can use the "option" parameter to say "-r"
[19:19] <jameson_> and then you can pass in the requirement.txt as the package name
In another word,s, in your recipe you can write something like this:

python_pip "/var/lib/natrium/requirements.txt" do
  virtualenv "/path/to/virtual-env/"
  action :install
end
 
I was about to write a new resource in ruby like this:

File.open('requirements.txt').each(spe="\n") do |line|
        if line.start_with?("#")
                print "i got a bad one \n"
                print "#{line}"
                print "\n continue \n"
        else
                # line.index("#") != 0
                print "#{line}"
        #if line.index("#") != 0
                print "#{line}"
        end
end

This snippet above is just to show you how you do readlines in ruby, for given content as the one shown below:
Django==1.2.1
#Pinax==0.9a1
PyYAML==3.10
Pygments==1.5
certifi==0.0.8
chardet==1.0.1
decorator==3.3.2

What the actual cookbook will do is this:

---- Begin output of /var/lib/shared/graphyte-virtualenv/natrium-env/bin/pip install -r /var/lib/natrium/requirements.txt ----

The whole Python cookbook is very useful. It can create virtualenv as specified.

Tuesday, August 21, 2012

Why blogging again? I am too old for this.

I touched HTML the first time when I was about eight, nine years old in Hong Kong. I am twenty-one now, and I can you that I suck at front-end development. I can barely write any good Javascript function without constantly looking up on Mozilla Developer Network, Stackoverflow or basically, everything from Google. I can't make any HTML layout with HTML or CSS because I just don't have the fundamental skill baked into my head. I tried, but I found them boring. I still do, although I love the enticement when I see that beautiful webpage I created.


Nonetheless, here I am, in my fourth year of computer science degree, I am still picking up so many tools and skills that are considered fundamentals by computer science education standard. The first time I started using version control was when I started working in my lab.

That's about 8 months ago.

I have to admit that in the last eight months I've learned so much that no one was able to teach me in any form. Ask me how to deal with a legacy system and fix bugs as seen? Ask me how to debug a system? Ask me how to become a duck-tape programmer rather than a perfect all-rounded programmer? How to scalable? Understand why testing is important? You have to pick up these things from doing a real project that isn't done under your own blanket.

Recently I watched Google I/O - The Myth of the Genius Programmer and these two Googlers were so damn right about "real genius" are very rare. The point is a good working environment is crucial to success. One interesting part was "don't hide your commits". It's so true that hiding commits are indication of perfections. The only reason why I have to erase the history if my lab work ever get open-sourced is to remove hardcoded credential.

The talk was really funny. I love it. It taught me, or I should say, remind me of who I should become. I admit that I don't like people entering my comfort zone and tell me "you are wrong, stop, stop, stop, you know nothing about X,Y,Z." That hurts because I am so desperate to become a good and skillfull programmer. I want my presentation to have credibility. It is so sad to see "oh wow, it's August and I haven't done much."

Who to blame? What to blame? Me? The new people who came to the lab? Or all the problems I ran into?

Let's not blame. Let's just say I am still learning, and I really need to manage my time better. I mean sleep early and really get to work more, because with school I am going to have less free time to work on everything at once.

Oh by the way, why should I be blogging? I am too old. Because I want to document what I am learning. I always want to make a blog about it, but I always feel so tired to update blog.... let's hope I won't give up, for the last time.